Source code for py2neo.data

#!/usr/bin/env python
# -*- encoding: utf-8 -*-

# Copyright 2011-2021, Nigel Small
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.


from __future__ import absolute_import


__all__ = [
    "Subgraph",
    "Walkable",
    "Entity",
    "Node",
    "Relationship",
    "Path",
    "walk",
    "UniquenessError",
]


from collections import OrderedDict
from itertools import chain
from uuid import uuid4

# noinspection PyUnresolvedReferences
from interchange import geo as spatial
# noinspection PyUnresolvedReferences
from interchange import time
from interchange.collections import SetView, PropertyDict

from py2neo.compat import string_types, ustr, xstr
from py2neo.cypher import cypher_escape, cypher_repr, cypher_join
from py2neo.cypher.encoding import CypherEncoder, LabelSetView
from py2neo.cypher.queries import (
    unwind_create_nodes_query,
    unwind_merge_nodes_query,
    unwind_merge_relationships_query,
)


[docs] class Subgraph(object): """ A :class:`.Subgraph` is an arbitrary collection of nodes and relationships. It is also the base class for :class:`.Node`, :class:`.Relationship` and :class:`.Path`. By definition, a subgraph must contain at least one node; `null subgraphs <http://mathworld.wolfram.com/NullGraph.html>`_ should be represented by :const:`None`. To test for `emptiness <http://mathworld.wolfram.com/EmptyGraph.html>`_ the built-in :func:`bool` function can be used. The simplest way to construct a subgraph is by combining nodes and relationships using standard set operations. For example:: >>> s = ab | ac >>> s {(alice:Person {name:"Alice"}), (bob:Person {name:"Bob"}), (carol:Person {name:"Carol"}), (Alice)-[:KNOWS]->(Bob), (Alice)-[:WORKS_WITH]->(Carol)} >>> s.nodes() frozenset({(alice:Person {name:"Alice"}), (bob:Person {name:"Bob"}), (carol:Person {name:"Carol"})}) >>> s.relationships() frozenset({(Alice)-[:KNOWS]->(Bob), (Alice)-[:WORKS_WITH]->(Carol)}) .. describe:: subgraph | other | ... Union. Return a new subgraph containing all nodes and relationships from *subgraph* as well as all those from *other*. Any entities common to both will only be included once. .. describe:: subgraph & other & ... Intersection. Return a new subgraph containing all nodes and relationships common to both *subgraph* and *other*. .. describe:: subgraph - other - ... Difference. Return a new subgraph containing all nodes and relationships that exist in *subgraph* but do not exist in *other*, as well as all nodes that are connected by the relationships in *subgraph* regardless of whether or not they exist in *other*. .. describe:: subgraph ^ other ^ ... Symmetric difference. Return a new subgraph containing all nodes and relationships that exist in *subgraph* or *other*, but not in both, as well as all nodes that are connected by those relationships regardless of whether or not they are common to *subgraph* and *other*. """ def __init__(self, nodes=None, relationships=None): self.__nodes = frozenset(nodes or []) self.__relationships = frozenset(relationships or []) self.__nodes |= frozenset(chain.from_iterable(r.nodes for r in self.__relationships)) #if not self.__nodes: # raise ValueError("Subgraphs must contain at least one node") def __repr__(self): return "Subgraph({%s}, {%s})" % (", ".join(map(repr, self.nodes)), ", ".join(map(repr, self.relationships))) def __eq__(self, other): try: return self.nodes == other.nodes and self.relationships == other.relationships except (AttributeError, TypeError): return False def __ne__(self, other): return not self.__eq__(other) def __hash__(self): value = 0 for entity in self.__nodes: value ^= hash(entity) for entity in self.__relationships: value ^= hash(entity) return value def __len__(self): return len(self.__relationships) def __iter__(self): return iter(self.__relationships) def __bool__(self): return bool(self.__relationships) def __nonzero__(self): return bool(self.__relationships) def __or__(self, other): return Subgraph(set(self.nodes) | set(other.nodes), set(self.relationships) | set(other.relationships)) def __and__(self, other): return Subgraph(set(self.nodes) & set(other.nodes), set(self.relationships) & set(other.relationships)) def __sub__(self, other): r = set(self.relationships) - set(other.relationships) n = (set(self.nodes) - set(other.nodes)) | set().union(*(set(rel.nodes) for rel in r)) return Subgraph(n, r) def __xor__(self, other): r = set(self.relationships) ^ set(other.relationships) n = (set(self.nodes) ^ set(other.nodes)) | set().union(*(set(rel.nodes) for rel in r)) return Subgraph(n, r) @classmethod def _is_bound(cls, entity, graph): if entity.graph is None: return False elif entity.graph == graph: return True else: raise ValueError("Entity %r is already bound to graph %r" % (entity, graph)) def __db_create__(self, tx): """ Create new data in a remote :class:`.Graph` from this :class:`.Subgraph`. :param tx: """ graph = tx.graph # Convert nodes into a dictionary of # {frozenset(labels): [Node, Node, ...]} node_dict = {} for node in self.nodes: if not self._is_bound(node, tx.graph): key = frozenset(node.labels) node_dict.setdefault(key, []).append(node) # Convert relationships into a dictionary of # {rel_type: [Rel, Rel, ...]} rel_dict = {} for relationship in self.relationships: if not self._is_bound(relationship, tx.graph): key = type(relationship).__name__ rel_dict.setdefault(key, []).append(relationship) for labels, nodes in node_dict.items(): pq = unwind_create_nodes_query(list(map(dict, nodes)), labels=labels) pq = cypher_join(pq, "RETURN id(_)") records = tx.run(*pq) for i, record in enumerate(records): node = nodes[i] node.graph = graph node.identity = record[0] node._remote_labels = labels for r_type, relationships in rel_dict.items(): data = map(lambda r: [r.start_node.identity, dict(r), r.end_node.identity], relationships) pq = unwind_merge_relationships_query(data, r_type) pq = cypher_join(pq, "RETURN id(_)") for i, record in enumerate(tx.run(*pq)): relationship = relationships[i] relationship.graph = graph relationship.identity = record[0] def __db_delete__(self, tx): """ Delete data in a remote :class:`.Graph` based on this :class:`.Subgraph`. :param tx: """ graph = tx.graph node_identities = [] for relationship in self.relationships: if self._is_bound(relationship, graph): relationship.graph = None relationship.identity = None for node in self.nodes: if self._is_bound(node, graph): node_identities.append(node.identity) node.graph = None node.identity = None # TODO: this might delete remote relationships that aren't # represented in the local subgraph - is this OK? list(tx.run("MATCH (_) WHERE id(_) IN $x DETACH DELETE _", x=node_identities)) def __db_exists__(self, tx): """ Determine whether one or more graph entities all exist within the database. Note that if any nodes or relationships in this :class:`.Subgraph` are not bound to remote counterparts, this method will return ``False``. :param tx: :returns: ``True`` if all entities exist remotely, ``False`` otherwise """ graph = tx.graph node_ids = set() relationship_ids = set() for i, node in enumerate(self.nodes): try: if self._is_bound(node, graph): node_ids.add(node.identity) else: return False except ValueError: return False for i, relationship in enumerate(self.relationships): try: if self._is_bound(relationship, graph): relationship_ids.add(relationship.identity) else: return False except ValueError: return False statement = ("OPTIONAL MATCH (a) WHERE id(a) IN $x " "OPTIONAL MATCH ()-[r]->() WHERE id(r) IN $y " "RETURN count(DISTINCT a) + count(DISTINCT r)") parameters = {"x": list(node_ids), "y": list(relationship_ids)} return tx.evaluate(statement, parameters) == len(node_ids) + len(relationship_ids) def __db_merge__(self, tx, primary_label=None, primary_key=None): """ Merge data into a remote :class:`.Graph` from this :class:`.Subgraph`. :param tx: :param primary_label: :param primary_key: """ graph = tx.graph # Convert nodes into a dictionary of # {(p_label, p_key, frozenset(labels)): [Node, Node, ...]} node_dict = {} for node in self.nodes: if not self._is_bound(node, graph): # Determine primary label if node.__primarylabel__ is not None: p_label = node.__primarylabel__ elif node.__model__ is not None: p_label = node.__model__.__primarylabel__ or primary_label else: p_label = primary_label # Determine primary key if node.__primarykey__ is not None: p_key = node.__primarykey__ elif node.__model__ is not None: p_key = node.__model__.__primarykey__ or primary_key else: p_key = primary_key # Add node to the node dictionary key = (p_label, p_key, frozenset(node.labels)) node_dict.setdefault(key, []).append(node) # Convert relationships into a dictionary of # {rel_type: [Rel, Rel, ...]} rel_dict = {} for relationship in self.relationships: if not self._is_bound(relationship, graph): key = type(relationship).__name__ rel_dict.setdefault(key, []).append(relationship) for (pl, pk, labels), nodes in node_dict.items(): if pl is None or pk is None: raise ValueError("Primary label and primary key are required for MERGE operation") pq = unwind_merge_nodes_query(map(dict, nodes), (pl, pk), labels) pq = cypher_join(pq, "RETURN id(_)") identities = [record[0] for record in tx.run(*pq)] if len(identities) > len(nodes): raise UniquenessError("Found %d matching nodes for primary label %r and primary " "key %r with labels %r but merging requires no more than " "one" % (len(identities), pl, pk, set(labels))) for i, identity in enumerate(identities): node = nodes[i] node.graph = graph node.identity = identity node._remote_labels = labels for r_type, relationships in rel_dict.items(): data = map(lambda r: [r.start_node.identity, dict(r), r.end_node.identity], relationships) pq = unwind_merge_relationships_query(data, r_type) pq = cypher_join(pq, "RETURN id(_)") for i, record in enumerate(tx.run(*pq)): relationship = relationships[i] relationship.graph = graph relationship.identity = record[0] def __db_pull__(self, tx): """ Copy data from a remote :class:`.Graph` into this :class:`.Subgraph`. :param tx: """ # Pull nodes nodes = {} for node in self.nodes: if self._is_bound(node, tx.graph): nodes[node.identity] = node query = tx.run("MATCH (_) WHERE id(_) in $x " "RETURN id(_), labels(_), properties(_)", x=list(nodes.keys())) for identity, new_labels, new_properties in query: node = nodes[identity] node.clear_labels() node.update_labels(new_labels) node.clear() node.update(new_properties) # Pull relationships relationships = {} for relationship in self.relationships: if self._is_bound(relationship, tx.graph): relationships[relationship.identity] = relationship query = tx.run("MATCH ()-[_]->() WHERE id(_) in $x " "RETURN id(_), properties(_)", x=list(relationships.keys())) for identity, new_properties in query: relationship = relationships[identity] relationship.clear() relationship.update(new_properties) def __db_push__(self, tx): """ Copy data into a remote :class:`.Graph` from this :class:`.Subgraph`. :param tx: """ for node in self.nodes: if self._is_bound(node, tx.graph): clauses = ["MATCH (_) WHERE id(_) = $x", "SET _ = $y"] parameters = {"x": node.identity, "y": dict(node)} old_labels = node._remote_labels - node._labels if old_labels: clauses.append("REMOVE _:%s" % ":".join(map(cypher_escape, old_labels))) new_labels = node._labels - node._remote_labels if new_labels: clauses.append("SET _:%s" % ":".join(map(cypher_escape, new_labels))) tx.run("\n".join(clauses), parameters) for relationship in self.relationships: if self._is_bound(relationship, tx.graph): clauses = ["MATCH ()-[_]->() WHERE id(_) = $x", "SET _ = $y"] parameters = {"x": relationship.identity, "y": dict(relationship)} tx.run("\n".join(clauses), parameters) def __db_separate__(self, tx): """ Delete relationships in a remote :class:`.Graph` based on those present in this :class:`.Subgraph`. :param tx: :return: """ relationship_identities = [] for relationship in self.relationships: if self._is_bound(relationship, tx.graph): relationship_identities.append(relationship.identity) relationship.graph = None relationship.identity = None list(tx.run("MATCH ()-[_]->() WHERE id(_) IN $x DELETE _", x=relationship_identities)) @property def graph(self): assert self.__nodes # assume there is at least one node return set(self.__nodes).pop().graph @property def nodes(self): """ The set of all nodes in this subgraph. """ return SetView(self.__nodes) @property def relationships(self): """ The set of all relationships in this subgraph. """ return SetView(self.__relationships)
[docs] def labels(self): """ Return the set of all node labels in this subgraph. *Changed in version 2020.0: this is now a method rather than a property, as in previous versions.* """ return frozenset(chain.from_iterable(node.labels for node in self.__nodes))
[docs] def types(self): """ Return the set of all relationship types in this subgraph. """ return frozenset(type(rel).__name__ for rel in self.__relationships)
[docs] def keys(self): """ Return the set of all property keys used by the nodes and relationships in this subgraph. """ return (frozenset(chain.from_iterable(node.keys() for node in self.__nodes)) | frozenset(chain.from_iterable(rel.keys() for rel in self.__relationships)))
class Walkable(Subgraph): """ A subgraph with added traversal information. """ def __init__(self, iterable): self.__sequence = tuple(iterable) nodes = self.__sequence[0::2] for node in nodes: _ = node.labels # ensure not stale Subgraph.__init__(self, nodes, self.__sequence[1::2]) def __repr__(self): return "%s(subgraph=%s, sequence=%r)" % (self.__class__.__name__, Subgraph.__repr__(self), self.__sequence) def __eq__(self, other): try: other_walk = tuple(walk(other)) except TypeError: return False else: return tuple(walk(self)) == other_walk def __ne__(self, other): return not self.__eq__(other) def __hash__(self): value = 0 for item in self.__sequence: value ^= hash(item) return value def __len__(self): return (len(self.__sequence) - 1) // 2 def __getitem__(self, index): if isinstance(index, slice): start, stop = index.start, index.stop if start is not None: if start < 0: start += len(self) start *= 2 if stop is not None: if stop < 0: stop += len(self) stop = 2 * stop + 1 return Path(*self.__sequence[start:stop]) elif index < 0: return self.__sequence[2 * index] else: return self.__sequence[2 * index + 1] def __iter__(self): for relationship in self.__sequence[1::2]: yield relationship def __add__(self, other): if other is None: return self return Path(*walk(self, other)) def __walk__(self): """ Traverse and yield all nodes and relationships in this object in order. """ return iter(self.__sequence) @property def start_node(self): """ The first node encountered on a :func:`.walk` of this object. """ return self.__sequence[0] @property def end_node(self): """ The last node encountered on a :func:`.walk` of this object. """ return self.__sequence[-1] @property def nodes(self): """ The sequence of nodes over which a :func:`.walk` of this object will traverse. """ return self.__sequence[0::2] @property def relationships(self): """ The sequence of relationships over which a :func:`.walk` of this object will traverse. """ return self.__sequence[1::2] class Entity(PropertyDict, Walkable): """ Base class for objects that can be optionally bound to a remote resource. This class is essentially a container for a :class:`.Resource` instance. """ _graph = None identity = None @classmethod def ref(cls, graph, identity): raise NotImplementedError def __init__(self, iterable, properties): Walkable.__init__(self, iterable) PropertyDict.__init__(self, properties) uuid = str(uuid4()) while "0" <= uuid[-7] <= "9": uuid = str(uuid4()) self.__uuid__ = uuid self._stale = set() def __bool__(self): return len(self) > 0 def __nonzero__(self): return len(self) > 0 @property def __name__(self): name = None if name is None and "__name__" in self: name = self["__name__"] if name is None and "name" in self: name = self["name"] if name is None and self.identity is not None: name = u"_" + ustr(self.identity) return name or u"" def __or__(self, other): # Python 3.9 added the | and |= operators to the dict # class (PEP584). This broke Entity union operations by # picking up the __or__ handler in PropertyDict before # the one in Walkable. The hack below forces Entity to # use the Walkable implementation. return Walkable.__or__(self, other) def __ior__(self, other): raise TypeError("In-place union is not permitted for %s objects" % self.__class__.__name__) def __iand__(self, other): raise TypeError("In-place intersection is not permitted for %s objects" % self.__class__.__name__) def __isub__(self, other): raise TypeError("In-place difference is not permitted for %s objects" % self.__class__.__name__) def __ixor__(self, other): raise TypeError("In-place symmetric difference is not permitted for %s objects" % self.__class__.__name__) @property def graph(self): return self._graph @graph.setter def graph(self, value): self._graph = value def clear(self): self._stale.discard("properties") super(Entity, self).clear()
[docs] class Node(Entity): """ A node is a fundamental unit of data storage within a property graph that may optionally be connected, via relationships, to other nodes. Node objects can either be created implicitly, by returning nodes in a Cypher query such as ``CREATE (a) RETURN a``, or can be created explicitly through the constructor. In the former case, the local Node object is *bound* to the remote node in the database; in the latter case, the Node object remains unbound until :meth:`created <.Transaction.create>` or :meth:`merged <.Transaction.merge>` into a Neo4j database. It possible to combine nodes (along with relationships and other graph data objects) into :class:`.Subgraph` objects using set operations. For more details, look at the documentation for the :class:`.Subgraph` class. All positional arguments passed to the constructor are interpreted as labels and all keyword arguments as properties:: >>> from py2neo import Node >>> a = Node("Person", name="Alice") """ #: OGM model class __model__ = None #: Entity-level override for merge label __primarylabel__ = None #: Entity-level override for merge key __primarykey__ = None @classmethod def ref(cls, graph, identity): obj = cls() obj.graph = graph obj.identity = identity obj._stale.add("labels") obj._stale.add("properties") return obj def __init__(self, *labels, **properties): self._remote_labels = frozenset() self._labels = set(labels) Entity.__init__(self, (self,), properties) def __repr__(self): args = list(map(repr, sorted(self.labels))) kwargs = OrderedDict() d = dict(self) for key in sorted(d): if CypherEncoder.is_safe_key(key): args.append("%s=%r" % (key, d[key])) else: kwargs[key] = d[key] if kwargs: args.append("**{%s}" % ", ".join("%r: %r" % (k, kwargs[k]) for k in kwargs)) return "Node(%s)" % ", ".join(args) def __str__(self): return xstr(cypher_repr(self)) def __eq__(self, other): if self is other: return True try: if any(x is None for x in [self.graph, other.graph, self.identity, other.identity]): return False return (issubclass(type(self), Node) and issubclass(type(other), Node) and self.graph == other.graph and self.identity == other.identity) except (AttributeError, TypeError): return False def __ne__(self, other): return not self.__eq__(other) def __hash__(self): if self.graph and self.identity: return hash(self.graph.service) ^ hash(self.graph.name) ^ hash(self.identity) else: return hash(id(self)) def __getitem__(self, item): if self.graph is not None and self.identity is not None and "properties" in self._stale: self.graph.pull(self) return Entity.__getitem__(self, item) def __ensure_labels(self): if self.graph is not None and self.identity is not None and "labels" in self._stale: self.graph.pull(self)
[docs] def keys(self): if self.graph is not None and self.identity is not None and "properties" in self._stale: self.graph.pull(self) return Entity.keys(self)
@property def labels(self): """ The full set of labels associated with with this *node*. This set is immutable and cannot be used to add or remove labels. Use methods such as :meth:`.add_label` and :meth:`.remove_label` for that instead. """ self.__ensure_labels() return LabelSetView(self._labels)
[docs] def has_label(self, label): """ Return :const:`True` if this node has the label `label`, :const:`False` otherwise. """ self.__ensure_labels() if isinstance(label, tuple): return all(lab in self._labels for lab in label) else: return label in self._labels
[docs] def add_label(self, label): """ Add the label `label` to this node. """ self.__ensure_labels() if isinstance(label, tuple): self._labels.update(label) else: self._labels.add(label)
[docs] def remove_label(self, label): """ Remove the label `label` from this node, if it exists. """ self.__ensure_labels() if isinstance(label, tuple): for lab in label: self._labels.discard(lab) else: self._labels.discard(label)
[docs] def clear_labels(self): """ Remove all labels from this node. """ self._stale.discard("labels") self._labels.clear()
[docs] def update_labels(self, labels): """ Add multiple labels to this node from the iterable `labels`. """ self.__ensure_labels() for label in labels: self.add_label(label)
[docs] class Relationship(Entity): """ A relationship represents a typed connection between a pair of nodes. The positional arguments passed to the constructor identify the nodes to relate and the type of the relationship. Keyword arguments describe the properties of the relationship:: >>> from py2neo import Node, Relationship >>> a = Node("Person", name="Alice") >>> b = Node("Person", name="Bob") >>> a_knows_b = Relationship(a, "KNOWS", b, since=1999) This class may be extended to allow relationship types names to be derived from the class name. For example:: >>> WORKS_WITH = Relationship.type("WORKS_WITH") >>> a_works_with_b = WORKS_WITH(a, b) >>> a_works_with_b (Alice)-[:WORKS_WITH {}]->(Bob) """
[docs] @staticmethod def type(name): """ Return the :class:`.Relationship` subclass corresponding to a given name. :param name: relationship type name :returns: `type` object Example:: >>> KNOWS = Relationship.type("KNOWS") >>> KNOWS(a, b) KNOWS(Node('Person', name='Alice'), Node('Person', name='Bob') """ for s in Relationship.__subclasses__(): if s.__name__ == name: return s return type(xstr(name), (Relationship,), {})
@classmethod def ref(cls, graph, identity, *nodes): obj = cls(*nodes) obj.graph = graph obj.identity = identity obj._stale.add("properties") return obj def __init__(self, *nodes, **properties): n = [] for value in nodes: if value is None: n.append(None) elif isinstance(value, string_types): n.append(value) elif isinstance(value, Node): n.append(value) else: raise TypeError("Unknown node type for %r" % value) num_args = len(n) if num_args == 0: raise TypeError("Relationships must specify at least one endpoint") elif num_args == 1: # Relationship(a) n = (n[0], n[0]) elif num_args == 2: if n[1] is None or isinstance(n[1], string_types): # Relationship(a, "TO") self.__class__ = Relationship.type(n[1]) n = (n[0], n[0]) else: # Relationship(a, b) n = (n[0], n[1]) elif num_args == 3: # Relationship(a, "TO", b) self.__class__ = Relationship.type(n[1]) n = (n[0], n[2]) else: raise TypeError("Hyperedges not supported") Entity.__init__(self, (n[0], self, n[1]), properties) def __repr__(self): args = [repr(self.nodes[0]), repr(self.nodes[-1])] kwargs = OrderedDict() d = dict(self) for key in sorted(d): if CypherEncoder.is_safe_key(key): args.append("%s=%r" % (key, d[key])) else: kwargs[key] = d[key] if kwargs: args.append("**{%s}" % ", ".join("%r: %r" % (k, kwargs[k]) for k in kwargs)) return "%s(%s)" % (self.__class__.__name__, ", ".join(args)) def __str__(self): return xstr(cypher_repr(self)) def __eq__(self, other): if self is other: return True try: if any(x is None for x in [self.graph, other.graph, self.identity, other.identity]): try: return type(self) is type(other) and list(self.nodes) == list(other.nodes) and dict(self) == dict(other) except (AttributeError, TypeError): return False return issubclass(type(self), Relationship) and issubclass(type(other), Relationship) and self.graph == other.graph and self.identity == other.identity except (AttributeError, TypeError): return False def __ne__(self, other): return not self.__eq__(other) def __hash__(self): return hash(self.nodes) ^ hash(type(self))
[docs] class Path(Walkable): """ A path represents a walk through a graph, starting on a node and visiting alternating relationships and nodes thereafter. Paths have a "overlaid" direction separate to that of the relationships they contain, and the nodes and relationships themselves may each be visited multiple times, in any order, within the same path. Paths can be returned from Cypher queries or can be constructed locally via the constructor or by using the addition operator. The `entities` provided to the constructor are walked in order to build up the new path. This is only possible if the end node of each entity is the same as either the start node or the end node of the next entity; in the latter case, the second entity will be walked in reverse. Nodes that overlap from one argument onto another are not duplicated. >>> from py2neo import Node, Path >>> alice, bob, carol = Node(name="Alice"), Node(name="Bob"), Node(name="Carol") >>> abc = Path(alice, "KNOWS", bob, Relationship(carol, "KNOWS", bob), carol) >>> abc <Path order=3 size=2> >>> abc.nodes (<Node labels=set() properties={'name': 'Alice'}>, <Node labels=set() properties={'name': 'Bob'}>, <Node labels=set() properties={'name': 'Carol'}>) >>> abc.relationships (<Relationship type='KNOWS' properties={}>, <Relationship type='KNOWS' properties={}>) >>> dave, eve = Node(name="Dave"), Node(name="Eve") >>> de = Path(dave, "KNOWS", eve) >>> de <Path order=2 size=1> >>> abcde = Path(abc, "KNOWS", de) >>> abcde <Path order=5 size=4> >>> for relationship in abcde.relationships: ... print(relationship) ({name:"Alice"})-[:KNOWS]->({name:"Bob"}) ({name:"Carol"})-[:KNOWS]->({name:"Bob"}) ({name:"Carol"})-[:KNOWS]->({name:"Dave"}) ({name:"Dave"})-[:KNOWS]->({name:"Eve"}) """ @classmethod def hydrate(cls, graph, nodes, u_rels, sequence): last_node = nodes[0] steps = [last_node] for i, rel_index in enumerate(sequence[::2]): next_node = nodes[sequence[2 * i + 1]] if rel_index > 0: u_rel = u_rels[rel_index - 1] start_node = Node.ref(graph, last_node.identity) end_node = Node.ref(graph, next_node.identity) else: u_rel = u_rels[-rel_index - 1] start_node = Node.ref(graph, next_node.identity) end_node = Node.ref(graph, last_node.identity) rel = Relationship.ref(graph, u_rel.id, start_node, u_rel.type, end_node) rel.clear() rel.update(u_rel.properties) steps.append(rel) last_node = next_node return cls(*steps) def __init__(self, *entities): entities = list(entities) for i, entity in enumerate(entities): if isinstance(entity, Entity): continue elif entity is None: entities[i] = Node() elif isinstance(entity, dict): entities[i] = Node(**entity) for i, entity in enumerate(entities): try: start_node = entities[i - 1].end_node end_node = entities[i + 1].start_node except (IndexError, AttributeError): pass else: if isinstance(entity, string_types): entities[i] = Relationship(start_node, entity, end_node) elif isinstance(entity, tuple) and len(entity) == 2: t, properties = entity entities[i] = Relationship(start_node, t, end_node, **properties) Walkable.__init__(self, walk(*entities)) def __str__(self): return xstr(cypher_repr(self)) def __repr__(self): entities = [self.start_node] + list(self.relationships) return "Path(%s)" % ", ".join(map(repr, entities))
[docs] @staticmethod def walk(*walkables): """ Traverse over the arguments supplied, in order, yielding alternating :class:`.Node` and :class:`.Relationship` objects. Any node or relationship may be traversed one or more times in any direction. :arg walkables: sequence of walkable objects """ if not walkables: return walkable = walkables[0] try: entities = walkable.__walk__() except AttributeError: raise TypeError("Object %r is not walkable" % walkable) for entity in entities: yield entity end_node = walkable.end_node for walkable in walkables[1:]: try: if end_node == walkable.start_node: entities = walkable.__walk__() end_node = walkable.end_node elif end_node == walkable.end_node: entities = reversed(list(walkable.__walk__())) end_node = walkable.start_node else: raise ValueError("Cannot append walkable %r " "to node %r" % (walkable, end_node)) except AttributeError: raise TypeError("Object %r is not walkable" % walkable) for i, entity in enumerate(entities): if i > 0: yield entity
walk = Path.walk # TODO: find a better home for this class class UniquenessError(Exception): """ Raised when a condition assumed to be unique is determined non-unique. """