@presenti/server
TypeScript icon, indicating that this package has built-in type declarations

1.2.6 • Public • Published

presenti

Service for tracking presences across various platforms.

Getting Started

Presenti is easy to deploy. My favorite method for maintaining an active process is pm2, though the process is unlikely to die, so you could run it in a detached screen if you watned.

Dependencies

  • PostgreSQL
  • Node >=12
  • TypeScript (only needed for development)
  • Discord (only needed if you want to have OAuth)
  1. Clone the latest version of presenti using git clone https://github.com/EricRabil/presenti.git
  2. Initialize the modules using npm i
  3. Initialize the service by running npm run run
  4. Presenti will ask you a few setup questions, like database credentials and other required details. It will then save the configuration file to config.json at the root of the project, and it will resemble this:
{
    "port": 8138,
    "registration": false,
    "discord": null,
    "crypto": {
        "jwtSecret": "<randomly generated secret>",
        "firstPartyKey": "<randomly generated secret>"
    },
    "web": {
        "host": "://localhost:8138"
    },
    "db": {
        "host": "localhost",
        "port": 5432,
        "name": "presenti",
        "username": "",
        "password": "",
        "cache": false
    },
    "modules": {}
}
  1. To generate a first-party API key for modules like presenti-additions, run the following in the Presenti shell and copy the results (I will be adding an admin panel for this eventually):
(Presenti) % await SecurityKit.firstPartyApiKey()

Modules

Presenti is built around modules, which can run in the server process or in their own process. If you'd like to load in modules, specify them and their configuration in the modules property of the config file.

Examples


Presenti Additions

Add the following entry to your modules file to add the @presenti/additions module, which includes Discord and Spotify integration.

{
  "modules": {
    "@presenti/additions": {
      "discord": {
          "token": "your token",
          "prefix": ">",
          "clientID": "your client ID",
          "clientSecret": "your client secret"
      },
      // It's important that you include this property, or spotifyInternal may not work correctly. Config generation for modules is coming soon.
      "spotifyInternal": {}
    }
  }
}

Streaming API

The Streaming API is for apps that are subscribing to presence data.

WebSocket

Currently, the Streaming API only exists on the WebSocket platform. If you have a use case for other platforms, please open an issue.

Subscribing to a presence is fairly straightforward. Simply open a connection to

ws://{host}/presence/{name}

Where host is the address to your server, and name is the name tied to the token.

Presence updates look like this, and are also sent upon connection:

{
  "activities": [
    {
      "presences": [
        {
          "title": "Listening to Spotify",
          "largeText": {
            "text": "Issues/Hold On",
            "link": "https://open.spotify.com/track/0bxmVPKnEopTyuMMkaTvUb"
          },
          "smallTexts": [
            {
              "text": "by Teyana Taylor",
              "link": "https://open.spotify.com/artist/4ULO7IGI3M2bo0Ap7B9h8a"
            },
            {
              "text": "on K.T.S.E.",
              "link": "https://open.spotify.com/album/0mwf6u9KVhZDCNVyIi6JuU"
            }
          ],
          "image": {
            "src": "https://i.scdn.co/image/ab67616d0000b273abca6b34e370af95f3b926bd",
            "link": "https://open.spotify.com/track/0bxmVPKnEopTyuMMkaTvUb"
          },
          "timestamps": {
            "start": 1589001619550,
            "stop": 1589002629550
          },
          "gradient": {
            "enabled": true
          },
          "isPaused": true
        }
      ]
    }
  ]
}

This is how that would be rendered

Rendered Presence

These are the only messages that will be sent by the server. If the connection closes, simply re-open the connection. There is no authentication necessary for this endpoint.

Presence API

The Presence API is for apps that are providing presence data to the service.

REST

The REST API is useful for platforms in which it is not feasible to maintain a WebSocket connection for updating your presence.

New Session

GET /session?token={your token}

Call this endpoint with a valid token to open a new presence session. The session will remain valid for as long as the value of expires, in milliseconds. This is server-configurable, and you can prevent the session from expiring by calling any of the below endpoints.

