-
Notifications
You must be signed in to change notification settings - Fork 0
/
syncalc.py
executable file
·314 lines (227 loc) · 8.93 KB
/
syncalc.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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
from itertools import chain, combinations, product
from pprint import pprint
import pickle
import csv
########################
# Simple hierarchies #
########################
number_hrcs = {'sp': {('s', 'p')},
'ps': {('p', 's')}}
person_hrcs = {'123': {('1', '2'), ('2', '3'), ('1', '3')},
'12|3': {('1', '2'), ('1', '3')},
'1|23': {('1', '3'), ('2', '3')}}
########################
# Build Crossproduct #
########################
def extract_elements(hierarchy: set) -> set:
return set([element for ranking in hierarchy for element in ranking])
def add_ranking(node1: str, node2: str, person: set, number: set) -> tuple:
person1, number1 = node1
person2, number2 = node2
if node1 != node2 and\
(person1 == person2 or (person1, person2) in person) and\
(number1 == number2 or (number1, number2) in number):
return (node1, node2)
def crossalgebra(person: set, number: set) -> set:
persons = extract_elements(person)
numbers = extract_elements(number)
nodes = set([person+number
for person in persons
for number in numbers])
output = set(add_ranking(node, other_node, person, number)
for node in nodes
for other_node in nodes)
output.remove(None)
return output
######################################
# Compute Algebras with Added Arcs #
######################################
def reachability_closure(algebra: set) -> set:
new_algebra = algebra.copy()
for ranking in algebra:
for other_ranking in algebra:
if ranking[1] == other_ranking[0] and\
ranking[0] != other_ranking[1]:
new_algebra.add((ranking[0], other_ranking[1]))
if new_algebra == algebra:
return new_algebra
else:
return reachability_closure(new_algebra)
def algebra_expansions(algebra: set, possible_additions: iter) -> iter:
for additions in possible_additions:
# print("adding", additions)
yield algebra.copy().union(additions)
def arc_addition(algebra: set, person: set, number: set) -> set:
# compute elements
persons = extract_elements(person)
numbers = extract_elements(number)
# set of all arcs
nodes = set(p+n for p in persons for n in numbers)
all_arcs = set((n1, n2) for n1 in nodes for n2 in nodes
if n1 != n2)
# remove arcs that are already in the algebra
all_arcs -= algebra
# the powerset of all remaining arcs,
# i.e. the set of all possible algebra extensions;
# this is an iterator, so memory usage should be manageable
possible_additions = chain.from_iterable(combinations(all_arcs, r)
for r in range(len(all_arcs)+1))
# compute iterator of all algebras with 0 or more added arcs
algebras = algebra_expansions(algebra, possible_additions)
# close each algebra under reachability
closed_algebras = set(frozenset(reachability_closure(algebra))
for algebra in algebras)
# return set of all remaining algebras
return closed_algebras
######################################
# Compute Syncretisms from Algebra #
######################################
def algebra_interpretation(algebra: set) -> tuple:
presentation = ['1s', '2s', '3s', '1p', '2p', '3p']
syncretism = [0]
for cell in presentation[1:]:
threshold = len(syncretism)
for pos in range(threshold):
other_cell = presentation[pos]
if (other_cell, cell) in algebra and\
(cell, other_cell) in algebra:
value = syncretism[pos]
syncretism.append(value)
break
if len(syncretism) == threshold:
syncretism.append(max(syncretism)+1)
return tuple(syncretism)
def number_to_letter(syncretism: tuple) -> tuple:
return tuple([chr(65 + number) for number in syncretism])
def syn_patterns(algebras: set) -> set:
return set(number_to_letter(algebra_interpretation(algebra))
for algebra in algebras)
def all_syncretisms() -> dict:
syncretisms = {}
for num_label, num_hier in number_hrcs.items():
for pers_label, pers_hier in person_hrcs.items():
algebra = crossalgebra(pers_hier, num_hier)
new_algebras = arc_addition(algebra, pers_hier, num_hier)
syncretisms[(pers_label, num_label)] = \
syn_patterns(new_algebras)
syncretisms['total'] = set().union(*syncretisms.values())
return syncretisms
try:
syncretisms = pickle.load(open("syncretisms.p", "rb"))
except:
syncretisms = all_syncretisms()
pickle.dump(syncretisms, open("syncretisms.p", "wb"))
##################
# Analyze Data #
##################
def read_data(filename: str='data.csv') -> list:
patterns = []
with open('data.csv') as datafile:
data = csv.reader(datafile)
for row in data:
patterns.append(tuple(row[2:8]))
return patterns
def overgeneration(algebra: set, data: list) -> list:
return [pattern for pattern in algebra if pattern not in data]
def undergeneration(algebra: set, data: list) -> list:
return [pattern for pattern in data if pattern not in algebra]
def all_generation(function=overgeneration) -> dict:
data = list(set(read_data()))
return {key: set(function(syncretisms[key], data))
for key in syncretisms.keys()}
def show_generation(function=overgeneration):
generated = all_generation(function=function)
for key, val in generated.items():
print(key, ":", len(val))
pprint(sorted(list(val)))
print("--------------")
def all_overgeneration() -> dict:
return all_generation(function=overgeneration)
def all_undergeneration() -> dict:
return all_generation(function=undergeneration)
def show_overgeneration():
return show_generation(function=overgeneration)
def show_undergeneration():
return show_generation(function=undergeneration)
def show_generators(algebras: dict, patterns: list) -> dict:
overview = {}
for pattern in patterns:
overview[pattern] = set()
for algebra, generated in algebras.items():
if pattern in generated and algebra != 'total':
overview[pattern].add(algebra)
return overview
#############################
# Monotonicity Calculator #
#############################
base_algebras = {(plab, nlab): reachability_closure(crossalgebra(p, n))
for plab, p in person_hrcs.items()
for nlab, n in number_hrcs.items()}
def relabel_algebra(algebra, syncretism):
presentation = ['1s', '2s', '3s', '1p', '2p', '3p']
labeling = dict(zip(presentation, syncretism))
return {(labeling[a], labeling[b])
for (a, b) in algebra}
def test_monotonicity(algebra, syncretism):
algebra = relabel_algebra(algebra, syncretism)
for a, b in algebra:
if a != b:
for c, d in algebra:
if (a, b) == (d, c):
return False
return True
def is_monotonic(algebras, syncretism):
return [key for key, val in algebras.items()
if test_monotonicity(val, syncretism)]
def show_monotonicity(algebras=base_algebras, data=read_data()):
return {syncretism: is_monotonic(algebras, syncretism)
for syncretism in data}
def all_patterns(cells=6):
letters = [[chr(65 + number) for number in range(cells)]
for _ in range(cells)]
return [i for i in product(*letters)]
try:
patterns = pickle.load(open("patterns.p", "rb"))
except:
patterns = all_patterns()
pickle.dump(patterns, open("patterns.p", "wb"))
def all_monotonicity():
return show_monotonicity(base_algebras, patterns)
def letter_to_number(xs):
if not xs:
return xs
ys = [1]
for pos in range(1, len(xs)):
x = xs[pos]
for left in range(pos):
if x == xs[left]:
ys.append(ys[left])
break
else:
ys.append(max(ys) + 1)
return ys
def isomorphic(xs, ys):
return letter_to_number(xs) == letter_to_number(ys)
def monotonicity_overgeneration():
all_monotonic = {key: val
for key, val in all_monotonicity().items()
if val}
unique_monotonic = {}
for key, val in all_monotonic.items():
for key2, val2 in unique_monotonic.items():
if isomorphic(key, key2):
break
else:
unique_monotonic[key] = val
attested_monotonic = show_monotonicity()
return {key: val
for key, val in unique_monotonic.items()
for key2 in attested_monotonic
if not isomorphic(key, key2)}
try:
monotonic_overgen = pickle.load(open("monotonic_overgen.p", "rb"))
except:
monotonic_overgen = monotonicity_overgeneration()
pickle.dump(monotonic_overgen, open("monotonic_overgen.p", "wb"))