Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tum-esi/shadow-thing

Creates and deploys a virtual Web of Things servient based on its Thing Description
https://github.com/tum-esi/shadow-thing

shadow virtual

Last synced: 3 months ago
JSON representation

Creates and deploys a virtual Web of Things servient based on its Thing Description

Awesome Lists containing this project

README

        

# Shadow Thing

![Node.js CI](https://github.com/tum-esi/shadow-thing/workflows/Node.js%20CI/badge.svg?branch=master)
![Workflow for all OS](https://github.com/tum-esi/shadow-thing/workflows/Workflow%20for%20all%20OS/badge.svg?branch=master)

Creates and deploys a Thing based on its TD

## Why use this software

- You want to create a mashup scenario or a test script for a WoT Thing that you only have a TD for. This tool allows you to simulate that Thing based only on its TD.
- You want to simulate a back-end logic of a WoT Thing without programming it. You can create a [Virtual Thing Description (VTD)][vtd] of that Thing and deploy it using this tool.
- You have a resource constrained device and you want to deploy its copy on a more powerful device to handle more Consumers (clients). You can deploy shadow-thing in twin mode.
- You want to change security scheme or the protocol of a device, without modifying it. You can deploy shadow-thing in twin mode.

## Prerequisites

All systems require:

- [NodeJS](https://nodejs.org/) version 10+ (e.g., 10.13.0 LTS)

### Linux

Meet the [node-gyp](https://github.com/nodejs/node-gyp#installation) requirements:

- Python 3.x
- make
- A proper C/C++ compiler toolchain, like GCC

### Windows

Install the Windows build tools through a CMD shell as administrator:

```
npm install -g --production windows-build-tools
```

### Mac OS

Meet the [node-gyp](https://github.com/nodejs/node-gyp#installation) requirements:

```
xcode-select --install
```

## How to start a shadow thing

### Install this package

Clone this repository and go into it:

```
git clone https://github.com/tum-esi/shadow-thing
cd shadow-thing
```

Install dependencies and build project:

```
npm install
npm run build
```

### Optional: create a link

Make the package available on your local machine (as a symlink). You can then use each package in its local version via `npm link ` instead of `npm install ` (see also https://docs.npmjs.com/cli/link).

```
npm link
```

This step also allows you to start a virtual-thing by just calling the command `shadow-thing` from anywhere within your computer, instead of having to call `node dist/cli.js` inside this package.

### Start with the default example TD

To get to know how the virtual-thing module works, you can start a virtual thing based on the default example TD provided.
Change directories to the root of this module with `cd` and run:

```
node dist/cli.js
```

or if you created a link, you can just call

```
shadow-thing
```

### Start a Shadow Thing based on any TD or VTD

You can create a shadow thing based on any given TD/VTD:

```
node dist/cli.js path/to/my/example_td.json
```

or if you created a symlink:

```
shadow-thing path/to/my/example_td.json
```

### Provide your own configuration file

By default, if no configuration file path is given as an argument, a default one is generated. Users have the possibility of changing the configuration by rejecting the default one when prompted.
For more flexibility, it is recommended to create a custom configuration file and pass the path as an argument to fit your specific requirements.

```
shadow-thing -c path/to/conf.json path/to/my/example_td.json
```

### Change the configuration

The configuration file is a JSON file that allows you to configure some aspect of the shadow thing. These include:
* Protocol parameters ( HTTP, CoaP or MQTT )
* Security related information for Basic Auth
* Logging levels
* Event Intervals
* Caching intervals of the properties if shadow-thing is used in twin mode
* Instances of different things

The configuration file format looks like this:

```JSON
{
"servient": {
"staticAddress": STATIC,
"http": {
"port": HPORT,
"serverKey":KEYLOCATION,
"serverCert":CERTIFICATELOCATION,
"security":{
"scheme":"basic"
}
}
},
"log": {
"level": LOGLEVEL
},
"things": {
"THING_ID": {
"eventIntervals": {
"EVENT_ID1": EVENTINTERVAL,
"EVENT_ID2": EVENTINTERVAL
},
"twinPropertyCaching": {
"PROPERTY_ID1": EVENTINTERVAL,
"PROPERTY_ID2": EVENTINTERVAL,
},
"credentials": {
"username": USERNAME,
"password": PASSWORD
}
}
}
}
```

For example, you can set-up the shadow-thing to generate a specific event every 60 seconds by replacing the value of EVENTINTERVAL of the specific event with 60.

You can also set the logging level between 0 and 4:
`{ error: 0, warn: 1, info: 2, log: 3, debug: 4 }`

You can also refer to the configuration file generated by default when first running virtual-thing to have a better idea.

### More Help:

If you need more help, run:

```
shadow-thing --help
```

## How to use the Digital Twin mode

### How to start a digital twin

To start a digital twin, use the `-t` or `--twin` command line options:

```
shadow-thing --twin path/to/real-thing/td.json
```

This will tell the shadow-thing to start in digital twin mode. To do so, it will consume the TD, and start a Thing instance that is supposed to act as a reverse proxy.
When this instance receives a request, it will try to pass it on to the real thing.
If this is not possible, it will generate a random response instead.
The response is annotated to make the source of the data clear.

It is also possible for the digital twin to be used as caching server for load balancing purposes. This is configurable in the config file.

### How to add a model to your digital twin

It is possible to use a model of your real thing in digital twin mode. This means that when the real thing is not reachable, the digital twin will use the model to create a response instead:

```
shadow-thing --twin path/to/real-thing/td.json::path/to/model.js
```

Upon reception of a property read request, and whenever the real Thing is unreachable, the digital twin will call on your model and pass on the last received property value, as well as its timestamp to it.
Based on those values, your model can return an approximate value, as well as an accuracy annotation ( from 0 to 100% ). This data is then sent as a response to the received request.

The model has to conform to a specific format. An example is provided under `examples/twin-models/coffee_machine_model.js`

## How to create a group of servers or clients

Users have the possibility to create a group of servers or clients based on a configuration file which must be provided. Examples of configuration files can be found under `config-files`.

### Servers

To launch a group of servers with default configuration file:

```
node dist/server-pool.js
```

To specify a specific configuration file :

```
node dist/server-pool.js -c path/to/server/config
```

Here is an example of a configuration file for the server pool.

```JSON
{
"mode": MODE,
"staticAddress": ADDRESS,
"servients": [
{
"instances": SERVER_INSTANCE,
"protocol": PROTOCOL,
"things":{
"path/to/td":{
"instances": THING_INSTANCE,
"eventIntervals": {
"EVENT_ID1": INTERVAL,
"EVENT_ID2": INTERVAL
}
}
}
}
]
}
```

`MODE` : specifies the mode in which the servients are created. Possible values are `'single'` for single-threaded or `'multi'` for multi-threaded.
`ADDRESS` : specifies a static IP address.
`SERVER_INSTANCE` : specifies the number of instances of the concerned servient to create.
`PROTOCOL` : specifies the protocol used by the servient. Possible values are `'http'` and `'coap'`.
`THING_INSTANCE` : specifies the number of things described by the TD provided as key to create and expose on the servient.
`INTERVAL` : specifies the interval between each emission of the concerned event.

A `server-pool` is able to contain instances of different things. To do so, specify configurations for multiple things under `"things"`.

### Clients

Group spawning of clients have the sole intention of testing the responsiveness of the servers. The clients will take measurements of the time between the moment when the request is sent and when the response is received. Once the number of measurements specified are taken, the script automatically ends and the clients are terminated. A directory containing csv files of the results is generated. A configuration file is also necessary.

To launch a group of clients with default configuration file :

```
node dist/client-pool.js path/to/put/results
```

To specify your own configuration file :

```
node dist/client-pool.js -c path/to/config path/to/put/results
```

Here is an example of a configuration file for the client pool:

```JSON
{
"clients": [
{
"instances": CLIENT_INSTANCE,
"protocol": PROTOCOL,
"thingURL": THING_URL,
"measures": NUM_MEASURES,
"events_to_sub": [EVENT_ID, EVENT_ID],
"actions_to_inv": {
"ACTION_ID1": INTERVAL
},
"prop_to_read": {
"PROP_ID1": INTERVAL
}
}
]
}
```

`CLIENT_INSTANCE` : specifies the number of instances of the concerned client to create.
`PROTOCOL` : specifies the protocol used. Possible values are `'http'` and `'coap'`. `THING_URL` : specifies the URL to fetch the TD of the concerned thing.
`NUM_MEASURES` : specifies the number of measures to take for each test.

To spawn multiple instances of different clients, add more configurations in the array value of the key `"clients"`.

## How to run tests for evaluation

### Automated Testing

Tests done by the `client-pool` can be automated by running :

```
node dist/auto-test.js
```

This command runs tests based on a configuration file which can be found in the `config-files` directory. It first generates all the corresponding configuration files for `server-pool` and `client-pool` and then executes them one by one. This happens in a single container.

Below is an example of how a configuration file for testing should look like:

```JSON
{
"modes": [], //applies only to server
"protocols": [],
"memory_limit": MEM_LIMIT,
"ports":{
"start": NUM_PORT,
"end": NUM_PORT
}, //applies only to server
"clients":{
"start": NUM_CLIENT,
"end": NUM_CLIENT
}, //applies only to client
"prop":{
"start": INTERVAL,
"end": INTERVAL,
"step": INTERVAL
}, //applies only to client
"action":{
"start": INTERVAL,
"end": INTERVAL,
"step": INTERVAL
}, //applies only to client
"event":{
"start": INTERVAL,
"end": INTERVAL,
"step": INTERVAL
}, //applies only to server
"nData": NUM_MEASURES, //applies only to client
"tdPath": PATH_TO_TD,
"thingInstance":{
"start": NUM_INSTANCE,
"end": NUM_INSTANCE,
"step": NUM_INSTANCE
} //applies only to server
}
```

#### Notes

* The configuration above is used to describe ranges when generating the corresponding configuration files.
* All the configuration files are generated in `tests/config`, a directory which will be created when the script is executed.
* Each test is identified by a number, and the corresponding results can be found in `tests/results`.
* It is only possible to use one TD for automatic testing. For more specific tests, the configuration has to be done manually.
* Objects without the property `"step"` in the configuration is incremented by one from one test to another.
* For automated tests, the client subscribes to all events, invokes all actions with the specified interval, and reads all properties at the specified interval.
* It is possible to run the tests in a docker container with `Dockerfile`.

### Container to Container

It is also possible to execute tests using the default configuration files `server-config.json` and `client-config.json` under the `config-files` directory using docker-compose. This uses the Docker files `Dockerfile.client` and `Dockerfile.server`.

```
docker-compose up
```

### Notes:

* When running tests between containers, the static address of the servers should be set to `server-pool`.
* To make custom tests, you should edit the default configuration files.
* The results of the tests will be located at `/app/results` in the client container.

### Docker Help

* Once the test finishes, the container will be shutdown automatically. To see the previously run 20 containers, use `docker container ls --last 20`.
* `docker build -t ege/test1 .` once the configuration is set, it will build the image that you can run
* `docker run -d --memory 10240m --cpuset-cpus="0-1" ege/test3` will run it with this given name but this is not the name of the container, that will be assigned automatically
* `docker cp stupefied_mendel:app/tests ./` to copy test to current folder
* `docker build -f NEWDOCKERFILE` allows you to pass other docker files

## Useful Links:

1. [Thing Description Specification](https://w3c.github.io/wot-thing-description)
2. [Scripting API Specification](https://w3c.github.io/wot-scripting-api/)
3. [node-wot implementation of the Scripting API](https://github.com/eclipse/thingweb.node-wot)
4. [Virtual Thing]([src/virtual_thing/documentation/specification/index.md])

## Continuous Integration Workflow with Github Actions

Every time a pull-request to master is done the CI workflow (currently including node.js installation and build) is triggered. This workflow only tests on a `linux` based Github hosted runner.

If a test on all available operating systems (`macOS`, `windows`, `linux`) is needed, another CI workflow for all OSs can be triggered manually.

To be able to do this, first a [personal access token](https://help.github.com/en/github/authenticating-to-github/creating-a-personal-access-token-for-the-command-line) needs to be generated. Then, this token needs to be added as _Secret_.
In the github repository go to `settings > secrets > 'add new secret'`. Give the secret a name (e.g. [your-name]_ACCESS_TOKEN ) and add `[your-username]:[your-private-access-token]` as value.

_Note: You can only do this if you have the required access rights._

Then you can trigger the 'Workflow for all OS' with the following command:

```
curl -X POST https://api.github.com/repos/tum-esi/shadow-thing/dispatches \
-H 'Accept: application/vnd.github.everest-preview+json' \
-H 'Authorization: token ' \
--data '{"event_type": "test-all-os"}'
```