Restspace is a new take on a web application builder platform which encompasses the functionality of a CMS, an integration tool like Zapier, a backend as a service like Parse or Firebase and a low-code tool which avoids a lot of the pitfalls
such tools have for developers, particularly it is open and very easy to extend. As an example, let's say we want to set up some basic CMS functionality. Restspace gives you an empty web server initially. Then you can configure 'service components'
to run on base paths within that server's url space. So we would configure a Data service component to run on /json. The component lets you read and write JSON files by POSTing/GETing them from e.g. /json/page-data. Then we configure a Template service
component at /templates and add a template file in e.g. Nunjucks by POSTing it to /templates/page-template. Then we configure a Pipeline component on / whose purpose is to take an HTTP request and send it to the first of a series of URLs, getting the response
from that and sending it to the second etc. We would create a pipeline specification (as a JSON file) like this:
[
	"GET /json/page-data",
	"POST /templates/page-template"
]
So when a request is made to /, the Pipeline component handles it and reads the JSON data in the Data service, then sends this to the Template service, which poplates the page-template template with it and returns the result to the user. Note that the internal routing prefers the service with the most specific URL match. Restspace has built-in role-based access control which you can use to hide the /json and /templates paths, and put the page behind a login if you want.
Service components can provide a huge range of functionality e.g. static site hosting, external API wrapping e.g. email, SMS providers, user and account management, infrastructure services etc. and enable easy composition of these functionalities without code using pipelines. The use of Deno as the base runtime enables creation of new service components by simply publishing a JSON manifest and a JS/TS module to the web.
How to build a service component is explained here.
Restspace has an Admin front end which avoids having to do all this in curl or somesuch, providing editors for entering data, giving an experience usable by non-technical people.
The repo has many dependencies on Restspace Core. The simplest approach to working with them is to have the two repos installed in the same parent directory. Restspace Core contains all the necessary dependencies for custom service or adapter code: when writing your own service or adapter, you won't need any dependencies in Restspace Runtime.
The technical overview in the documentation for Restspace is at https://restspace.io/docs/overview/Technical%20Overview.
server.ts sets up and runs the http server. Code for handling requests is in handleRequest.ts.
The server code lazy-loads configuration for a tenant (tenant.ts) when it first encounters a url which maps to that tenant. This mapping is based on subdomain or mappings in serverConfig.json. This file also specifies an infra (see below) which is a configuration element that tells it how and from where to load the tenant configuration.
tenant.ts contains code to process a tenant configuration JSON (services.json), into an in-memory representation of the tenant. This process preloads the manifests of the services that are configured.
A manifest is a JSON specification of metadata about a service component. The built-in services have their manifests in services/ and these files normally have the standard suffix .rsm.json. In the built-in case, these are actually .js files to simplify loading. The manifest specifies such things as the name and description of the service, the location of the code file for the code of the service, the schema of its custom configuration etc.
Modules.ts contains a class that holds in memory the manifests of the services specified in tenants' configurations. On demand, it loads the code for the service from the filesystem or internet from the location specified in the manifest and holds it in memory. Deno makes this very straightforward compared to Node and this is the key reason for the choice of Deno as the underlying JS runtime.
Manifests can also be compositions of services as they can specify a pipeline to run either/both before and after the main service they refer to.
A service component provides an HTTP interface to functionality. It can optionally be configured to load an adapter which is a class with a standard interface that provides pluggable access to functionality provided by a range of infrastructure services. Again this is lazily loaded at runtime from local files or the internet, and it has its own manifest suffixed .ram.json. Adapters have their own configuration: this may include things like access keys or tokens to access the infrastructure service.
An infra is an adapter whose declared in the serverConfig.json file or [to add] in the services.json file with partial configuration. A service can be configured to use an infra rather than an adapter. This has the dual advantage that it requires less configuration, and that its preset configuration can be kept hidden from users with a lower level of authorisation who can still make use of the infrastructure it connects to.
A request is received at the server in server.ts. This is passed to handleRequest.js handleIncomingRequest(). This loads the relevant tenant if missing, then calls tenant.ts getMessageFunctionByUrl() which forwards to the same-named function in ServiceFactory.ts. This class exists for each tenant, holding the service configuration structures for that tenant. It analyses the base urls of the configured services on the tenant, and picks the most specific match if any.
Then ServiceFactory.ts creates an adapter for the service, attaches it to a ServiceContext (defined in rs-core). It finds the configuration data for the service. It creates a wrapped service using ServiceWrapper.ts which handles commmon pre and post processing for all services. Then it calls one of two functions on this class, internal or external. The former wraps the custom code for the service with standard processing needed when the call to the service is made from within the runtime e.g. as part of a pipeline. The latter wraps this in turn with common code for receiving and sending messages to the web.
external checks authentication and also manages CORS and caching headers. internal handles pre and post pipelines around the service. It calls the actual code of the service, passing in the HTTP request message, a context object with the adapter if any, plus some system functions including one which calls a local url on an internally, and the configuration data for the service.
The /test directory contains some tools which allow mocking an entire multitenant server configuration. handleIncomingRequest can then be called directly to run tests across nearly the whole system. Most of the test files provide examples of how to do this.
This directory also contains tests for adapter interfaces which should ensure any new adapter follows implicit behavioural assumptions about how it should present its functionality e.g. a store adapter should persist data written returning it when subsequently read from the same location.
In the /services directory are some core built-in services.
- auth.tssupplies authentication to other services, it has a special function- setUserwhich checks a user for an auth header or cookie and attaches the encoded user information to the request if present and correct. It has a built in role system with the core roles 'A' - administrator, 'E' - content editor and 'U' - site user.
- user-data.rsm.jsis a manifest that provides a data store service for user information with security features forbidding certain kinds of edits of sensitive information e.g. passwords. This works in conjunction with- auth.ts
- services.tsis an API to manage the tenant's configuration, normally only accessible to an authenticated administrator. The configuration can be changed at runtime with no special precautions, this being greatly simplified by JS being single-threaded. This is still a rather tricky undertaking and it is quite possible this may cause issues with edge cases where a request is blocked on an async call and has configuration changed under it. This also supplies the list of available services and will provide search functions on this catalogue. Additionally it supports 'chords' which are configuration fragments the list of which are composed into a consistent deduplicated single configuration every time it is changed.
- pipeline.tsprovides a full-featured facility for a request to be passed through a series of services, the response data from one being passed into the next as a request. Requests can be run in parallel and JSON data transformed between calls. This is the main composition primitive for Restspace.