from sklearn.feature_extraction.text import TfidfVectorizer, CountVectorizer from sklearn.linear_model import LogisticRegression import quapy as qp from data import LabelledCollection import numpy as np from laboratory.custom_vectorizers import * from protocol import APP from quapy.method.aggregative import _get_divergence, HDy, DistributionMatching from quapy.method.base import BaseQuantifier from scipy import optimize import pandas as pd # TODO: explore the bernoulli (term presence/absence) variant # TODO: explore the multinomial (term frequency) variant # TODO: explore the multinomial + length normalization variant # TODO: consolidate the TSR-variant (e.g., using information gain) variant; # - works better with the idf? # - works better with length normalization? # - etc class DxS(BaseQuantifier): def __init__(self, vectorizer=None, divergence='topsoe'): self.vectorizer = vectorizer self.divergence = divergence # def __as_distribution(self, instances): # return np.asarray(instances.sum(axis=0) / instances.sum()).flatten() def __as_distribution(self, instances): dist = instances.sum(axis=0) / instances.sum() return np.asarray(dist).flatten() def fit(self, data: LabelledCollection): text_instances, labels = data.Xy if self.vectorizer is not None: text_instances = self.vectorizer.fit_transform(text_instances, y=labels) distributions = [] for class_i in data.classes_: distributions.append(self.__as_distribution(text_instances[labels == class_i])) self.validation_distribution = np.asarray(distributions) return self def quantify(self, text_instances): if self.vectorizer is not None: text_instances = self.vectorizer.transform(text_instances) test_distribution = self.__as_distribution(text_instances) divergence = _get_divergence(self.divergence) n_classes, n_feats = self.validation_distribution.shape def match(prev): prev = np.expand_dims(prev, axis=0) mixture_distribution = (prev @ self.validation_distribution).flatten() return divergence(test_distribution, mixture_distribution) # the initial point is set as the uniform distribution uniform_distribution = np.full(fill_value=1 / n_classes, shape=(n_classes,)) # solutions are bounded to those contained in the unit-simplex bounds = tuple((0, 1) for x in range(n_classes)) # values in [0,1] constraints = ({'type': 'eq', 'fun': lambda x: 1 - sum(x)}) # values summing up to 1 r = optimize.minimize(match, x0=uniform_distribution, method='SLSQP', bounds=bounds, constraints=constraints) return r.x if __name__ == '__main__': qp.environ['SAMPLE_SIZE'] = 250 qp.environ['N_JOBS'] = -1 min_df = 10 # dataset = 'imdb' repeats = 10 error = 'mae' div = 'topsoe' # generates tuples (dataset, method, method_name) # (the dataset is needed for methods that process the dataset differently) def gen_methods(): for dataset in qp.datasets.REVIEWS_SENTIMENT_DATASETS: data = qp.datasets.fetch_reviews(dataset, tfidf=False) bernoulli_vectorizer = CountVectorizer(min_df=min_df, binary=True) dxs = DxS(divergence=div, vectorizer=bernoulli_vectorizer) yield data, dxs, 'DxS-Bernoulli' multinomial_vectorizer = CountVectorizer(min_df=min_df, binary=False) dxs = DxS(divergence=div, vectorizer=multinomial_vectorizer) yield data, dxs, 'DxS-multinomial' tf_vectorizer = TfidfVectorizer(sublinear_tf=False, use_idf=False, min_df=min_df, norm=None) dxs = DxS(divergence=div, vectorizer=tf_vectorizer) yield data, dxs, 'DxS-TF' logtf_vectorizer = TfidfVectorizer(sublinear_tf=True, use_idf=False, min_df=min_df, norm=None) dxs = DxS(divergence=div, vectorizer=logtf_vectorizer) yield data, dxs, 'DxS-logTF' tfidf_vectorizer = TfidfVectorizer(use_idf=True, min_df=min_df, norm=None) dxs = DxS(divergence=div, vectorizer=tfidf_vectorizer) yield data, dxs, 'DxS-TFIDF' tfidf_vectorizer = TfidfVectorizer(use_idf=True, min_df=min_df, norm='l2') dxs = DxS(divergence=div, vectorizer=tfidf_vectorizer) yield data, dxs, 'DxS-TFIDF-l2' tsr_vectorizer = TSRweighting(tsr_function=information_gain, min_df=min_df, norm='l2') dxs = DxS(divergence=div, vectorizer=tsr_vectorizer) yield data, dxs, 'DxS-TFTSR-l2' data = qp.datasets.fetch_reviews(dataset, tfidf=True, min_df=min_df) hdy = HDy(LogisticRegression()) yield data, hdy, 'HDy' dm = DistributionMatching(LogisticRegression(), divergence=div, nbins=5) yield data, dm, 'DM-5b' dm = DistributionMatching(LogisticRegression(), divergence=div, nbins=10) yield data, dm, 'DM-10b' result_path = 'results.csv' with open(result_path, 'wt') as csv: csv.write(f'Method\tDataset\tMAE\tMRAE\n') for data, quantifier, quant_name in gen_methods(): quantifier.fit(data.training) report = qp.evaluation.evaluation_report(quantifier, APP(data.test, repeats=repeats), error_metrics=['mae','mrae'], verbose=True) means = report.mean() csv.write(f'{quant_name}\t{data.name}\t{means["mae"]:.5f}\t{means["mrae"]:.5f}\n') df = pd.read_csv(result_path, sep='\t') # print(df) pv = df.pivot_table(index='Method', columns="Dataset", values=["MAE", "MRAE"]) print(pv)