rhizome/README.md

215 lines
6.3 KiB
Markdown
Raw Normal View History

2024-12-24 13:58:02 -06:00
# Concepts
2024-12-24 14:02:21 -06:00
| | Implemented | Notes |
| ------------- | ----------- | ------------------------------------------------------------------------ |
| Peering | Yes | Currently using `RHIZOME_SEED_PEERS`, no gossip / discovery |
| Schemas | Not really | Currently very thin layer allowing TypedCollections |
| Relationships | No | Supporting relational algebra among domain entities |
2024-12-25 00:42:16 -06:00
| Views | Yes | Lossless: Map the `targetContext`s as properties of domain entities. |
| | | Lossy: Use a delta filter and a resolver function to produce a view. |
| | | Currently using functions rather than JSON-Logic expressions. |
2024-12-24 14:02:21 -06:00
| Functions | No | Arbitrary subscribers to delta stream (that can also emit deltas?) |
2024-12-25 19:33:33 -06:00
| Tests | Yes | We are set up to run unit tests and multi-node tests |
2024-12-24 14:02:21 -06:00
| Identity | No | Probably a public key / private key system |
| Contexts | No | Each context may involve different lossy functions and delta filters |
| HTTP API | Yes | Basic peering info and entity CRUD |
2024-12-24 13:58:02 -06:00
2024-12-25 00:42:16 -06:00
If we express views and filter rules as JSON-Logic, we can easily include them in records.
2024-12-24 13:58:02 -06:00
# Development / Demo
2024-12-22 17:30:41 -06:00
## Setup
2024-12-19 23:16:45 -06:00
Install [`nvm`](https://nvm.sh)
2024-12-22 17:30:41 -06:00
Clone repo
```bash
git clone https://gitea.dgov.io/ladd/rhizome
```
2024-12-22 17:30:41 -06:00
Use `nvm` to install and activate the target nodejs version
2024-12-22 17:30:41 -06:00
```bash
nvm install
```
Install nodejs packages
```bash
2024-12-22 17:30:41 -06:00
npm install
```
## Build
Compile Typescript
2024-12-22 17:30:41 -06:00
```bash
npm run build
```
2024-12-22 17:30:41 -06:00
During development, it's useful to run the compiler in watch mode:
```bash
npm run build:watch
2024-12-22 17:30:41 -06:00
```
## Run tests
```bash
npm run test
```
## Run test coverage report
2024-12-22 17:30:41 -06:00
```bash
npm run test:coverage
```
## Run multiple live nodes locally as separate processes
To demonstrate the example application, you can open multiple terminals, and in each terminal execute something like the following:
2024-12-22 17:30:41 -06:00
2024-12-22 17:33:22 -06:00
```bash
export DEBUG="*,-express"
2024-12-22 17:33:22 -06:00
export RHIZOME_REQUEST_BIND_PORT=4000
export RHIZOME_PUBLISH_BIND_PORT=4001
2024-12-25 16:13:48 -06:00
export RHIZOME_SEED_PEERS='localhost:4002, localhost:4004'
2024-12-22 17:33:22 -06:00
export RHIZOME_HTTP_API_PORT=3000
export RHIZOME_PEER_ID=peer1
npm run example-app
2024-12-22 17:33:22 -06:00
```
2024-12-22 17:32:00 -06:00
2024-12-22 17:33:22 -06:00
```bash
export DEBUG="*,-express"
2024-12-22 17:33:22 -06:00
export RHIZOME_REQUEST_BIND_PORT=4002
export RHIZOME_PUBLISH_BIND_PORT=4003
2024-12-25 16:13:48 -06:00
export RHIZOME_SEED_PEERS='localhost:4000, localhost:4004'
export RHIZOME_HTTP_API_PORT=3001
2024-12-22 17:33:22 -06:00
export RHIZOME_PEER_ID=peer2
npm run example-app
2024-12-22 17:33:22 -06:00
```
2024-12-22 17:32:19 -06:00
2024-12-22 17:33:22 -06:00
```bash
export DEBUG="*,-express"
2024-12-22 17:33:22 -06:00
export RHIZOME_REQUEST_BIND_PORT=4004
export RHIZOME_PUBLISH_BIND_PORT=4005
2024-12-25 16:13:48 -06:00
export RHIZOME_SEED_PEERS='localhost:4000, localhost:4002'
export RHIZOME_HTTP_API_PORT=3002
2024-12-22 17:33:22 -06:00
export RHIZOME_PEER_ID=peer3
npm run example-app
2024-12-22 17:33:22 -06:00
```
2024-12-22 17:30:41 -06:00
In a separate terminal, you can use `curl` to interact with an instance.
`jq` is helpful for formatting the json responses.
Query the number of peers seen by a given node (including itself)
2024-12-22 17:33:22 -06:00
```bash
curl -s http://localhost:3000/peers/count | jq
```
Query the list of peers seen by a given node (including itself)
```bash
2024-12-22 17:33:22 -06:00
curl -s http://localhost:3000/peers | jq
```
Query the number of deltas ingested by this node
```bash
2024-12-22 17:33:22 -06:00
curl -s http://localhost:3000/deltas/count | jq
```
Query the list of deltas ingested by this node
```bash
2024-12-22 17:33:22 -06:00
curl -s http://localhost:3000/deltas | jq
```
2024-12-23 17:29:38 -06:00
The example creates a `new TypedCollection<User>("user")` and calls `connectRhizome` to join it with the network.
The collection is synchronized across the cluster and optionally CRUD type operations are served via HTTP.
Query the list of User IDs
```bash
curl -s http://localhost:3000/user/ids
```
Query the list of User IDs
```bash
curl -s http://localhost:3000/user/ids
```
Read a User by ID
```bash
curl -s http://localhost:3000/user/taliesin-1
```
Create a User
```bash
cat <<EOF >/tmp/user.json
{"id": "optional-id",
"properties": {
"name": "required",
"nameLong": "optional",
"email": "optional"}}
EOF
curl -s -X PUT -H 'content-type:application/json' -d @/tmp/user.json http://localhost:3000/user | jq
```
# More About Concepts
## Clocks?
Do we want to involve a time synchronization protocol? e.g. ntpd
If not, what's the best we could do?
Maybe just expect nodes to record relative times, and
patch together a sequence based on the relative times.
This adds complexity and still has limited precision.
We could just let the clocks drift and so on, and make inferences at
query resolution time.
We could do some extra work and keep track of what time our peers think it is.
Then if their clocks drift relative to ours, we can seek consensus among a broader range of peers.
But at that point just run ntpd. Can still do consensus to verify
but probably no need to implement custom time synchronization protocol.
Apparently PTP, Precision Time Protocol, is a thing.
PTP affords for a layer of user defined priority for best clock selection.
## Peering
### ZeroMQ
Currently we're handling networking with ZeroMQ pub/sub over TCP transport.
* ZeroMQ supports encryption, with public/private key pairs.
* A subscriber needs to know the public key of the publisher in order to connect.
* We're aiming for symmetry, so we'll need a strategy to establish these reciprocal relationships.
### GossipSub
One option is to replace ZeroMQ with GossipSub,
which may function better in an open network envoronment.
Considerations with GossipSub may include
* topics -- namespacing
* peer discovery
### TincVPN
Another layer which is available would be [Tinc VPN](https://tinc-vpn.org).
Tinc...
* is a daemon
* creates a mesh VPN
* uses tap/tun network devices
* network can run in router, switch, or hub mode
* performs UDP hole punching
* forwards packets among peers
* performs spanning tree routing
* participants only see messages if they've added the sender's public key to their configuration
Ideally at least one node in a given network
needs to listen on a public interface address.
[Tinc configuration docs](https://tinc-vpn.org/documentation/Main-configuration-variables.html)
provide some insight into its functioning.
Considerations imposed by Tinc would include
* IP addressing
* public key management