ai-privacy-toolkit/tests/test_anonymizer.py

102 lines
4.7 KiB
Python
Raw Normal View History

2021-04-28 14:00:19 +03:00
import pytest
import numpy as np
from sklearn.tree import DecisionTreeClassifier, DecisionTreeRegressor
2021-04-28 14:00:19 +03:00
from sklearn.preprocessing import OneHotEncoder
from apt.anonymization import Anonymize
from apt.utils.dataset_utils import get_iris_dataset, get_adult_dataset, get_nursery_dataset
from sklearn.datasets import load_diabetes
from sklearn.model_selection import train_test_split
from apt.utils.datasets import ArrayDataset, Data
2021-04-28 14:00:19 +03:00
def test_anonymize_ndarray_iris():
2022-03-01 02:28:41 +02:00
dataset = get_iris_dataset()
2021-04-28 14:00:19 +03:00
model = DecisionTreeClassifier()
2022-03-01 02:28:41 +02:00
model.fit(dataset.get_train_samples(), dataset.get_train_labels())
pred = model.predict(dataset.get_train_samples())
2021-04-28 14:00:19 +03:00
k = 10
QI = [0, 2]
anonymizer = Anonymize(k, QI)
anon = anonymizer.anonymize(ArrayDataset(dataset.get_train_samples(), pred))
2022-03-01 02:28:41 +02:00
assert(len(np.unique(anon[:, QI], axis=0)) < len(np.unique(dataset.get_train_samples()[:, QI], axis=0)))
2021-04-28 14:00:19 +03:00
_, counts_elements = np.unique(anon[:, QI], return_counts=True)
assert (np.min(counts_elements) >= k)
2022-03-01 02:28:41 +02:00
assert ((np.delete(anon, QI, axis=1) == np.delete(dataset.get_train_samples(), QI, axis=1)).all())
2021-04-28 14:00:19 +03:00
def test_anonymize_pandas_adult():
2022-03-01 02:28:41 +02:00
dataset = get_adult_dataset()
encoded = OneHotEncoder().fit_transform(dataset.get_train_samples())
2021-04-28 14:00:19 +03:00
model = DecisionTreeClassifier()
2022-03-01 02:28:41 +02:00
model.fit(encoded, dataset.get_train_labels())
2021-04-28 14:00:19 +03:00
pred = model.predict(encoded)
k = 100
QI = ['age', 'workclass', 'education-num', 'marital-status', 'occupation', 'relationship', 'race', 'sex',
'native-country']
categorical_features = ['workclass', 'marital-status', 'occupation', 'relationship', 'race', 'sex',
'native-country']
anonymizer = Anonymize(k, QI, categorical_features=categorical_features)
anon = anonymizer.anonymize(ArrayDataset(dataset.get_train_samples(), pred))
2021-04-28 14:00:19 +03:00
2022-03-01 02:28:41 +02:00
assert(anon.loc[:, QI].drop_duplicates().shape[0] < dataset.get_train_samples().loc[:, QI].drop_duplicates().shape[0])
2021-04-28 14:00:19 +03:00
assert (anon.loc[:, QI].value_counts().min() >= k)
2022-03-01 02:28:41 +02:00
assert (anon.drop(QI, axis=1).equals(dataset.get_train_samples().drop(QI, axis=1)))
2021-04-28 14:00:19 +03:00
def test_anonymize_pandas_nursery():
2022-03-01 02:28:41 +02:00
dataset = get_nursery_dataset()
encoded = OneHotEncoder().fit_transform(dataset.get_train_samples())
2021-04-28 14:00:19 +03:00
model = DecisionTreeClassifier()
2022-03-01 02:28:41 +02:00
model.fit(encoded, dataset.get_train_labels())
2021-04-28 14:00:19 +03:00
pred = model.predict(encoded)
k = 100
QI = ["finance", "social", "health"]
categorical_features = ["parents", "has_nurs", "form", "housing", "finance", "social", "health", 'children']
anonymizer = Anonymize(k, QI, categorical_features=categorical_features)
anon = anonymizer.anonymize(ArrayDataset(dataset.get_train_samples(), pred))
2021-04-28 14:00:19 +03:00
2022-03-01 02:28:41 +02:00
assert(anon.loc[:, QI].drop_duplicates().shape[0] < dataset.get_train_samples().loc[:, QI].drop_duplicates().shape[0])
2021-04-28 14:00:19 +03:00
assert (anon.loc[:, QI].value_counts().min() >= k)
2022-03-01 02:28:41 +02:00
assert (anon.drop(QI, axis=1).equals(dataset.get_train_samples().drop(QI, axis=1)))
2021-04-28 14:00:19 +03:00
def test_regression():
2022-03-01 02:28:41 +02:00
x_train, x_test, y_train, y_test = train_test_split(load_diabetes().data, load_diabetes().target, test_size=0.5, random_state=14)
train_dataset = ArrayDataset(x_train, y_train)
test_dataset = ArrayDataset(x_test, y_test)
2022-03-01 02:28:41 +02:00
dataset = Data(train_dataset, test_dataset)
model = DecisionTreeRegressor(random_state=10, min_samples_split=2)
2022-03-01 02:28:41 +02:00
model.fit(dataset.get_train_samples(), dataset.get_train_labels())
pred = model.predict(dataset.get_train_samples())
k = 10
QI = [0, 2, 5, 8]
anonymizer = Anonymize(k, QI, is_regression=True)
anon = anonymizer.anonymize(ArrayDataset(dataset.get_train_samples(), pred))
2022-03-01 02:28:41 +02:00
print('Base model accuracy (R2 score): ', model.score(dataset.get_test_samples(), dataset.get_test_labels()))
model.fit(anon, dataset.get_train_labels())
print('Base model accuracy (R2 score) after anonymization: ', model.score(dataset.get_test_samples(), dataset.get_test_labels()))
assert(len(np.unique(anon[:, QI], axis=0)) < len(np.unique(dataset.get_train_samples()[:, QI], axis=0)))
_, counts_elements = np.unique(anon[:, QI], return_counts=True)
assert (np.min(counts_elements) >= k)
2022-03-01 02:28:41 +02:00
assert ((np.delete(anon, QI, axis=1) == np.delete(dataset.get_train_samples(), QI, axis=1)).all())
2021-04-28 14:00:19 +03:00
def test_errors():
with pytest.raises(ValueError):
Anonymize(1, [0, 2])
with pytest.raises(ValueError):
Anonymize(2, [])
with pytest.raises(ValueError):
Anonymize(2, None)
anonymizer = Anonymize(10, [0, 2])
2022-03-01 02:28:41 +02:00
dataset = get_iris_dataset()
2021-04-28 14:00:19 +03:00
with pytest.raises(ValueError):
anonymizer.anonymize(ArrayDataset(dataset.get_train_samples(), dataset.get_test_labels()))
2022-03-01 02:28:41 +02:00
dataset = get_adult_dataset()
2021-04-28 14:00:19 +03:00
with pytest.raises(ValueError):
anonymizer.anonymize(ArrayDataset(dataset.get_train_samples(), dataset.get_train_labels()))