mirror of https://github.com/dnomd343/klotski.git
Dnomd343
3 months ago
1 changed files with 176 additions and 0 deletions
@ -0,0 +1,176 @@ |
|||
#!/usr/bin/env python3 |
|||
|
|||
import igraph as ig |
|||
|
|||
|
|||
def split_neighbor_layer(g: ig.Graph, n: int) -> tuple[list[set[ig.Vertex]], list[set[ig.Vertex]]]: |
|||
layer_a: set[ig.Vertex] = set(g.vs.select(step=n)) |
|||
layer_b: set[ig.Vertex] = set(g.vs.select(step=n+1)) |
|||
|
|||
layer_a_data: list[set[ig.Vertex]] = [] |
|||
layer_b_data: list[set[ig.Vertex]] = [] |
|||
|
|||
select_next = lambda p: set(x for x in p.neighbors() if x['step'] == n + 1) |
|||
select_last = lambda p: set(x for x in p.neighbors() if x['step'] == n) |
|||
|
|||
while len(layer_a) != 0: |
|||
side_a: set[ig.Vertex] = set() |
|||
side_b: set[ig.Vertex] = set() |
|||
|
|||
point = layer_a.pop() |
|||
side_a.add(point) |
|||
|
|||
while True: |
|||
nexts = set() |
|||
for x in side_a: |
|||
nexts.update(select_next(x)) |
|||
if nexts == side_b: |
|||
break |
|||
side_b.update(nexts) |
|||
for x in nexts: |
|||
if x in layer_b: |
|||
layer_b.remove(x) |
|||
|
|||
lasts = set() |
|||
for x in side_b: |
|||
lasts.update(select_last(x)) |
|||
if lasts == side_a: |
|||
break |
|||
side_a.update(lasts) |
|||
for x in lasts: |
|||
if x in layer_a: |
|||
layer_a.remove(x) |
|||
|
|||
layer_a_data.append(side_a) |
|||
layer_b_data.append(side_b) |
|||
|
|||
assert len(layer_b) == 0 |
|||
return layer_a_data, layer_b_data |
|||
|
|||
|
|||
def combine_split_result(data_a: list[set[ig.Vertex]], data_b: list[set[ig.Vertex]]) -> list[set[ig.Vertex]]: |
|||
result = [] |
|||
|
|||
while True: |
|||
data_a = list(filter(None, data_a)) |
|||
data_b = list(filter(None, data_b)) |
|||
data_a = sorted(data_a, key=lambda x: len(x)) |
|||
data_b = sorted(data_b, key=lambda x: len(x)) |
|||
|
|||
if len(data_a) == 0: |
|||
assert len(data_b) == 0 |
|||
break |
|||
|
|||
# print(len(data_a)) |
|||
# for x in data_a: |
|||
# print([y.attributes() for y in x]) |
|||
# print(len(data_b)) |
|||
# for x in data_b: |
|||
# print([y.attributes() for y in x]) |
|||
# print('-' * 64) |
|||
|
|||
if len(data_a[0]) <= len(data_b[0]): |
|||
union = data_a[0] |
|||
peer_unions = data_b |
|||
else: |
|||
union = data_b[0] |
|||
peer_unions = data_a |
|||
|
|||
for peer_union in peer_unions: |
|||
mid = union & peer_union |
|||
if len(mid) == 0: |
|||
continue |
|||
result.append(mid) |
|||
for x in mid: |
|||
union.remove(x) |
|||
peer_union.remove(x) |
|||
|
|||
return result |
|||
|
|||
|
|||
def split_layers(g: ig.Graph) -> list[list[set[ig.Vertex]]]: |
|||
assert min(g.vs['step']) == 0 |
|||
layer_num = max(g.vs['step']) + 1 |
|||
print(layer_num) |
|||
|
|||
layers = [{'up': [], 'down': []} for x in range(layer_num)] |
|||
layers[0]['up'] = [set(g.vs.select(step=0))] |
|||
layers[-1]['down'] = [set(g.vs.select(step=layer_num-1))] |
|||
|
|||
for layer_num in range(layer_num - 1): |
|||
data_a, data_b = split_neighbor_layer(g, layer_num) |
|||
layers[layer_num]['down'] = data_a |
|||
layers[layer_num + 1]['up'] = list(filter(None, data_b)) |
|||
|
|||
for layer in layers: |
|||
assert set() not in layer['up'] |
|||
assert set() not in layer['down'] |
|||
up = set() |
|||
[up.update(x) for x in layer['up']] |
|||
down = set() |
|||
[down.update(x) for x in layer['down']] |
|||
assert up == down |
|||
|
|||
result = [] |
|||
for layer in layers: |
|||
result.append(combine_split_result(layer['up'], layer['down'])) |
|||
|
|||
for layer_num, layer in enumerate(result): |
|||
assert set() not in layer |
|||
layer_nodes = set() |
|||
[layer_nodes.update(x) for x in layer] |
|||
assert layer_nodes == set(g.vs.select(step=layer_num)) |
|||
return result |
|||
|
|||
|
|||
def export_new_graph(g: ig.Graph, split_data: list[list[set[ig.Vertex]]]) -> ig.Graph: |
|||
ng = ig.Graph(sum([len(x) for x in split_data])) |
|||
|
|||
g_index = 0 |
|||
index_map = {} |
|||
for layer_index, unions in enumerate(split_data): |
|||
for union_index, nodes in enumerate(unions): |
|||
index_map[(layer_index, union_index)] = g_index |
|||
ng.vs[g_index]['step'] = layer_index |
|||
# ng.vs[g_index]['codes'] = [x['code'] for x in nodes] |
|||
g_index += 1 |
|||
|
|||
for layer_index in range(len(split_data)-1): |
|||
curr_layer = split_data[layer_index] |
|||
next_layer = split_data[layer_index+1] |
|||
|
|||
def union_neighbors(curr_union_index: int) -> list[int]: |
|||
next_union_indexes = set() |
|||
for node in curr_layer[curr_union_index]: |
|||
next_nodes = [x for x in node.neighbors() if x['step'] == layer_index+1] |
|||
for next_node in next_nodes: |
|||
for next_union_index in range(len(next_layer)): |
|||
if next_node in next_layer[next_union_index]: |
|||
next_union_indexes.add(next_union_index) |
|||
return sorted(next_union_indexes) |
|||
|
|||
for union_index in range(len(curr_layer)): |
|||
union_a = index_map[(layer_index, union_index)] |
|||
edges = [(union_a, index_map[(layer_index+1, x)]) for x in union_neighbors(union_index)] |
|||
ng.add_edges(edges) |
|||
|
|||
return ng |
|||
|
|||
|
|||
if __name__ == '__main__': |
|||
raw = ig.Graph.Read_Pickle('data/DAA7F30.pkl') |
|||
# raw = ig.Graph.Read_Pickle('data/DBAB4CC.pkl') |
|||
# raw = ig.Graph.Read_Pickle('main_combined.pkl') |
|||
print(raw.summary()) |
|||
gg = export_new_graph(raw, split_layers(raw)) |
|||
print(gg.summary()) |
|||
# print(gg.isomorphic(raw)) |
|||
|
|||
# for x in gg.vs: |
|||
# x['color'] = 'yellow' |
|||
# gg.vs[0]['color'] = 'red' |
|||
# print(gg) |
|||
# ig.plot(gg, 'demo.png', vertex_size=10) |
|||
|
|||
# gg.write_pickle('main_combined.pkl') |
|||
# gg.write_graphml('main_combined.graphml') |
Loading…
Reference in new issue