Database Schema
v1: PoC
In v1, the schema looks as follows:
erDiagram
    USER ||--o{ DEVICE : has
    USER ||--|| ZONE : belongs_to
    DEVICE ||--o{ PEER: has
    ZONE ||--o{ PEER: contains
    USER{
        string id
        string username
    }
    DEVICE{
        string id
        string user_id
        string public_key
        string hostname
    }
    ZONE{
        string id
        string name
        string description
        string cidr
        string hub_zone
    }
    PEER{
        string id
        string device_id
        string zone_id
        string endpoint_ip
        string allowed_ips
        string node_address
        string child_prefix
        string hub_router
        string hub_zone
        string zone_prefix
        string reflexive_ipv4
        string endpoint_local_address_ipv4
        string symmetric_nat
    }We have 4 concepts:
- Users: Unique users of the service
- Devices: A unique end-user device
- Zone: A way of describing which devices may communicate with eachother
- Peer: An association between a Device and a Zone. This is used by the Nexodus agent to identify who it may peer with.
Since connectivity was Zone-based, we would create a "Default Zone" where all devices would be onboarded into. Adding > 1 Zone was disabled, since there were technical challenges around moving a device from one Zone to another, or for a device to have peers in > 1 Zone.
The device-onboarding flow was as follows:
- Operator/Customer sets up OIDC to provide users with access to Nexodus
- User runs the Nexodus agent
- Nexodus agent does a POST /deviceto register the unique device with Nexodus for the user. It is assigned into the single zone that a user is in (either the default zone, or a new zone where the user was moved to by an admin).
- Once the device is registered a POST /zones/$id/peerswas issued to create an association between this device and a zone.
- GET /zones/$id/peerswas polled to retrieve the peer listing for a zone.
On top of this flow, we were somewhat abusing POST /zones/$id/peers to also update peers after NAT-T.
This model was built under the assumption that multi-tenancy would be achieved by deploying an Nexodus stack per-tenant.
v2: SaaS Preview Milestone
See issue #348.
In order to create a more SaaS-ready schema, we've made the following changes.
- Replace the Zoneconstruct withOrganization. This is a GitHub-like construct, where each uniqueUseris also its ownOrganization.
- Deprecate Peerand combine it's functionality with theDevicetable.
erDiagram
    USER ||--o{ DEVICE : has
    USER ||--|{ ORGANIZATION : belongs_to
    ORGANIZATION ||--o{ DEVICE: contains
    INVITE ||--|| USER : has
    INVITE ||--|| ORGANIZATION: has
    USER{
        string id
        string username
    }
    DEVICE{
        string id
        string user_id
        string organization_id
        string public_key
        string hostname
        string tunnel_ip
        string allowed_ips
        string local_ip
        string child_prefix
        string relay
        string organization_prefix
        string reflexive_ipv4
        string endpoint_local_address_ipv4
        string symmetric_nat
    }
    ORGANIZATION{
        string id
        string name
        string description
        string cidr
        string hub_zone
    }
    INVITATION{
        string id
        string user_id
        string organization_id
        string expiry
    }In this simplified model we only have 3 concepts:
- Users: Unique users of the service
- Devices: A unique end-user device
- Organization: A collection of users who have permissions to onboard one or more devices into this organization
Since a User is also an Organization, once someone has registered to Nexodus we onboard devices into their personal organization.
The addition of users to an Organization happens via invitation. An organization owner creates an which is valid for a period of time. During that validity period, a user may accept this invitaion which will add them to the organization. An organization may also revoke and invitation after offer, but before it was accepted.
The device-onboarding flow is as follows:
- A User registers for Nexodus using a sign-up form on https://try.nexodus.io/, with the option to use SSO from Google, Github, Facebook etc..
- User downloads and runs the Nexodus agent
- Nexodus agent does a POST /deviceto register the unique device with Nexodus for the user - it's onboarded into their personal organization.
- GET /organization/$id/devicesis polled to retrieve the peer listing for a zone.
- An dedicated API endpoint was added to update device information if the public-facing address changes etc...
A key point here is that there is no admin intervention required to onboard a new user as anyone can sign-up and register.
Supporting a user belonging to multiple organizations is easy in this model since a Device may only be in one Organization at a time. A User may chose to move their device at any time, although this functionality will not be implemented during the transition from v1->v2.