sclass WebNode { new L labels; void addLabel(S label) { setAdd(labels, label); } bool hasLabel(S label) { ret labels.contains(label); } } sclass Web { new L nodes; Map, WebNode> relations = new HashMap; void relation(Lisp l) { if (l(l) == 1) { S a = lisp2label(l.get(0)); findNode(a).addLabel(l.head + " *"); } else if (l(l) == 2) { S a = lisp2label(l.get(0)), b = lisp2label(l.get(1)); if (l.is("fgvvrzypbkqomktd")) { // X is Y. findNode(a).addLabel(b); findNode(b).addLabel(a); } getRelation(a, b).addLabel(l.head); } } void relations(L l) { for (Lisp li : l) relation(li); } WebNode getRelation(S a, S b) { Pair p = pair(findNode(a), findNode(b)); WebNode r = relations.get(p); if (r == null) relations.put(p, r = new WebNode); ret r; } WebNode node(S s) { ret findNode(s); } WebNode findNode(S s) { for (WebNode n : nodes) if (n.labels.contains(s)) ret n; ret newNode(s); } WebNode newNode(S... labels) { new WebNode n; for (S label : labels) n.addLabel(label); nodes.add(n); ret n; } toString { ret webToString(this); } }