Example Response

{
  "sessionID": "34e7f94b-7fa3-4196-912a-88186b25299d",
  "expires": 10000
}

Update Presence

PUT /session?token={token}&id={sessionID}

Call this endpoint to update the presence state for the session. Calling this endpoint doubles as a session refresh, resetting the time until session expiration.

Example Request

{
  "presences": [
    {
      "presences": [
        {
          "title": "Listening to Spotify",
          "largeText": {
            "text": "Issues/Hold On",
            "link": "https://open.spotify.com/track/0bxmVPKnEopTyuMMkaTvUb"
          },
          "smallTexts": [
            {
              "text": "by Teyana Taylor",
              "link": "https://open.spotify.com/artist/4ULO7IGI3M2bo0Ap7B9h8a"
            },
            {
              "text": "on K.T.S.E.",
              "link": "https://open.spotify.com/album/0mwf6u9KVhZDCNVyIi6JuU"
            }
          ],
          "image": {
            "src": "https://i.scdn.co/image/ab67616d0000b273abca6b34e370af95f3b926bd",
            "link": "https://open.spotify.com/track/0bxmVPKnEopTyuMMkaTvUb"
          },
          "timestamps": {
            "start": 1589001619550,
            "stop": 1589002629550
          },
          "gradient": {
            "enabled": true
          },
          "isPaused": true
        }
      ]
    }
  ]
}

This is how that would be rendered

Rendered Presence

Example Response

{
  "ok": true
}

Refresh Session

PUT /session/refresh?token={token}&id={sessionID}

Call this endpoint to prevent the session from expiring, but don't have an updated payload to send.

Example Response

{
  "ok": true
}

WebSocket

The WebSocket API is useful for dedicated presence servers that will be sending a large or unknown amount of data.

An Overview

The WebSocket API follows the following general schema when exchanging data

{
  "type": 0,
  "data": []
}

Type Codes

These are the valid codes that may be sent or received over the WebSocket API.

Code Name Description Sender Response?
0 Ping Pings the server Client
1 Pong Sent upon ping Server
2 Presence Updates presence, where the data property is an array of presence objects Client
3 Identify Authenticates session, where the data property is the user token Client
4 Greetings Sent by the server, upon successful authentication. Start sending pings after receiving greetings. Server

Example messages

Identify

Sent upon connection open, this identifies the session for the server.

{
  "type": 3,
  "data": "{token}"
}
Ping

Though not required, these can help prevent a connection time-out and unnecessarily reconnecting.

{
  "type": 0
}
Presence Update
{
  "type":2,
  "data": [
    {
      "presences": [
        {
          "title": "Listening to Spotify",
          "largeText": {
            "text": "Issues/Hold On",
            "link": "https://open.spotify.com/track/0bxmVPKnEopTyuMMkaTvUb"
          },
          "smallTexts": [
            {
              "text": "by Teyana Taylor",
              "link": "https://open.spotify.com/artist/4ULO7IGI3M2bo0Ap7B9h8a"
            },
            {
              "text": "on K.T.S.E.",
              "link": "https://open.spotify.com/album/0mwf6u9KVhZDCNVyIi6JuU"
            }
          ],
          "image": {
            "src": "https://i.scdn.co/image/ab67616d0000b273abca6b34e370af95f3b926bd",
            "link": "https://open.spotify.com/track/0bxmVPKnEopTyuMMkaTvUb"
          },
          "timestamps": {
            "start": 1589001619550,
            "stop": 1589002629550
          },
          "gradient": {
            "enabled": true
          },
          "isPaused": true
        }
      ]
    }
  ]
}

This is how that would be rendered

Rendered Presence

Pre-requisites

  • Node 12
  • node-gyp capable device

Package Sidebar

Install

npm i @presenti/server

Weekly Downloads

1

Version

1.2.6

License

MIT

Unpacked Size

392 kB

Total Files

148

Last publish

Collaborators

  • ericrabil