Skip to content

Commit

Permalink
Merge pull request #1162 from SignalK/dynamic-component-loading
Browse files Browse the repository at this point in the history
feature: embeddable plugin/webapp components
  • Loading branch information
tkurki authored Nov 22, 2020
2 parents 6db116d + 6428217 commit 3cd9fc5
Show file tree
Hide file tree
Showing 40 changed files with 1,814 additions and 350 deletions.
76 changes: 62 additions & 14 deletions WEBAPPS.md
Original file line number Diff line number Diff line change
@@ -1,25 +1,73 @@
## Webapps
# Webapps and components

### 1. Introduction
## Introduction

A Signal K webapp is an HTML5 application that runs on a web browser and is typically loaded from a Signal K server, using its HTTP interface.
There are four ways to add web-based user interfaces to the Signal K Server:

### 2. Sample webapps
- Standalone Webapps
- Embedded Webapps
- Embedded Plugin Configuration Forms
- Embedded Compontents

There are several prototype level [browser based webapps that are installed during the server installation](http://localhost:3000/apps) such as:
- [Instrumentpanel](https://github.com/SignalK/instrumentpanel) - Show your boat's live data with customizable gauges you can hide or move around.
- [freeboard-sk](https://github.com/signalk/freeboard-sk) - Chart plotting webpage
- [sailgauge](https://github.com/signalk/sailgauge) - html page to show sailing-related data on one screen.
**Standalone Webapps** are Webapps that can be installed on the server via Appstore. The server provides a list of installed webapps. Once you navigate to them the server admin UI disappears and the webapp controls the whole page (browser window / tab).

Other apps are published on npm, and appear in the App Store (http://localhost:3000/appstore)
**Embedded Webapps** are installed and listed like Standalone Webapps, but they open **embedded in the server admin UI**, leaving the header and footer available so that the user can perform login, restart server and use the admin UI's sidebar to navigate to a different part of the admin UI.

### 3. Getting started with your own webapps
![vesselpositions](img/vesselpositions.png?raw=true "Vesselpositions Embedded Webapp")

The simplest form of a Signal K client is a process or web page that receives and displays the data in some way. The server contains a very simple web page that connects to the server with a WebSocket connection and updates the display as new json messages are received from the server. If you have installed the server and started it with one of the file-based startup scripts in the bin/ directory you can access it at [http://localhost:3000/examples/consumer-example.html](http://localhost:3000/examples/consumer-example.html). The [html code of the web page](https://github.com/SignalK/signalk-server-node/blob/master/public/examples/consumer-example.html) is in the examples directory.
**Embedded Plugin Configuration Forms** are related to server plugins. A plugin provides a schema for the configuration data it uses. The server uses the schema - a description of the structure of the data used to configure the plugin - to generate configuration forms for the installed plugins. The generated form is often lackin in usability due to it being totally generic. To address this a plugin can provide its own **Configuration Form** that the server embeds within the Plugin Configuration of the server admin UI.

If you want to integrate your webapp in the main structure of http://localhost:3000/apps, you should make it installable via npm and add the keyword `signalk-webapp` to your package.json file.
![calibration](img/calibration.png?raw=true "Calibration plugin configuration form")

**Embedded Components** are individual UI components provided by a plugin or a webapp. They are currently available at the bottom of the Webapps page of the admin UI. The idea with embedding components would be to allow a plugin to add individual components to different parts of the server, but this is more an idea than a fully implemented feature at this stage.

### 4. Going further
## Webapp/Component Structure

All different webapps (and server plugins) are installed with npm, from npm registry or for example from your own Github repository. Private plugins need not be published to npm - see the documentation for [npm install](https://docs.npmjs.com/cli/v6/commands/npm-install) for the exact details. Only webapps that are relevant for all users should be published in npm to be available in App store of everybody's server.

The basic structure of a webapp is
- directory named `public` that contains the actual webapp: html, JavaScript and resources such as images and css files. This directory is automatically mounted by the server so that the webapp is available via http once installed
- package.json with special keywords that classifies the webapp
- `signalk-webapp` - standalone webapp
- `signalk-embeddable-webapp` - embeddable webapp
- `signalk-plugin-configurator` - plugin configuration form

This structure is all that is needed for a standalone webapp.

There is no keyword for a module that provides only embedded components, use `signalk-webapp` instead.

The embedded components are implemented using [Webpack Federated Modules](https://webpack.js.org/concepts/module-federation/) and [React Code Splitting](https://reactjs.org/docs/code-splitting.html).

You need to configured Webpack to create the necessary code for federation using *ModuleFederationPlugin* and expose the component with fixed names:
- embeddable webapp: `./AppPanel`
- plugin configuration form: `./PluginConfigurationPanel`
- embedded component: `./AddonPanel`

The ModuleFederationPlugin library name must match the package name and be a "safe" name for a webpack module like in `library: { type: 'var', name: packageJson.name.replace(/[-@/]/g, '_') },`

The exposed modules need to `export default` a React component - both class based components and stateless functional components can be used. The server dependencies like `reactstrap` can and should be used. Add `@signalk/server-admin-ui-dependencies` as a dependency to the webapp, it defines the depedencies used by the server admin UI.

See the vesselpositions embedded webapp/component and Calibration plugin for examples of each. It is probably easier to start with either one and modify them to suit your needs. Don't forget to change the module id and name in package.json!


## Webapp / Component and Admin UI / Server interfaces

Standalone Webapps can use the server's APIs (Standard Signal K http and WebSocket APIS as well as the server's endpoints) but they need to implement everything themselves.

Embedded Webapps, Components and Plugin Configuration Forms work inside the Admin UI and they can interact with the Admin UI and the server with APIs exposed by the Admin UI as component properties.

This documentation is rudimentary on purpose, as the details need to be worked out.

Embedded webapp properties
- access to the login status of the browser user
- ability to render Login form instead of the webapp content
- getting and setting application data
- opening an automatically reconnecting WebSocket connection to the server
- getting Signal K data via `get`
- [Embedded](packages/server-admin-ui/src/views/Webapps/Embedded.js)

PluginConfigurationForm properties
- `configuration` : the configuration data of the plugin
- `save`: function to save the configuration data
- [EmbeddedPluginConfigurationForm](packages/server-admin-ui/src/views/Configuration/EmbeddedPluginConfigurationForm.js)

Have an idea for a great way to visualise Signal K data? Something bugging you about the sample consumers? [Read the documentation](https://signalk.org/specification/master/), share your input, write some code, [join the mailing list](https://groups.google.com/forum/#!forum/signalk) and our [Slack chat](http://slack-invite.signalk.org/)!
Binary file added img/calibration.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added img/vesselpositions.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
30 changes: 30 additions & 0 deletions packages/server-admin-ui-dependencies/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@

require('validate-peer-dependencies')(__dirname, {
resolvePeerDependenciesFrom: './',
handleFailure(result) {
let { missingPeerDependencies, incompatibleRanges } = result;

let missingWithVersions = (missingPeerDependencies || []).reduce(
(message, metadata) => {
return `${message}${metadata.name}@${metadata.specifiedPeerDependencyRange} `;
},
''
);

let incompatiblePeerDependenciesMessage = (incompatibleRanges || []).reduce(
(message, metadata) => {
return `${message}\n\t* ${metadata.name}: \`${metadata.specifiedPeerDependencyRange}\`; it was resolved to \`${metadata.version}\``;
},
''
);

if (missingWithVersions.length >0) {
console.error((`Please INSTALL MISSING PEERDEPENDENCIES with
npm install --save-dev ${missingWithVersions}`))
process.exit(-1)
}

console.error((`Incompatible peerDependencies: ${incompatiblePeerDependenciesMessage}`))
process.exit(-1)
},
})
91 changes: 91 additions & 0 deletions packages/server-admin-ui-dependencies/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

24 changes: 24 additions & 0 deletions packages/server-admin-ui-dependencies/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
{
"name": "@signalk/server-admin-ui-dependencies",
"version": "1.0.0",
"description": "UI dependencies of the admin UI exposed to plugins that expose federated components",
"main": "index.js",
"author": "[email protected]",
"license": "Apache-2.0",
"optionalDependencies": {
"validate-peer-dependencies": "^1.1.0"
},
"peerDependencies": {
"@fortawesome/fontawesome-free": "^5.15.1",
"bootstrap": "^4.5.3",
"font-awesome": "^4.7.0",
"react": "^16.13.1",
"react-dom": "^16.13.1",
"react-redux": "^5.1.2",
"react-select": "^3.1.0",
"reactstrap": "^5.0.0",
"redux": "^3.7.2",
"redux-thunk": "^2.3.0",
"simple-line-icons": "^2.5.5"
}
}
6 changes: 4 additions & 2 deletions packages/server-admin-ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,15 @@
"@babel/core": "^7.11.6",
"@babel/preset-react": "^7.10.4",
"@fortawesome/fontawesome-free": "^5.15.1",
"@signalk/server-admin-ui-dependencies": "file:../server-admin-ui-dependencies/signalk-server-admin-ui-dependencies-1.0.0.tgz",
"ansi-to-html": "^0.6.14",
"babel-loader": "^8.1.0",
"bootstrap": "^4.5.3",
"copy-webpack-plugin": "^6.2.1",
"css-loader": "^5.0.0",
"file-loader": "^6.1.1",
"font-awesome": "^4.7.0",
"html-webpack-plugin": "^4.5.0",
"html-webpack-plugin": "^5.0.0-alpha.6",
"jsonlint-mod": "^1.7.6",
"lodash.get": "^4.4.2",
"lodash.remove": "^4.7.0",
Expand All @@ -38,14 +39,15 @@
"react-router-dom": "^4.3.1",
"react-select": "^3.1.0",
"reactstrap": "^5.0.0",
"reconnecting-websocket": "^4.4.0",
"redux": "^3.7.2",
"redux-thunk": "^2.3.0",
"sass-loader": "^6.0.7",
"simple-line-icons": "^2.5.5",
"style-loader": "^2.0.0",
"webpack": "^5.0.0",
"webpack-bundle-analyzer": "^3.9.0",
"webpack-cli": "^4.0.0"
"webpack-cli": "^4.2.0"
},
"scripts": {
"prepublishOnly": "npm run clean && npm run build",
Expand Down
3 changes: 3 additions & 0 deletions packages/server-admin-ui/public_src/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,9 @@
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
<meta name="description" content="Signal K Node Server">

%ADDONSCRIPTS%

<link rel="shortcut icon" href="img/favicon.ico" >
<title>Signal K Server</title>
</head>
Expand Down
2 changes: 2 additions & 0 deletions packages/server-admin-ui/src/actions.js
Original file line number Diff line number Diff line change
Expand Up @@ -128,13 +128,15 @@ export const buildFetchAction = (endpoint, type, prefix) => dispatch =>
export const fetchLoginStatus = buildFetchAction('/loginStatus', 'RECEIVE_LOGIN_STATUS')
export const fetchPlugins = buildFetchAction('/plugins', 'RECEIVE_PLUGIN_LIST')
export const fetchWebapps = buildFetchAction('/webapps', 'RECEIVE_WEBAPPS_LIST')
export const fetchAddons = buildFetchAction('/addons', 'RECEIVE_ADDONS_LIST')
export const fetchApps = buildFetchAction('/appstore/available', 'RECEIVE_APPSTORE_LIST')
export const fetchAccessRequests = buildFetchAction('/security/access/requests', 'ACCESS_REQUEST')
export const fetchServerSpecification = buildFetchAction('/signalk', 'RECEIVE_SERVER_SPEC', '')

export function fetchAllData (dispatch) {
fetchPlugins(dispatch)
fetchWebapps(dispatch)
fetchAddons(dispatch)
fetchApps(dispatch)
fetchLoginStatus(dispatch)
fetchServerSpecification(dispatch),
Expand Down
Loading

0 comments on commit 3cd9fc5

Please sign in to comment.