2017-07-11 00:17:18 +08:00
|
|
|
"""
|
|
|
|
Generates a random graph with some configurable statistics.
|
|
|
|
"""
|
|
|
|
|
2017-11-15 21:26:19 +08:00
|
|
|
from random import randint, seed
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
|
2017-11-15 21:26:19 +08:00
|
|
|
seed(0)
|
|
|
|
|
2017-07-11 00:17:18 +08:00
|
|
|
def rint(upper_bound_exclusive):
|
|
|
|
return randint(0, upper_bound_exclusive - 1)
|
|
|
|
|
2017-09-05 19:54:28 +08:00
|
|
|
VERTEX_COUNT = 1500
|
|
|
|
EDGE_COUNT = VERTEX_COUNT * 15
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
# numbers of *different* labels, edge types and properties
|
|
|
|
LABEL_COUNT = 10
|
|
|
|
|
2017-09-05 19:54:28 +08:00
|
|
|
MAX_LABELS = 5 # maximum number of labels in a vertex
|
2017-07-11 00:17:18 +08:00
|
|
|
MAX_PROPS = 4 # maximum number of properties in a vertex/edge
|
2017-09-05 19:54:28 +08:00
|
|
|
MAX_PROP_VALUE = 1000
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
# some consts used in mutiple files
|
2017-09-05 19:54:28 +08:00
|
|
|
LABEL_INDEX = "LabelIndex"
|
2017-07-11 00:17:18 +08:00
|
|
|
LABEL_PREFIX = "Label"
|
2017-09-05 19:54:28 +08:00
|
|
|
PROP_PREFIX = "Prop"
|
2017-07-11 00:17:18 +08:00
|
|
|
ID = "id"
|
|
|
|
|
|
|
|
|
2017-09-05 19:54:28 +08:00
|
|
|
|
2017-07-11 00:17:18 +08:00
|
|
|
def labels():
|
2017-09-05 19:54:28 +08:00
|
|
|
labels = ":" + LABEL_INDEX
|
|
|
|
for _ in range(rint(MAX_LABELS)):
|
|
|
|
labels += ":" + LABEL_PREFIX + str(rint(LABEL_COUNT))
|
|
|
|
return labels
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
|
|
|
|
def properties(id):
|
|
|
|
""" Generates a properties string with [0, MAX_PROPS) properties.
|
|
|
|
Note that if PropX is generated, then all the PropY where Y < X
|
|
|
|
are generated. Thus most labels have Prop0, and least have PropMAX_PROPS.
|
|
|
|
"""
|
2017-09-05 19:54:28 +08:00
|
|
|
props = {"%s%d" % (PROP_PREFIX, i): rint(MAX_PROP_VALUE)
|
|
|
|
for i in range(rint(MAX_PROPS))}
|
|
|
|
props[ID] = id
|
|
|
|
return "{" + ", ".join("%s: %s" % kv for kv in props.items()) + "}"
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
|
|
|
|
def vertex(vertex_index):
|
|
|
|
return "(%s %s)" % (labels(), properties(vertex_index))
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
2017-09-05 19:54:28 +08:00
|
|
|
# create an index to speed setup up
|
2019-12-05 20:24:30 +08:00
|
|
|
print("CREATE INDEX ON :" + LABEL_INDEX + ";")
|
|
|
|
for i in range(LABEL_COUNT):
|
|
|
|
print("CREATE INDEX ON :" + LABEL_PREFIX + str(i) + ";")
|
2017-09-05 19:54:28 +08:00
|
|
|
print("CREATE INDEX ON :%s(%s);" % (LABEL_INDEX, ID))
|
|
|
|
|
2017-07-11 00:17:18 +08:00
|
|
|
# we batch CREATEs because to speed creation up
|
2017-09-05 19:54:28 +08:00
|
|
|
BATCH_SIZE = 30
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
# create vertices
|
|
|
|
for vertex_index in range(VERTEX_COUNT):
|
|
|
|
print("CREATE %s" % vertex(vertex_index))
|
2017-07-11 20:43:26 +08:00
|
|
|
if (vertex_index != 0 and vertex_index % BATCH_SIZE == 0) or \
|
|
|
|
vertex_index + 1 == VERTEX_COUNT:
|
2017-07-11 00:17:18 +08:00
|
|
|
print(";")
|
2017-09-05 19:54:28 +08:00
|
|
|
print("MATCH (n) RETURN assert(count(n) = %d);" % VERTEX_COUNT)
|
2017-07-11 00:17:18 +08:00
|
|
|
|
2017-09-12 20:29:35 +08:00
|
|
|
# create edges stohastically
|
|
|
|
attempts = VERTEX_COUNT ** 2
|
|
|
|
p = EDGE_COUNT / VERTEX_COUNT ** 2
|
2017-09-05 19:54:28 +08:00
|
|
|
print("MATCH (a) WITH a MATCH (b) WITH a, b WHERE rand() < %f "
|
2017-09-12 20:29:35 +08:00
|
|
|
" CREATE (a)-[:EdgeType]->(b);" % p)
|
|
|
|
sigma = (attempts * p * (1 - p)) ** 0.5
|
|
|
|
delta = 5 * sigma
|
2017-09-05 19:54:28 +08:00
|
|
|
print("MATCH (n)-[r]->() WITH count(r) AS c "
|
|
|
|
"RETURN assert(c >= %d AND c <= %d);" % (
|
2017-09-12 20:29:35 +08:00
|
|
|
EDGE_COUNT - delta, EDGE_COUNT + delta))
|
2017-07-11 00:17:18 +08:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|