- Documentation website
- Module documentation
- UI screenshots
- API documentation
- Environment configuration
- Introduction
- Import this module
- Provided server, content and proxy handlers
- Storage engine
- Customizing membership profiles
- Access the API
- Github repository
- NPM package
Dashboard bundles everything a web app needs, all the "boilerplate" like signing in and changing passwords, into a parallel server so you can write a much smaller web app.
The Organizations module allows users to create organizations and invitations other users can accept to join. Users must share the invitations themselves with the recipients. Your application server can use the Organizations module's API to fetch what organizations a user is in and use that data to allow shared access or assign ownership or whatever other purpose. A complete UI is provided for users to create and manage their organizations and memberships, and a basic administrator UI is provided for oversight.
On your Dashboard server you need to install this module with NPM:
$ npm install @layeredapps/organizations
Edit your package.json
to activate the module:
"dashboard": {
"modules": [
"@layeredapps/organizations"
]
}
Dashboard comes with some convenience scripts you can add to your package.json
:
Type | Script path | Description |
---|---|---|
proxy | @layeredapps/organizations/src/proxy/x-memberships.js | Dashboard will bundle the user's Membership objects in x-memberships header. |
proxy | @layeredapps/organizations/src/proxy/x-organizations.js | Dashboard will bundle the user's Organization objects in x-organizations header. |
server | @layeredapps/organizations/src/server/check-before-delete-organization.js | Require users complete steps, such as deleting subscriptions, before deleting their organization. Set a CHECK_BEFORE_DELETE_ORGANIZATION path such as /check-delete on your Application server, Dashboard will query this API passing ?organizationid=xxxxx and you may respond with { "redirect": "/your-delete-requirements" } or { "redirect": false }" to enforce the requirements. |
By default this module will share whatever storage you use for Dashboard. You can specify nothing, specify an alternate storage backend, or specify the same type with a separate database.
ORGANIZATIONS_STORAGE=postgresql
ORGANIZATIONS_DATABASE_URL=postgres://localhost:5432/organizations
If your Dashboard is configured with database read replication servers this module will follow that configuration. You can also specify module-specific read replication:
ORGANIZATIONS_STORAGE_REPLICATION=true
ORGANIZATIONS_DATABASE_URL=postgres://1.0.0.0:5432/organizations
ORGANIZATIONS_READ_DATABASE_URL1=postgres://1.0.0.1:5432/organizations
ORGANIZATIONS_READ_DATABASE_URL2=postgres://1.0.0.2:5432/organizations
ORGANIZATIONS_READ_DATABASE_URL#=postgres://1.0.0.3:5432/organizations
Memberships designate a Profile which you can configure to collect the information relevant to your organizations. You specify the fields you want in an environment variable:
MEMBERSHIP_PROFILE_FIELDS="any,of,the,below"
Field |
---|
display-name |
display-email |
contact-email |
full-name |
dob |
phone |
occupation |
location |
location |
company-name |
website |
Dashboard and official modules are completely API-driven and you can access the same APIs on behalf of the user making requests. You perform GET
, POST
, PATCH
, and DELETE
HTTP requests against the API endpoints to fetch or modify data. This example fetches the user's country information using NodeJS, you can do this with any language:
You can view API documentation within the NodeJS modules' api.txt
files, or on the documentation site.
const memberships = await proxy(`/api/user/organizations/memberships?accountid=${accountid}&all=true`, accountid, sessionid)
const proxy = util.promisify((path, accountid, sessionid, callback) => {
const requestOptions = {
host: 'dashboard.example.com',
path: path,
port: '443',
method: 'GET',
headers: {
'x-application-server': 'application.example.com',
'x-application-server-token': process.env.APPLICATION_SERVER_TOKEN,
'x-accountid': accountid,
'x-sessionid': sessionid
}
}
const proxyRequest = require('https').request(requestOptions, (proxyResponse) => {
let body = ''
proxyResponse.on('data', (chunk) => {
body += chunk
})
return proxyResponse.on('end', () => {
return callback(null, JSON.parse(body))
})
})
proxyRequest.on('error', (error) => {
return callback(error)
})
return proxyRequest.end()
})
}