Skip to main content

A label property graph on an RDBMS (SQLite): nodes, typed edges, an append-only event log, and an optional MCP server.

Project description

knowledge-graph-rdbms

Python License: MIT core dependencies: 0 tests: 62 passing storage: SQLite MCP

A knowledge graph for modeling meaning — entities, the kinds of things they are, and the relationships between them — in a single SQLite file.

No graph database. No Cypher. No server, no JVM, no Docker. Five tables, a small Python API, and — when you want them — an MCP server and a kg command line. The core library has zero third-party dependencies.

Python 3.10+ · MIT · zero-dependency core · library + CLI + MCP

The world isn't rows in a table — it's things, the kinds of things they are, and how they relate. A label property graph captures exactly that, and not much more: nodes (entities), typed edges (relationships), labels (sets), and JSON properties (everything else). There's no schema to design up front; meaning accretes as facts, and the shape stays as flexible as the domain it describes.

That flexibility is what makes it a natural substrate for AI agents. Hand an agent an MCP connection to this graph and it can do what agents are uniquely good at: read a domain, model what it learns, connect ideas, and reason over structure instead of prose. Every write is gated, attributed, and appended to an event log — so an agent can reshape the graph freely while you keep the receipts: audit it, replay it to any point in time, or roll a change back with one command. A memory that records why, not just what — in one embeddable file that travels wherever the agent runs: a laptop, a CI job, a serverless function, a Pi.

Small enough to hold in your head. Flexible enough to model anything.


Contents


Where it fits

It's built for graphs you want to own completely — small enough to inspect, fast enough to embed, transparent enough to trust:

  • Embedded ontologies — a knowledge graph that ships inside a single app or service, in one file you can copy and version.
  • Agent & assistant memory — facts an AI can read and write over MCP, with a full audit trail and one-command rollback.
  • Declarative datasets — describe your base graph as YAML/JSON facts and replay them into a queryable graph, deterministically.
  • "SQLite, but my data is a graph" — the moment relational rows start describing relationships, this is their natural home.

The design center is read-heavy, single-writer, up to low millions of nodes — the same sweet spot as SQLite itself: one file, in-process, no server to run. That covers a surprising amount of real work. For workloads past that center, Performance maps out exactly where the curve bends and a purpose-built graph engine starts to earn its extra moving parts.


The idea in 30 seconds

A label property graph needs only four primitives:

Primitive What it is
Node a stable id, a kind, a display name
Edge a typed, directed relationship between two nodes
Label set memberships on a node (many per node)
Property a JSON-valued key/value bag on a node or an edge

Store those in SQLite, add a few indexes, and you have a knowledge graph. Everything else in this project — traversal, an append-only history, a safety gate, the CLI, the MCP server — is built on top of those four facts.

from kgrdbms import Graph

with Graph(path="demo.db") as g:
    g.add_node("person:ada", kind="Person", name="Ada Lovelace",
               labels={"Person"}, properties={"born": 1815})
    g.add_node("field:cs", kind="Field", name="Computer Science")
    g.add_edge("person:ada", "field:cs", "FOUNDED", properties={"year": 1843})

    print(g.shortest_path("person:ada", "field:cs"))
flowchart LR
    ada(["person:ada<br/>:Person<br/>born=1815"])
    cs(["field:cs<br/>:Field"])
    ada -->|"FOUNDED (year=1843)"| cs

Design philosophy

1. Boring storage is a feature. SQLite already solved durability, transactions, indexes, and recursive queries. We don't reinvent any of it. One file, copy it to back it up, open it with any SQLite tool to inspect it.

2. The schema fits on a screen. Five tables, no surprises. You can read the entire storage layer and know exactly where every fact lives. Legibility beats cleverness.

3. The state is a pure function of data. What you query is a projection. The source of truth is an append-only log of facts plus an optional declared seed. Your whole graph is reproducible: state = replay(seed + log). That one property buys audit, undo, and time-travel for free.

4. Mutation is gated, and the gate has two layers. When you let an agent rewrite your graph over a wire, you need rules. Some rules are configurable (policy); some must be un-negotiable (compiled-in invariants). We separate mechanism from policy on purpose.

5. One engine, many doors. A library call, a kg command, and an MCP tool all flow through the same gated, logged write path into the same file. There is no "CLI version" of the truth and "MCP version" of the truth — there's one.

6. Pay for speed only when you ask. Every single write commits on its own (safe by default). Bulk paths (batch(), add_nodes, add_edges) let you opt into ~10× throughput when you mean to.


