forked from jappeace/distrowatch1graph1svg
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgraph.py
117 lines (101 loc) · 4.13 KB
/
graph.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
# This program tries to parse distrowatch and create a svg graph simliar to: <https://en.wikipedia.org/wiki/Linux_distribution#/media/File:Linux_Distribution_Timeline_with_Android.svg>
# Copyright (C) 2016 Jappe Klooster
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
# GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program.If not, see <http://www.gnu.org/licenses/>.
"""
This file makes a graph from the data structure
this is mostly usefull to make sure the structure of the collected data is
sound, We expect a tree (mostly, maybe some code sharings but that's not
implemented now)
"""
import json
import strings
from correct import fixrelations, correct
def to_graph(son):
"""From flat json to python graph"""
def dumps(item):
return json.dumps(item, indent=4)
def printjson(item):
print(dumps(item))
categories = json.loads(son)
categories = [correct(fixrelations(i)) for i in categories]
independents = filter(
lambda x: x[strings.based] == strings.independend,
categories
)
def listToDict(keyFunction, values):
return dict((keyFunction(v), v) for v in values)
# A list is just a great way to waste time for this usecase
independents = listToDict(lambda x: x["Name"], independents)
def addChildTo(parent, child):
parent[strings.children].append(child)
return parent
# Recursivly find the parents. True on succes, False on failure
# If succesfull the child will be added to the found parents.
def findparents(child, bases, parents):
if len(bases) == 0:
for p in parents:
addChildTo(p, child)
return True
current = bases[0]
if len(parents) == 0:
try:
parents.insert(0, independents[current])
except KeyError as e:
printjson(child)
raise e
return findparents(child, bases[1:], parents)
base = next(
(x
for x
in parents[0][strings.children]
if x[strings.name] == current),
None
)
if base is None:
if current not in independents:
if current == child[strings.name]:
# ubuntu dependson ubuntu... yes distrowatch that's just
# silly
return findparents(child, [], parents)
# the base is not added yet to the structure
# lets just ignore this one for now.
return False
parents.insert(0, independents[current])
return findparents(child, bases[1:], parents)
parents[0] = base
return findparents(child, bases[1:], parents)
def deepen(collection):
counter = 0
while len(collection) > 0:
current = collection[0]
basedstr = current[strings.based]
bases = basedstr.split(",")
if not findparents(current, bases, []):
counter += 1
if counter > len(collection) * 10:
printjson(list(collection))
raise Exception(
"Made five full circles in the deque, the data is " +
"just invalid, deque size %i " % counter
)
collection.append(current)
else:
counter = 0
collection.popleft()
return collection
from collections import deque
notindependents = deque(filter(
lambda x: not x[strings.based] == strings.independend,
categories
))
deepen(notindependents)
return dumps(list(map(lambda item: item[1], independents.items())))