Skip to main content

A Python package for modelling data in a Neo4j graph database with Pydantic and Pandas.

Project description

Neontology: Neo4j, Python and Pydantic

PyPI - Version Read the Docs PyPI - Python Version GitHub Actions Workflow Status GitHub License Pydantic v2

Easily ingest data into a openCypher / GQL (Graph Query Language) graph database like Neo4j using Python, Pydantic and pandas.

Neontology is a simple object-graph mapper which lets you use Pydantic models to define Nodes and Relationships. It imposes certain restrictions on how you model data, which aims to make life easier for most users in areas like the construction of knowledge graphs and development of graph database applications.

Neontology is inspired by projects like py2neo (which is no longer maintained), Beanie and SQLModel.

Read the documentation here.

Installation

pip install neontology

Note on v2

Version 2 introduces some new features and breaking changes.

Initialization has changed to use a config object (see below) and 'merged'/'created' properties have been removed from the base node & base relationship (at a minimum you may need to add them back to a custom base model if required).

See the changelog or read the docs for more.

Example

from typing import ClassVar, Optional, List
import pandas as pd
from neontology import BaseNode, BaseRelationship, init_neontology, Neo4jConfig

# We define nodes by inheriting from BaseNode
class PersonNode(BaseNode):
    __primarylabel__: ClassVar[str] = "Person"
    __primaryproperty__: ClassVar[str] = "name"
    __secondarylabels__: ClassVar[Optional[List]] = ["Individual", "Somebody"]
    
    name: str
    age: int

# We define relationships by inheriting from BaseRelationship
class FollowsRel(BaseRelationship):
    __relationshiptype__: ClassVar[str] = "FOLLOWS"
    
    source: PersonNode
    target: PersonNode

# initialise the connection to the database
config = Neo4jConfig(
    uri="neo4j+s://mydatabaseid.databases.neo4j.io", 
    username="neo4j",
    password="<PASSWORD>"
)
init_neontology(config) 

# Define a couple of people
alice = PersonNode(name="Alice", age=40)

bob = PersonNode(name="Bob", age=40)

# Create them in the database
alice.create()
bob.create()

# Create a follows relationship between them
rel = FollowsRel(source=bob,target=alice)
rel.merge()

# We can also use pandas DataFrames to create multiple nodes
node_records = [{"name": "Freddy", "age": 42}, {"name": "Philippa", "age":42}]
node_df = pd.DataFrame.from_records(node_records)

PersonNode.merge_df(node_df)

# We can also merge relationships from a pandas DataFrame, using the primary property values of the nodes
rel_records = [
    {"source": "Freddy", "target": "Philippa"},
    {"source": "Alice", "target": "Freddy"}
]
rel_df = pd.DataFrame.from_records(rel_records)

FollowsRel.merge_df(rel_df)

Configuring your graph connection

On initialisation

You can explicitly provide access information as in the example above with a config object.

With a dotenv file or environment variables

You can use a .env file as below (or explicitly set them as environment variables), which should automatically get picked up by neontology.

# .env
NEO4J_URI=neo4j+s://myneo4j.example.com
NEO4J_USERNAME=neo4j
NEO4J_PASSWORD=<PASSWORD>

With the above environment variables defined, you can just use init_neontology() without providing any arguments.

Executing Queries

Neontology has limited, experimental support for running GQL/cypher queries.

Using a GraphConnection, you can call evaluate_query with a GQL/cypher query which returns nodes and relationships and get them back as Neontology Nodes Relationships.

Once neontology is initialized, only one connection to the database is used under the hood which can be accessed with GraphConnection.

from neontology import init_neontology, GraphConnection

# Only nodes and relationships that have already been defined can be returned as results.
class PersonNode(BaseNode):
    __primarylabel__: ClassVar[str] = "Person"
    __primaryproperty__: ClassVar[str] = "name"
    
    name: str
    age: int

class FollowsRel(BaseRelationship):
    __relationshiptype__: ClassVar[str] = "FOLLOWS"
    
    source: PersonNode
    target: PersonNode


init_neontology()

gc = GraphConnection()

cypher_query = """
MATCH (n)
WHERE n.name = $name
OPTIONAL MATCH (n)-[r:FOLLOWS]-(o:Person)
RETURN r, o
"""

results = gc.evaluate_query(cypher_query, {"name": "bob"})

results.records[0]["nodes"]["o"]    # Get the "o" result of the 1st record as a PersonNode
results.nodes                       # The nodes returned by the query as PersonNode objects
results.relationships               # The relationships as FollowsRel objects

For large or complex queries, data science or visualization/exploration, consider using a native driver or built-in interface (like Neo4j Browser/Bloom or Memgraph Lab).

Alternative Graph Engines

Neontology has experimental support for GQL/openCypher property graph databases other than Neo4j:

  • Memgraph

Memgraph Engine

Memgraph is a Neo4j compatible database.

from neontology import init_neontology, MemgraphConfig
from neontology.graphengines import MemgraphEngine

config = MemgraphConfig(
                "uri": "bolt://localhost:9687",
                "username": "memgraphuser",
                "password": "<MEMGRAPH PASSWORD>"
            )

init_neontology(config)

You can also use the following environment variables and just init_neontology(MemgraphConfig()):

  • MEMGRAPH_URI
  • MEMGRAPH_USERNAME
  • MEMGRAPH_PASSWORD

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

neontology-2.0.4.tar.gz (36.1 kB view details)

Uploaded Source

Built Distribution

neontology-2.0.4-py3-none-any.whl (32.0 kB view details)

Uploaded Python 3

File details

Details for the file neontology-2.0.4.tar.gz.

File metadata

  • Download URL: neontology-2.0.4.tar.gz
  • Upload date:
  • Size: 36.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.0.1 CPython/3.12.8

File hashes

Hashes for neontology-2.0.4.tar.gz
Algorithm Hash digest
SHA256 c9f2aa132d39e36bd14c77e6248833771960b94f7ca9f6c7769968b8a1e7889a
MD5 ec75096d46ea96b144d0dd07ad74983e
BLAKE2b-256 7f4c78d7e0baa22d038b85992a5920e106ddc9d4bc22165fb6ec83fe99c3fc8c

See more details on using hashes here.

Provenance

The following attestation bundles were made for neontology-2.0.4.tar.gz:

Publisher: pypi.yml on ontolocy/neontology

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file neontology-2.0.4-py3-none-any.whl.

File metadata

  • Download URL: neontology-2.0.4-py3-none-any.whl
  • Upload date:
  • Size: 32.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.0.1 CPython/3.12.8

File hashes

Hashes for neontology-2.0.4-py3-none-any.whl
Algorithm Hash digest
SHA256 3c9395f3a6d01fdba14c409be3551f41b53d45f3c8a9b524647e8f2fd3ba39e1
MD5 568add209501b41d385f74cd94f560a4
BLAKE2b-256 434a37827bc485b8960c36fb5ea2afe6dc473bf05a71a441fdab046d18515e33

See more details on using hashes here.

Provenance

The following attestation bundles were made for neontology-2.0.4-py3-none-any.whl:

Publisher: pypi.yml on ontolocy/neontology

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page