GitXplorerGitXplorer
e

dignum

public
6 stars
0 forks
0 issues

Commits

List of commits on branch main.
Unverified
7a4293f84691a4222146addd4393e23f44eef7bf

fix bad target name in Makefile

eelh committed a year ago
Unverified
802e8a0b71a57c22c889207dd5cf330caf0a3d42

update project.clj

eelh committed 2 years ago
Unverified
fa4428d99a00d6c24ed777929a101a8e7eac29f5

allow running in-memory xtdb node. fix env var handling

eelh committed 2 years ago
Unverified
84c83ac64b5ce30875d30740b759ae6172e11d22

update README

eelh committed 2 years ago
Unverified
fbb7602448ae7a344aa2e47131b7125d34563795

remove a redundant check + overload

eelh committed 2 years ago
Unverified
7aedf10e36ae2528ea20e2ee6441918b38d5e079

push collection validation into json schema

eelh committed 2 years ago

README

The README file for this repository.

𝕯𝖎𝖌𝖓𝖚𝖒 𝖒𝖊𝖒𝖔𝖗𝖎𝖆:

A REST API generator for XTDB-backed resources w/ schema-on-write. It offers a simple, user-defined data model and then dynamically creates resource-oriented CRUD APIs for them.

There are two core data types:

  • Collections - which are a set of resources of the same type
  • Resources - which are JSON documents that conform to their collection's type

A collection defines this resource type via JSON Schema when the collection (which is itself a resource) is created using the Create Resource endpoint. This is a schema-as-data approach where a collection is just a resource that belongs to the (hardcoded) collections/collections! ♻️


DM: It's just a proof of concept

API

CRUD collections ...

# Create
curl -X POST "localhost:3000/collections" -H "Content-Type: application/json" -d '{
    "_name": "collections/users",
    "schema": {
        "properties": {
            "user_id": {"type": "string"},
            "name": {"type": "string"}
        },
        "type": "object"
    }
}'

# Read
curl "localhost:3000/collections" -H "Content-Type: application/json" | json_pp
curl "localhost:3000/collections/users" -H "Content-Type: application/json" | json_pp

# Update
# Backward compatible schema changes and migrations left as exercise for the reader
curl -X PUT "localhost:3000/collections/users" -H "Content-Type: application/json" -d '{
    "_name": "collections/users",
    "schema": {
        "properties": {
            "user_id": {"type": "string"},
            "name": {"type": "string"},
            "age": {"type": "number"}
        },
        "type": "object"
    }
}'

# Patch with JSON Patch
# Backward compatible schema changes and migrations left as exercise for the reader
curl -X PATCH "localhost:3000/collections/users" -H "Content-Type: application/json-patch+json" -d '[
    { "op": "test", "path": "/schema/properties/name", "value": {"type": "string"} },
    { "op": "add", "path": "/schema/properties/name", "value": {"type": "object"} }
]'

# Delete not implemented at the moment

... and then CRUD resources defined by those collections.

# Create
curl -X POST "localhost:3000/users" -H "Content-Type: application/json" -d '{"user_id": "1", "name": "Alice", "age": 30}'
curl -X POST "localhost:3000/users" -H "Content-Type: application/json" -d '{"user_id": "2", "name": "Bob", "age": 32}'

# Read
curl "localhost:3000/users" -H "Content-Type: application/json" | json_pp
curl "localhost:3000/users?name=Bob" -H "Content-Type: application/json" | json_pp
curl "localhost:3000/users/f9faf42c-3fec-48d5-907f-b3e8b0debfcb" -H "Content-Type: application/json" | json_pp

# Update
curl -X PUT "localhost:3000/users/61c22266-ee21-4ef8-9000-a9e786b3cf59" -H "Content-Type: application/json" -d '{"user_id": "2", "name": "Bob", "age": 33}'

# Patch with JSON Patch
curl -X PATCH "localhost:3000/users/61c22266-ee21-4ef8-9000-a9e786b3cf59" -H "Content-Type: application/json-patch+json" -d '[
    { "op": "test", "path": "/age", "value": 33 },
    { "op": "add", "path": "/age", "value": 34 }
]'

# Delete
curl -X DELETE "localhost:3000/users/61c22266-ee21-4ef8-9000-a9e786b3cf59" -H "Content-Type: application/json"

Usage

Connect to a running XTDB node via XTDB_URL. If not provided, we will start with an in-memory node.

XTDB_URL="http://localhost:9999" PORT=3000 make run

# development
make test
make lint

Codebase

src/dignum
├── core.clj                # api handler
├── middleware.clj          # api handler middleware
├── server.clj              # http server
└── util.clj                # utils

Why?

Built this as a future hacking tool and for learning purposes. I want to:

  • Have a really simple, conventional API I can huck data into and get flexible, bitemporal querying out of the box
    • See TODOs I'll probably never get to
  • Learn XTDB
    • Basic usage and schema-on-write approaches
    • Demystify it for others with a basic E2E CRUD API

I have previously investigated XTDB when doing some hands-on learning w/ a toy bitemporal db and visualizer.