The data model

The graph is five tables. That's the whole storage layer.

erDiagram
    nodes ||--o{ node_labels : "is labeled"
    nodes ||--o{ node_properties : "has"
    nodes ||--o{ edges : "from / to"
    edges ||--o{ edge_properties : "has"

    nodes {
        text id PK
        text kind
        text name
        text created_at
    }
    node_labels {
        text node_id FK
        text label
    }
    node_properties {
        text node_id FK
        text key
        text value_json
    }
    edges {
        text id PK
        text from_node FK
        text to_node FK
        text type
    }
    edge_properties {
        text edge_id FK
        text key
        text value_json
    }

Design notes that matter:

  • Edges are unique on (from_node, type, to_node). Re-adding the same triple updates its properties instead of duplicating it. Idempotent by construction.
  • Properties are JSON. Values round-trip as whatever JSON type you store — ints, bools, lists, nested objects.
  • Foreign keys cascade. Delete a node and its labels, properties, and incident edges go with it, enforced by SQLite.
  • slug() deduplicates natural-language ids. Two strings that slugify the same become the same node — the load-bearing trick for turning prose concepts into stable ids.
  • Ids are CURIEs. person:ada-lovelace, company:apple — a compact URI: prefix:reference, where the prefix is a short stable type token and the reference is slugged (slug(name, prefix="person") mints them). It stays a plain string today and expands to a full IRI only the day you publish to the linked-data world, so interop is a cheap, additive option rather than a tax paid up front. The id is an address, not a record: identity goes in the id, changeable facts go in properties.

A sixth table, graph_events, holds the append-only history (see below). It shares the same file and connection.


Architecture: three front doors, one engine

flowchart TD
    subgraph doors["front doors"]
        CLI["kg<br/>(command line)"]
        MCP["kgrdbms-mcp<br/>(MCP server)"]
        LIB["import kgrdbms<br/>(library)"]
    end

    SVC["service.py — gated + logged write path"]
    GATE{"invariants.enforce<br/>then policy.mutation_check"}
    GRAPH[("SQLite file<br/>nodes · edges · labels · properties")]
    LOG[["graph_events<br/>(append-only log)"]]

    CLI -->|writes| SVC
    MCP -->|writes| SVC
    LIB -->|writes| SVC
    SVC --> GATE
    GATE -->|"pass"| GRAPH
    GATE -->|"record fact"| LOG

    CLI -.->|reads| GRAPH
    MCP -.->|reads| GRAPH
    LIB -.->|reads / fast bulk writes| GRAPH

The CLI, the MCP server, and your own Python code all mutate through service.py, so the safety gate and the event-log bookkeeping exist in exactly one place and can't drift between front doors.

The gate resolves invariants.enforce and policy.mutation_check through their modules at call time — so editing your policy (or monkeypatching it in a test) takes effect across every door at once.

The library also has a direct, fast, unlogged path (g.add_node(...), g.add_nodes(...)). It's the right tool for bulk loading, but those writes are not in the event log — see the warning under Event sourcing.


Many ontologies: one control plane

One graph is the engine. The control plane lets all three front doors address many named ontologies through that one engine — each its own SQLite file (and, when a workload earns it, its own engine entirely). You name an ontology; the resolver routes. Nothing else changes — the gate, the log, and the service.py write path are exactly the same; only which graph + log they act on is chosen by name.

flowchart TD
    subgraph doors["front doors, now ontology-aware"]
        CLI["kg --ontology coffee …"]
        MCP["kg_node_get(ontology='coffee')"]
        LIB["resolve('coffee')"]
    end
    RES["resolver.py<br/>name → (backend, events, entry)"]
    IDX[["index.db<br/>the registry — itself a kg"]]
    REG{"backend registry"}
    SQ[("sqlite · live")]
    PG[("postgres · live")]
    NEO[("neo4j · stub")]

    CLI --> RES
    MCP --> RES
    LIB --> RES
    RES -->|"look up the name"| IDX
    RES -->|"open the engine"| REG
    REG --> SQ
    REG -.-> PG
    REG -.-> NEO
  • The registry is itself a kg. Ontologies are nodes in an index graph (<root>/index.db), so listing them is a query and registering one is an upsert — no new storage machinery. A database of databases.
  • The default ontology is the legacy file. Omit the name and you hit <root>/graph.db, exactly as before. Multi-ontology is purely additive; nothing moves, and every existing command behaves identically.
  • Isolation is filesystem-shaped. Each ontology is its own file with its own event log. "Coffee doesn't know Ada" because they are different files — no tenant ids, no row filtering, no leak surface.
  • The engine is pluggable. A backend is a factory registered under a name. sqlite and postgres are live; neo4j is a stub that routes and fails loudly until built. Most ontologies stay embedded SQLite — the zero-dependency default — while a specific heavy one can be escalated to a purpose-built engine when its workload (not its row count) turns deep. Philosophy #6, "pay for speed only when you ask," generalized from batching to whole engines.
  • History is owned by the control plane, not the engine. A non-sqlite ontology keeps its append-only event log in a control-plane SQLite store (<root>/ontologies/<slug>/events.db); the engine is just the projection that replay and undo apply to. So a Postgres-backed ontology still gets the full audit / time-travel / one-command-revert story — graph data in Postgres, history in SQLite.
pip install "knowledge-graph-rdbms[postgres]"     # the psycopg-backed engine
kg ontology create big --backend postgres \
   --location "postgresql://user:pass@host:5432/db" --stance inferential
kg --ontology big node add company:acme --kind Company --name Acme   # writes to Postgres
kg ontology create coffee --stance inferential       # register (lands in index.db)
kg ontology list                                     # the database of databases
kg --ontology coffee node add drink:latte --kind Drink --name Latte
kg --ontology coffee out drink:latte                 # scoped to that ontology

Two ways to target a graph, mirroring the MCP ontology argument: --ontology NAME routes through the resolver (named, registered, multi-engine), while --db PATH stays the raw escape hatch onto one exact file, registry untouched.


Event sourcing: the graph is a projection

The graph you query is a cache. The append-only event log is the source of truth. Every gated mutation records a reversible fact.

flowchart LR
    SEED["genesis seed<br/>(optional, declarative)"] -->|"seed once"| PROJ
    LOG[["event log<br/>append-only, never deletes a row"]] -->|"replay in order"| PROJ[("graph projection<br/>what you query")]
    PROJ -->|"every gated mutation appends a fact"| LOG

Because the log never loses a row, you get three things at once:

  • Audit is archaeology. Every change is timestamped and attributed to an actor. kg events tails the history.
  • Undo is an event, not a delete. compensate() (CLI: kg revert <id>) emits the inverse event. The original row stays; you can see that it was reverted and by what.
  • Time travel. replay(upto_ts=...) rebuilds the projection as of any past instant.
sequenceDiagram
    participant U as you
    participant L as event log
    participant G as graph projection
    U->>L: NODE_UPSERT person:ada
    L->>G: apply → ada exists
    U->>L: revert(that event)
    L->>L: append NODE_DELETE (compensates)
    L->>G: apply → ada gone
    Note over L: both rows remain — history is intact

Two write paths, by design

You choose how each write relates to history:

  • Logged path — the CLI, the MCP server, and service.*. Every mutation is gated and appended to the log, so it's audited, reversible, and reproduced exactly by replay(). This is the path you want when the timeline is the truth.
  • Direct pathg.add_node, g.add_nodes, g.batch(). Writes go straight to the projection: the fastest way to bulk-load or stage data. They aren't in the log, so replay() (which rebuilds from the log) won't include them.

Pick per workload: direct for raw loading speed, logged when you need the history. And you can have both — kg import runs the logged path inside a single batch(), so a bulk load is fast and fully recorded.

The replay seed is just a callable, so you can declare your base graph in YAML/JSON and re-seed deterministically before the logged deltas are applied:

from kgrdbms import Graph, EventLog, replay

def genesis(g):
    # re-create your declared base facts (e.g. parsed from YAML)
    g.add_node("root", kind="Root", name="root")

replay(graph, events, genesis=genesis)              # rebuild from seed + log
replay(graph, events, genesis=genesis, upto_ts=ts)  # ...as of an instant

The safety gate: invariants vs. policy

When you expose the graph for live mutation — especially to an AI agent over MCP — "who is allowed to change what" becomes a real question. The answer here is two layers, and the order matters.

flowchart LR
    REQ(["mutation request"]) --> INV{"invariants.enforce<br/>compiled-in rule violated?"}
    INV -->|yes| E1[/"InvariantViolation<br/>(cannot be configured away)"/]
    INV -->|no| POL{"policy.mutation_check<br/>configured rule denies?"}
    POL -->|yes| E2[/"PermissionError"/]
    POL -->|no| APPLY["apply to graph"]
    APPLY --> REC[("record reversible event")]
  • invariants.py is mechanism. Rules here are enforced in code, ahead of policy, and cannot be turned off by configuration or talked around over the wire. Changing one is a code change and a redeploy. The default enforces nothing — invariants are inherently domain-specific.
  • policy.py is configuration. A single mutation_check(ctx) -> Decision function. The default is permissive (everything allowed). Edit it to seal the parts that must not change. Five to ten lines is usually enough.

Invariants run first, so a permissive (or compromised) policy can never re-open something an invariant has sealed. That's the whole reason to separate them.

# policy.py — append-only example: callers may add, never delete or modify
def mutation_check(ctx):
    if ctx.operation in {"node_delete", "edge_remove", "graph_clear"}:
        return Decision.deny("policy is append-only; no deletions")
    return Decision.allow()

Install

pip install knowledge-graph-rdbms            # core library + the kg CLI
pip install "knowledge-graph-rdbms[mcp]"     # + the MCP server

Or, to get the kg / kgrdbms-mcp commands on your PATH globally (uv or pipx):

uv tool install "knowledge-graph-rdbms[mcp]"
# or, from a local checkout, editable:
uv tool install --editable "/path/to/knowledge-graph-rdbms[mcp]"

Storage defaults to ~/.kgrdbms/graph.db. Override with the KGRDBMS_HOME environment variable, or per-command with kg --db PATH, or in code with Graph(path=...).


Quickstart

As a library

from kgrdbms import Graph

with Graph(path="demo.db") as g:
    g.add_node("person:ada", kind="Person", name="Ada Lovelace", labels={"Person"})
    g.add_node("field:cs", kind="Field", name="Computer Science")
    g.add_edge("person:ada", "field:cs", "FOUNDED", properties={"year": 1843})

    for edge, target in g.out("person:ada"):
        print(edge.type, "->", target.name)
    print(g.shortest_path("person:ada", "field:cs"))

Bulk loading — every single write commits on its own (one fsync each), so for bulk work opt into one transaction and go ~10× faster:

# fastest: executemany under a single commit
g.add_nodes([
    {"id": "person:ada", "kind": "Person", "name": "Ada", "labels": ["Person"]},
    {"id": "field:cs", "kind": "Field", "name": "Computer Science"},
])
g.add_edges([("person:ada", "field:cs", "FOUNDED")])  # dicts / Edge objects work too

# or batch() — defer commits for any mix of writes, atomic rollback on error
with g.batch():
    for spec in many_specs:
        g.add_node(**spec)

As a CLI

The kg command ships with the core install (stdlib argparse, no extra deps). Reads hit the graph directly; writes go through the same gate + event log as the MCP server, so kg replay / kg revert work and a custom policy is honored at the console too.

kg node add person:ada --kind Person --name "Ada Lovelace" \
    --label Person --prop born=1815 --prop fields='["math","cs"]'
kg node add field:cs --kind Field --name "Computer Science"
kg edge add person:ada field:cs FOUNDED --prop year=1843

kg out person:ada                 # outbound edges
kg path person:ada field:cs       # shortest path
kg nodes-by-kind Person
kg stats
kg --json node get person:ada     # machine-readable output for piping

kg events -n 10                   # tail the event log
kg revert <event-id>              # undo a mutation (compensating event)
kg replay                         # rebuild the projection from the log

kg import graph.json              # bulk {nodes, edges} import (gated + logged)

kg ontology create coffee --stance inferential   # register a named ontology
kg ontology list                  # the registry (database of databases)
kg --ontology coffee node add drink:latte --kind Drink   # route to it
kg serve                          # launch the MCP server (needs [mcp])

--prop key=value values are parsed as JSON when possible (born=1815 → int, ok=true → bool, tags='["a"]' → list) and kept as a plain string otherwise. Target a named ontology with --ontology NAME (routed through the resolver, default: the default ontology); --db PATH is the raw escape hatch onto one exact file. Exit codes: 0 ok · 1 not found / bad input · 2 policy denial · 3 invariant violation.

As an MCP server

pip install "knowledge-graph-rdbms[mcp]"
claude mcp add kgrdbms -- kgrdbms-mcp          # register with Claude Code

Or hand-edit a client config (e.g. Claude Desktop):

{ "mcpServers": { "kgrdbms": { "command": "kgrdbms-mcp" } } }

It exposes kg_-prefixed tools for reads (kg_node_get, kg_nodes_by_kind, kg_neighborhood, kg_shortest_path, kg_descendants, …), gated writes (kg_node_upsert, kg_edge_add, kg_node_delete, …), bulk composition (kg_import — a whole {nodes, edges} batch in one call, so an agent populates an ontology in a single tool call instead of dozens), and the event log (kg_events_tail, kg_event_revert, kg_replay). Every write passes through the invariants + policy gate and is recorded — same engine, same file as the CLI. Every tool also takes an optional ontology name (omit for the default), and kg_ontologies_list / kg_ontology_create manage the registry — so an agent can discover, create, and route between ontologies entirely over MCP.


Performance

All figures come from bench/benchmark.py, which reports full distributions (p50–p99), not single shots — and the charts below are rendered straight from that data by bench/charts.py. Run both on your own machine in one command. (Shown: Apple Silicon, CPython 3.14, SQLite 3.50 — illustrative, not a promise.)

Operation Throughput
node(id) point lookup ~120,000 / s
add_node (per-call, durable) ~17,000 / s
add_node inside batch() ~157,000 / s
add_nodes([...]) bulk ~189,000 / s
replay() (events/sec) ~26,000 / s

Writes — the batching lever

Write throughput — batch the commit, ~10× faster

Each single write commits on its own for durability. Wrapping a bulk load in batch() / add_nodes / add_edges collapses those per-call commits into one transaction for an ~10× jump — same engine, you just tell it a batch is coming. The gated + logged path (what the CLI and MCP server use) adds the invariants+policy check and an event record per write, and still clears tens of thousands per second.

Reads — fast, with an honest tail

Read latency — p50 marker, whisker to p99, log scale

Point lookups land in single-digit microseconds, and multi-node reads hydrate the whole result set in a constant number of queries (no N+1 fan-out). The chart plots p50 → p99 on purpose: randomized shortest_path endpoints make some walks short and some span the whole chain, and an average would bury that tail.

A note on the runtime

kgrdbms is Python, and for performance that's a deliberate non-issue: the same SQLite engine runs under CPython, Node, and Bun, so the gap between them is pure binding overhead — under 2×, and it doesn't even favor one runtime across operations.

Same SQLite across CPython, Node, and Bun

The lever that actually moved the needle was transaction batching (~10×, above), not the language. Reproduce it with python bench/runtimes/compare.py.

Where the curve bends

We measured it against Neo4j — same graph, same queries, identical methodology (full harness and reproduction in bench/neo4j/):

Where the crossover is — kgrdbms vs Neo4j

Queries compile to SQL over B-tree indexes, so each traversal hop is an index lookup — wonderfully cheap for point reads and shallow traversals. An in-process lookup here is ~7µs, while the same query to Neo4j pays a Bolt round-trip (~0.4ms) before it even touches data. So for the small, frequent operations that are the bread and butter of agent memory, the embedded graph wins by 30–60×.

A purpose-built engine pulls ahead exactly where the workload — not the row count — turns deep:

  • Deep, high-fan-out traversal. Index-free adjacency follows direct pointers between nodes. A 1,000-deep walk costs kgrdbms ~52ms (recursive CTE + row hydration) but ~0.7ms for Neo4j, which pointer-chases under its own round-trip budget — a 76× swing the other way.
  • Complex pattern matching. A Cypher planner optimizes multi-pattern queries in ways a fixed traversal API doesn't attempt.
  • Concurrent writers and scale-out. Single-file SQLite is one writer at a time; clustered engines aren't.

Rule of thumb: read-heavy and shallow up to low millions of nodes is firmly home turf; deep-traversal or pattern-heavy work is where a dedicated engine earns its complexity. The crossover is workload-shaped, not a single magic number — so we measured ours, and you can measure yours.

SQLite vs the live Postgres engine

Because postgres is a live backend, you can run the same op suite against both engines and watch the round-trip tax directly (bench/postgres/). Embedded SQLite wins the small, frequent ops by 30–60× — a point lookup is in-process, the Postgres one pays a localhost round-trip. The exception is the one deep traversal that runs as a single server-side query: the recursive-CTE descendants is where Postgres pulls ahead (~0.5×), while the per-hop-BFS shortest_path over the same chain is 67× slower — identical traversal, opposite verdict, decided entirely by how many times the work crosses the wire. Postgres earns its place on concurrency and scale, not single-thread latency; the control plane lets you escalate one ontology to it while the hot, shallow ones stay embedded.


Command reference

Command What it does
kg stats node/edge counts and db path
kg node add ID --kind K … create or update a node (gated + logged)
kg node get ID fetch a node
kg node del ID delete a node (cascades edges)
kg node set-prop ID KEY VAL set one property
kg node add-label ID LABEL add a label
kg edge add FROM TO TYPE add an edge
kg edge rm FROM TO TYPE remove an edge
kg nodes-by-kind KIND list nodes of a kind
kg nodes-by-label LABEL list nodes with a label
kg out ID [--type T] outbound edges
kg in ID [--type T] inbound edges
kg path FROM TO shortest undirected path
kg neighbors ID [--depth N] nodes within N hops
kg descendants ID TYPE nodes reachable along one edge type
kg events [-n N] tail the event log
kg revert EVENT_ID undo an event (compensating event)
kg replay [--upto TS] rebuild the projection from the log
kg import FILE bulk {nodes, edges} import (gated + logged)
kg ontology list list registered ontologies (the registry)
kg ontology create NAME … register an ontology (--backend, --stance)
kg serve [--transport T] run the MCP server

Add --json to any command for machine-readable output. Target a graph with --ontology NAME (routed through the resolver; default: the default ontology) or --db PATH (the raw escape hatch onto one exact file, registry bypassed).


Project layout

kgrdbms/
├── graph.py        # the label property graph over SQLite (no internal deps)
├── events.py       # append-only event log: record, compensate, replay
├── policy.py       # configurable mutation policy (permissive by default)
├── invariants.py   # compiled-in invariants, checked before policy (no-op default)
├── service.py      # the shared gated + logged write path
├── resolver.py     # control plane: ontology name → (backend, events, entry) + the index
├── backends/       # pluggable engine registry
│   ├── base.py     #   GraphBackend protocol + raising stub skeleton
│   ├── sqlite.py   #   live engine (adapter over Graph)
│   ├── postgres.py #   live engine (psycopg; jsonb + recursive CTEs); [postgres] extra
│   └── neo4j.py    #   stub (deep-traversal escalation)
├── cli.py          # the `kg` command (stdlib argparse)
└── mcp_server.py   # the MCP server (optional [mcp] extra)

graph.py imports nothing internal — it's a usable, dependency-free LPG on its own. Everything else layers on top; service.py depends only on the GraphBackend protocol, never a concrete engine.


Development

git clone <repo> && cd knowledge-graph-rdbms
uv venv && uv pip install -e ".[dev]"
pytest                       # 62 tests
python bench/benchmark.py    # benchmark with p50–p99 (see bench/README.md)

License

MIT.

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

knowledge_graph_rdbms-0.1.1.tar.gz (557.9 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

knowledge_graph_rdbms-0.1.1-py3-none-any.whl (55.4 kB view details)

Uploaded Python 3

File details

Details for the file knowledge_graph_rdbms-0.1.1.tar.gz.

File metadata

  • Download URL: knowledge_graph_rdbms-0.1.1.tar.gz
  • Upload date:
  • Size: 557.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for knowledge_graph_rdbms-0.1.1.tar.gz
Algorithm Hash digest
SHA256 db2ad29d415ca226ee21a4339991689d1bf57a6ac09fde920b1678e91aef7885
MD5 5e16349048cd18c20dc9f509c0fd0089
BLAKE2b-256 cc6e82594698d375870ea9f9e79ea85b9077de0aa36869f4525e234d7e3c72ad

See more details on using hashes here.

Provenance

The following attestation bundles were made for knowledge_graph_rdbms-0.1.1.tar.gz:

Publisher: publish.yml on cunicopia-dev/knowledge-graph-rdbms

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

File details

Details for the file knowledge_graph_rdbms-0.1.1-py3-none-any.whl.

File metadata

File hashes

Hashes for knowledge_graph_rdbms-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 b0ab40b46e9c8abb010c7c3d7b776f2d256095ddc38622aeb0da31dbdc877622
MD5 e98b6b4470e4815f2e80bfa313910298
BLAKE2b-256 6a0cda208745f2faf473df2bd26f3c540cf7f8e6f9f130c0c8fbd23e1e177ba5

See more details on using hashes here.

Provenance

The following attestation bundles were made for knowledge_graph_rdbms-0.1.1-py3-none-any.whl:

Publisher: publish.yml on cunicopia-dev/knowledge-graph-rdbms

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 Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page