forked from AndRossi/Kelpie
-
Notifications
You must be signed in to change notification settings - Fork 0
/
criage.py
120 lines (102 loc) · 6.8 KB
/
criage.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
from typing import Tuple, Any
from dataset import Dataset
from relevance_engines.criage_engine import CriageEngine
from link_prediction.models.model import Model
from prefilters.criage_prefilter import CriagePreFilter
from explanation_builders.criage_necessary_builder import CriageNecessaryExplanationBuilder
from explanation_builders.criage_sufficient_builder import CriageSufficientExplanationBuilder
class Criage:
"""
The Criage object is the overall manager of the Criage explanation process.
It implements the whole explanation pipeline, requesting the suitable operations to the ExplanationEngines
and to the entity_similarity modules.
"""
def __init__(self,
model: Model,
dataset: Dataset,
hyperparameters: dict):
"""
Criage object constructor.
:param model: the model to explain
:param dataset: the dataset used to train the model
:param hyperparameters: the hyperparameters of the model and of its optimization process
"""
self.model = model
self.dataset = dataset
self.hyperparameters = hyperparameters
self.prefilter = CriagePreFilter(model=model,
dataset=dataset)
self.engine = CriageEngine(model=model,
dataset=dataset,
hyperparameters=hyperparameters)
def explain_necessary(self,
sample_to_explain: Tuple[Any, Any, Any],
perspective: str,
num_promising_samples=-1):
"""
This method extracts necessary explanations for a specific sample,
from the perspective of either its head or its tail.
:param sample_to_explain: the sample to explain
:param perspective: a string conveying the perspective of the requested explanations.
It can be either "head" or "tail":
- if "head", Kelpie answers the question
"given the sample head and relation, why is the sample tail predicted as tail?"
- if "tail", Kelpie answers the question
"given the sample relation and tail, why is the sample head predicted as head?"
:param num_promising_samples: the number of samples relevant to the sample to explain
that must be identified and removed from the entity under analysis
to verify whether they worsen the target prediction or not
:return: a list containing for each relevant n-ple extracted, a couple containing
- that relevant n-ple
- its value of relevance
"""
top_promising_samples = self.prefilter.top_promising_samples_for(sample_to_explain=sample_to_explain,
perspective=perspective,
top_k=num_promising_samples)
explanation_builder = CriageNecessaryExplanationBuilder(model=self.model,
dataset=self.dataset,
hyperparameters=self.hyperparameters,
sample_to_explain=sample_to_explain,
perspective=perspective)
rules_with_relevance = explanation_builder.build_explanations(samples_to_remove=top_promising_samples)
return rules_with_relevance
def explain_sufficient(self,
sample_to_explain: Tuple[Any, Any, Any],
perspective: str,
num_promising_samples=50,
num_entities_to_convert=10,
entities_to_convert=None):
"""
This method extracts necessary explanations for a specific sample,
from the perspective of either its head or its tail.
:param sample_to_explain: the sample to explain
:param perspective: a string conveying the perspective of the requested explanations.
It can be either "head" or "tail":
- if "head", Kelpie answers the question
"given the sample head and relation, why is the sample tail predicted as tail?"
- if "tail", Kelpie answers the question
"given the sample relation and tail, why is the sample head predicted as head?"
:param num_promising_samples: the number of samples relevant to the sample to explain
that must be identified and removed from the entity under analysis
to verify whether they worsen the target prediction or not
:param num_entities_to_convert: the number of entities to convert to extract
(if they have to be extracted)
:param entities_to_convert: the entities to convert
(if they are passed instead of having to be extracted)
:return: a list containing for each relevant n-ple extracted, a couple containing
- that relevant n-ple
- its value of relevance
"""
most_promising_samples = self.prefilter.top_promising_samples_for(sample_to_explain=sample_to_explain,
perspective=perspective,
top_k=num_promising_samples)
explanation_builder = CriageSufficientExplanationBuilder(model=self.model,
dataset=self.dataset,
hyperparameters=self.hyperparameters,
sample_to_explain=sample_to_explain,
perspective=perspective,
num_entities_to_convert=num_entities_to_convert,
entities_to_convert=entities_to_convert)
explanations_with_relevance = explanation_builder.build_explanations(samples_to_add=most_promising_samples,
top_k=10)
return explanations_with_relevance, explanation_builder.entities_to_convert