Download OpenAPI specification:Download
aivis Anomaly Detection is one of the engines of the aivis Technology Platform by Vernaio and a component of multiple products such as aivis Insights, Process Booster, and more. aivis Anomaly Detection is designed to be highly sensitive to various kinds of abnormal behavior, while minimizing at the same time required user inputs.
Abnormal behavior of your system is detected early on as aivis Anomaly Detection is adapted specifically to your system. Instead of comparing signal values to standard or manufacturer specific thresholds, aivis Anomaly Detection learns these from your own data. Instead of monitoring each signal or component independently, aivis Anomaly Detection always takes into account the present production setting and state. The user only needs to tag some signal as target signal. Then, aivis Anomaly Detection automatically detects all other relevant signals that are related to this target signal. By considering these signals, the expected range of the target signal can be narrowed down, and already small deviations from normal behavior can be detected. Problems therefore can be notified early. Using revolutionary new mathematical concepts of multivariate & non-linear statistics (i.e., Geometrical Kernel Machines), aivis Anomaly Detection runs fast and stable and is accurate already with little training data. Many anomaly detection models can be trained simultaneously without any hassle, and complicated systems can be monitored.
During Training, the engine generates a model trained on historical data consisting of normal operation. Before the actual training, data preparation steps are taken automatically, such as time synchronization, automatic feature engineering, and selection. The resulting model is then used during Inference: given the current signal values it returns a score indicating how normal or abnormal the current signal states are.
This documentation explains the usage and principles behind aivis Anomaly Detection to data and software engineers. For detailed API descriptions of docker images, web endpoints and SDK functions, please consult the reference manual of the regarding component:
SDKs
Docker Images
For additional support, go to Vernaio Support.
The workflow consists of 2 steps:
Currently, aivis Anomaly Detection is distributed to a closed user base only. To gain access to the artifacts, as well as for any other questions, you can open a support ticket via aivis Support.
We consider a factory with different machines and production processes. Most of the time, everything works smoothly. However, from time to time, various issues may arise.
To make explanations more concrete, let's assume we want to monitor the extrusion of synthetic foils. This is a complex process that generates a lot of process data. Sensors watch the production process, measuring for example temperatures, torques, or electric currents. Other sensors verify quality characteristics of the product, for example transparency. The same machine is used to produce different foils, for example foils of different thickness. Therefore, also the machine configuration needs to be tracked. Due to the overall complexity it is impossible for machine operators to monitor each detail of the production without a powerful tool. However, it is important to detect any problems soon. Rapid countermeasures can often help to avoid machine failure or minimize the production of defective goods. aivis Anomaly Detection learns very fast how your signals behave and can therefore readily be applied to new machines or production processes. It notifies you early if any abnormal behavior occurs and therefore allows you to run your processes more reliably. Besides the automatic learning process, a major advantage is that aivis Anomaly Detection infers about anomalies by taking into account the relationships between many signals. Looking comprehensively at many signals often provides much better information than looking at each signal separately.
In the following, we will discuss the use of aivis Anomaly Detection based on some production plant example data. The data was sourced from Kaggle and is shared under the creative commons terms. Apart from a simple shift and rescaling of the time variable, the data itself was not altered. Here, we are going to discuss some results obtained from these data. In the Getting Started sections below, you will learn how to obtain these results.
The model always provides a score between 0 and 1. High scores indicate the presence of abnormal behavior. The example data was chosen because the inferred anomaly score is a good starting point to explain several typical behaviors:
The figure shows several patterns. Very short peaks, such as the ones on 12 June may occur if the monitored system switches between different states. In synthetic foil production, the machine may be switched to produce another foil thickness. During most of the time, machines run under steady conditions. Switches between different states are usually fast and may be rare. Therefore, the behavior of the machine during the switch is not part of normal behavior. This leads to peaks in the score. During live monitoring, you know when you switch the machine state and such peaks do not bother.
The next pattern is a slowly increasing anomaly score that falls back to zero at 14 June. As this pattern persists for a longer time, clearly some signal behavior changes compared to the training period. A slow increase is indicative for progressive detoriation of some, e.g. mechanical defect. Again, the very short peak at the end of the slow increase can be interpreted as sudden change of operating conditions. Depending on the operating conditions, the anomaly may be more easy or more difficult to detect. For example, a mechanical defect becomes less eminent under lower load. This can explain why the anomaly score is quite small during about the next day, and then starts to rise again on a lower level.
On 16 and 17 June the anomaly score fluctuates between 0 and 0.2. An anomaly score in this range means that the observed behavior is slightly different to the behavior in the training data. It occurs for example if the training data did not cover all possible machine configurations and external factors. Whether or not such increase can already be taken as first warning, depends on the stability of the monitored system. By evaluation on historic data that was not part of the training, you can learn about the typical range of the anomaly score for your system.
Finally, a clear anomaly is building up in the second half of the plotted time interval.
If you integrate an aivis Anomaly Detection anomaly score into your monitoring system, you may define your own threshold to trigger alarms or notifications. In this example, even an (unnecessarily) high threshold of 0.8 would translate to an early warning period of more than four days.
Another example is the monitoring of an aivis Signal Prediction model. To be able to predict some signal, aivis Signal Prediction has to be trained on some training data. Inferences are made on some other data. While aivis Signal Prediction always returns the best estimate, this estimate may be misguided if the training data does not fit the inference data. In particular, the machine may be run in an operating mode that was never used during the training data. Other possible reasons include the issues listed above. Therefore, aivis Anomaly Detection has a mode specifically to monitor the input of an aivis Signal Prediction model, see also the section on the modeling configuration.
The SDK of aivis Anomaly Detection allows for direct calls from your Python, Java or C program code. All language SDKs internally use our native shared library (FFI). As C APIs can be called from various other languages as well, the C-SDK can also be used with languages such as R, Go, Julia, Rust, and more. Compared to the docker images, the SDK enables a more fine-grained usage and tighter integration.
In this chapter we will show you how to get started using the SDK.
A working sdk example that builds on the code explained below can be downloaded directly here:
This zip file contains example code for docker
, python
and java
in respective subfolders. All of them use the same dataset which is in the data subfolder.
Required artifacts:
.whl
-files which you will receive in a libs.zip directly from aivis Support:aivis_engine_v2_ad_runtime_python_full-{VERSION}-py3-none-win_amd64.whl
: An anomaly detection full python runtimeaivis_engine_v2_base_sdk_python-{VERSION}-py3-none-any.whl
: The base python sdk aivis_engine_v2_ad_sdk_python-{VERSION}-py3-none-any.whl
: The anomaly detection python sdk aivis_engine_v2_toolbox-{TOOLBOX-VERSION}-py3-none-any.whl
: The toolbox python sdk - optional for HTML report generationPreparations:
>=3.10
) installation.AIVIS_ENGINE_V2_API_KEY
and assign the licensing key to it.anomaly-detection-examples.zip
. The data CSVs train_ad.csv
and eval_ad.csv
need to stay in **/data
. libs.zip
. These .whl
-files need to be in **/libs
.The folder now has the following structure:
+- data
| +- train_ad.csv
| +- eval_ad.csv
|
+- docker
| +- # files to run the example via docker images, which we will not need now
|
+- java
| +- # files to run the example via java sdk, which we will not need now
|
+- libs
| +- # the .whl files to run aivis
|
+- python
| +- # files to run the example via python sdk
Running the example code:
**/python
subfolder. Here, you find the classic python script example_ad.py
and the jupyter notebook example_ad.ipynb
.
Both run the exact same example and output the same result. Choose which one you want to run. .whl
files. We will now explain two options, which are installing them via pip install or installing them via poetry. Many other options are also possible, of course.Option A: pip install (only for the classic python script example_ad.py
, not for the jupyter notebook example_ad.ipynb
)
**/python
subfolder and run the following commands: # installs the `.whl` files
pip install -r requirements-<platform>.txt
# runs the classic python script `example_ad.py`
python example_ad.py --input=../data --output=output
Option B: poetry install
# installs poetry (a package manager)
python -m pip install poetry
# installs the `.whl` files
poetry install --no-root
# runs the classic python script `example_ad.py`
poetry run python example_ad.py --input=../data --output=output
**/python
subfolder. The first one might take a while, the third one opens a tab in your browser. # installs the `.whl` files
poetry install --no-root
# installs jupyter kernel
poetry run ipython kernel install --user --name=test_ad
# runs the jupyter python script `example_ad.ipynb`
poetry run jupyter notebook example_ad.ipynb
After running the scripts, you will find your computation results in **/python/output
.
Additionally to the anomaly-detection-examples.zip
you just downloaded, you need the following artifacts. To gain access, you can open a support ticket via aivis Support.
Required artifacts:
.jar
files which you will receive in a libs.zip directly from aivis Support:aivis-engine-v2-ad-runtime-java-full-win-x8664-{VERSION}.jar
: An anomaly detection full java runtime, here for windows, fitting your operating system - see artifacts for other options on linux and macos. aivis-engine-v2-base-sdk-java-{VERSION}.jar
: The base java sdk aivis-engine-v2-ad-sdk-java-{VERSION}.jar
: The anomaly detection java sdk Preparations:
>=11
) installation.AIVIS_ENGINE_V2_API_KEY
and assign the licensing key to it.anomaly-detection-examples.zip
. The data CSVs train_ad.csv
and eval_ad.csv
need to stay in **/data
. libs.zip
. These .jar
-files need to be in **/libs
.The folder now has the following structure:
+- data
| +- train_ad.csv
| +- eval_ad.csv
|
+- docker
| +- # files to run the example via docker images, which we will not need now
|
+- java
| +- # files to run the example via java sdk
|
+- libs
| +- # the .jar files to run aivis
|
+- python
| +- # files to run the example via python sdk, which we will not need now
Running the example code:
**/java
subfolder. Here, you find the build.gradle
. Check, if the paths locate correctly to your aivis engine v2 .jar files in the **/libs
subfolder. **/java
subfolder and run the following commands: # builds this Java project with gradle wrapper
./gradlew clean build
# runs Java with parameters referring to input and output folder
java -jar build/libs/example_ad.jar --input=../data --output=output
After running the scripts, you will find your computation results in **/java/output
.
Our SDK artifacts come in two flavours:
full
packages provide the full functionality and are available for mainstream targets only:win-x8664
macos-armv8
* (SDK >= 11.0) 2.3macos-x8664
* (SDK >= 11.0; until aivis engine version 2.9.0) 2.3linux-x8664
(glibc >= 2.14)inf
packages contain only API functions regarding the inference of a model. As lightweight artifacts they are available for a broader target audience:win-x8664
macos-armv8
* (SDK >= 11.0) 2.3macos-x8664
* (SDK >= 11.0; until aivis engine version 2.9.0) 2.3linux-x8664
(glibc >= 2.14)linux-armv7
(glibc >= 2.18; until aivis engine version 2.9.0)linux-armv8
(glibc >= 2.18; until aivis engine version 2.9.0)linux-ppc64
(glibc >= 2.18; until aivis engine version 2.2.0)* Only Python and C SDKs are supported. Java SDK is not available for this target.
In this chapter we want to demonstrate the full API functionality and thus always use the full
package.
To use the Python-SDK you must download the SDK artifact (flavour and target generic) for your pythonpath at build time. Additionally at installation time, the runtime artifact must be downloaded with the right flavour and target.
The artifacts are distributed through a PyPI registry.
Using Poetry you can simply set a dependency on the artifacts specifying flavour and version. The target is chosen depending on your installation system:
aivis_engine_v2_ad_sdk_python = "{VERSION}"
aivis_engine_v2_ad_runtime_python_{FLAVOUR} = "{VERSION}"
The SDK supports the full API and will throw a runtime exception if a non-inference function is invoked with an inference-flavoured runtime.
To use the Java-SDK, you must download at build time:
It is possible to include multiple runtime artifacts for different targets in your application to allow cross-platform usage. The SDK chooses the right runtime artifact at runtime.
The artifacts are distributed through a Maven registry.
Using Maven, you can simply set a dependency on the artifacts specifying flavour, version and target:
<dependency>
<groupId>com.vernaio</groupId>
<artifactId>aivis-engine-v2-ad-sdk-java</artifactId>
<version>{VERSION}</version>
</dependency>
<dependency>
<groupId>com.vernaio</groupId>
<artifactId>aivis-engine-v2-ad-runtime-java-{FLAVOUR}-{TARGET}</artifactId>
<version>{VERSION}</version>
<scope>runtime</scope>
</dependency>
Alternativly, with Gradle:
implementation 'com.vernaio:aivis-engine-v2-ad-sdk-java:{VERSION}'
runtimeOnly 'com.vernaio:aivis-engine-v2-ad-runtime-java-{FLAVOUR}-{TARGET}:{VERSION}'
The SDK supports the full API and will throw a runtime exception if a non-inference function is invoked with an inference-flavoured runtime.
To use the C-SDK, you must download the SDK artifact at build time (flavour and target generic). For final linkage/execution you need the runtime artifact with the right flavour and target.
The artifacts are distributed through a Conan registry.
Using Conan, you can simply set a dependency on the artifact specifying flavour and version. The target is chosen depending on your build settings:
aivis-engine-v2-ad-sdk-c/{VERSION}
aivis-engine-v2-ad-runtime-c-{FLAVOUR}/{VERSION}
The SDK artifact contains:
include/aivis-engine-v2-ad-core-full.h
The runtime artifact contains:
lib/aivis-engine-v2-ad-{FLAVOUR}-{TARGET}.lib
bin/aivis-engine-v2-ad-{FLAVOUR}-{TARGET}.dll
(also containing the import library)lib/aivis-engine-v2-ad-{FLAVOUR}-{TARGET}.so
(also containing the import library)The runtime library must be shipped to the final execution system.
A valid licensing key is necessary for every aivis calculation in every engine and every component.
It has to be set (exported) as environment variable AIVIS_ENGINE_V2_API_KEY
.
aivis will send HTTPS requests to https://v3.aivis-engine-v2.vernaio-licensing.com
(before release 2.7: https://v2.aivis-engine-v2.vernaio-licensing.com
, before release 2.3: https://aivis-engine-v2.perfectpattern-licensing.de
) to check if your licensing key is valid. Therefore, requirements are an active internet connection as well as no firewall blocking an application other than the browser calling this url.
If aivis returns a licensing error, please check the following items before contacting aivis Support:
<FirstPartOfKey>.<SecondPartOfKey>
with first and second part being UUIDs. In particular, there must be no whitespace. https://v3.aivis-engine-v2.vernaio-licensing.com
in your browser. The expected outcome is "Method Not Allowed". In that case, at least the url is not generally blocked. Before we can invoke API functions of our SDK, we need to set it up for proper usage and consider the following things.
It is important to ensure the release of allocated memory for unused objects.
In Python, freeing objects and destroying engine resources like Data-, Training- and Inference-objects is done automatically. You can force resource destruction with the appropriate destroy function.
In Java, freeing objects is done automatically, but you need to destroy all engine resources like Data-, Training- and Inference-objects with the appropriate destroy function. As they all implement Java’s AutoClosable
interface, we can also write a try-with-resource
statement to auto-destroy them:
try(final AnomalyDetectionData trainingData = AnomalyDetectionData.create()) {
// ... do stuff ...
} // auto-destroy when leaving block
In C, you must always
Errors and exceptions report what went wrong on a function call. They can be caught and processed by the outside.
In Python, an Exception
is thrown and can be caught conveniently.
In Java, an AbstractAivisException
is thrown and can be caught conveniently.
In C, every API function can write an error to the given output function parameter &err
(to disable this, just set it to NULL
). This parameter can then be checked by a helper function similar to the following:
const Error *err = NULL;
void check_err(const Error **err, const char *action) {
// everything is fine, no error
if (*err == NULL)
return;
// print information
printf("\taivis Error: %s - %s\n", action, (*err)->json);
// release error pointer
aivis_free(*err);
*err = NULL;
// exit program
exit(EXIT_FAILURE);
}
Failures within function calls will never affect the state of the engine.
The engine emits log messages to report on the progress of each task and to give valuable insights. These log messages can be caught via registered loggers.
# create logger
class Logger(EngineLogger):
def log(self, level, thread, module, message):
if (level <= 3):
print("\t... %s" % message)
# register logger
AnomalyDetectionSetup.register_logger(Logger())
// create and register logger
AnomalyDetectionSetup.registerLogger(new EngineLogger() {
public void log(int level, String thread, String module, String message) {
if (level <= 3) {
System.out.println(String.format("\t... %s", message));
}
}
});
// create logger
void logger(const uint8_t level, const char *thread, const char *module, const char *message) {
if (lvl <= 3)
printf("\t... %s\n", message);
}
// register logger
aivis_setup_register_logger(&logger, &err);
check_err(&err, "Register logger");
During the usage of the engine, a lot of calculations are done. Parallelism can drastically speed things up. Therefore, set the maximal threads to a limited number of CPU cores or set it to 0
to use all available cores (defaults to 0
).
# init thread count
AnomalyDetectionSetup.init_thread_count(4)
// init thread count
AnomalyDetectionSetup.initThreadCount(4);
// init thread count
aivis_setup_init_thread_count(4, &err);
check_err(&err, "Init thread count");
Now that we are done setting up the SDK, we need to create a data store that holds our historical Training Data. In general, all data must always be provided through data stores. You can create as many as you want.
After the creation of the data store, you can fill it with signal data. The classic way to do it is writing your own reading function and adding signals, i.e. lists of data points, to the data context yourself, as it is shown in Data Reader Options.
We recommend to use the built-in files reader, which processes a folder with csv files that have to follow the CSV Format Specification.
We assume, that the folder path/to/input/folder/
contains train_ad.csv
.
# create empty data context for training data
training_data = AnomalyDetectionData.create()
# create config for files reader
files_reader_config = json.dumps(
{
"folder": "path/to/input/folder/"
}
)
# read data
training_data.read_files(files_reader_config)
# ... use training data ...
// create empty data context for training data
try(final AnomalyDetectionData trainingData = AnomalyDetectionData.create()) {
// create config for files reader
final DtoTimeseriesFilesReaderConfig trainingFilesReaderConfig = new DtoTimeseriesFilesReaderConfig("path/to/input/folder/");
// read data
trainingData.readFiles(trainingFilesReaderConfig);
// ... use training data ...
} // auto-destroy training data
// create empty data context for training data
TimeseriesDataHandle training_data = aivis_timeseries_data_create(&err);
check_err(&err, "Create training data context");
// create config for files reader
const char *reader_config = "{"
"\"folder\": \"path_to_input_folder\""
"}";
// read data
aivis_timeseries_data_read_files(training_data, (uint8_t *) reader_config, strlen(reader_config), &err);
check_err(&err, "Read Files");
// ... use training data ...
// destroy data context
aivis_timeseries_data_destroy(training_data, &err);
check_err(&err, "Destroy data context");
training_data = 0;
In the following, we will assume you have read in the file train_ad.csv
shipped with the example project. In particular we assume some signal has been added with signal ID L-1
.
With the data store filled with historical Training Data, we can now create our training:
# build training config
training_config = json.dumps({
"target": {
"signal": "L-1"
}
})
# create training and train the model
training = AnomalyDetectionTraining.create(training_data, training_config)
# ... use training ...
// build training config
final DtoTrainingConfig trainingConfig = new DtoTrainingConfig(
new DtoTargetConfig("L-1"));
// create training and train the model
final AnomalyDetectionTraining training = AnomalyDetectionTraining.create(trainingData, trainingConfig) {
// ... use training ...
} // auto-destroy training
// build training config
const char *training_config = "{"
"\"target\": {"
"\"signal\": \"L-1\""
"}"
"}";
// create training and train the model
AnomalyDetectionTrainingHandle training_handle = aivis_anomaly_detection_training_create(
training_data,
(uint8_t *) training_config,
strlen(training_config),
&err
);
check_err(&err, "Create training");
// ... use training ...
// destroy training
aivis_anomaly_detection_training_destroy(training_handle, &err);
check_err(&err, "Destroy training");
training_handle = 0;
For the moment, you may take the training_config as it is. The different keys will be explained in the training section, and the reference manual.
After the training has finished, we can evaluate it by running a historical evaluation (bulk inference) on the inference data (out-of-sample). This way, we obtain a continuous stream of values — exactly as it would be desired by the machine operator.
As we do the inference in the same process with the training, we can create the inference directly from the training. If these two processes were separated we could get the model explicitly from the training and write it to a file. The inference could then be created based on the content of the model file.
# build inference config
inference_config = json.dumps({"skipOnInsufficientData": True})
# create inference
inference = AnomalyDetectionInference.create_by_training(training, inference_config)
# ... use inference ...
// build inference config
final DtoInferenceConfig inferenceConfig = new DtoInferenceConfig(true);
// create inference
try(final AnomalyDetectionInference inference = AnomalyDetectionInference.createByTraining(training, inferenceConfig)) {
// ... use inference ...
} // auto-destroy inference
// build inference config
const char *inference_config = "{\"skipOnInsufficientData\": true}";
// create inference
AnomalyDetectionInferenceHandle inference_handle = aivis_anomaly_detection_inference_create_by_training_handle(
training_handle,
(uint8_t *) inference_config,
strlen(inference_config),
&err
);
check_err(&err, "Create inference");
// ... use inference ...
// destroy inference
aivis_anomaly_detection_inference_destroy(inference_handle, &err);
check_err(&err, "Destroy inference");
inference_handle = 0;
Finally, we want to infer some anomaly scores for a list of Inference Timestamps. Therefore we again need to provide a filled data store which this time holds our Inference Data, just in the same way as we created our Training Data store. We then invoke the appropriate infer function.
Note that the infer with next normal function is an alternative, experimental function to help identifying the causes of observed anomalies, see the section on Infer With Next Normal.
# choose inference timestamps
timestamps = ...
# infer anomaly scores
scores = inference.infer(inference_data, timestamps)
# choose timestamps for next normal, possibly a subset of the upper list of timestamps
timestamps_next_normal = ...
# ... use scores e.g. for plotting ...
// choose inference timestamps
final List<Long> timestamps = ...
// infer anomaly scores
final List<DtoFloatDataPoint> scores = inference.infer(inferenceData, timestamps);
// ... use scores e.g. for plotting ...
// choose inference timestamps
Time *timestamps = ...
// infer anomaly scores
const List_DtoFloatDataPoint *scores = aivis_anomaly_detection_inference_infer(
inference_handle,
inference_data,
timestamps,
timestamps_len,
&err
);
check_err(&err, "Infer Scores");
// ... use scores e.g. for plotting ...
// free scores
aivis_free(scores);
scores = NULL;
// free timestamps
free(timestamps);
timestamps = NULL;
Using the file eval_ad.csv
shipped with the Example Project, you may obtain the anomaly scores visualized above.
Next, we will do the same calculations with direct function calls via an SDK.
The docker images of aivis Anomaly Detection are prepared for easy usage. They use the SDK internally, but have a simpler file-based interface. If you have a working docker workflow system like Argo, you can build your own automated workflow based on these images.
In this chapter, we will show you how to get started using docker images. Usage of the SDK will be covered in the next chapter.
A working example that builds on the code explained below can be downloaded directly here: anomaly-detection-examples.zip.
This zip file contains example code for docker, python and java in respective subfolders. All of them use the same dataset which is in the data subfolder.
Prerequisites: Additionally to the anomaly-detection-examples.zip
you just downloaded, you need the following artifacts. To gain access, you can open a support ticket via aivis Support.
aivis-engine-v2-ad-training-worker
, aivis-engine-v2-ad-inference-worker
and (optionally for HTML report generation) aivis-engine-v2-toolbox
As a Kubernetes user even without deeper Argo knowledge, the aivis-engine-v2-example-ad-argo.yaml
shows best how the containers are executed after each other, how training and inference workers are provided with folders that contain the data csvs and how the toolbox assembles a HTML report at the end.
There are 3 different docker images:
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-training-worker:{VERSION}
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-inference-worker:{VERSION}
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-inference-service:{VERSION}
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-incremental-learning-worker:{VERSION}
All docker images are Linux-based.
You need an installation of Docker on your machine as well as access to the engine artifacts:
docker -v
docker pull {REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-training-worker:{VERSION}
docker -v
docker pull {REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-training-worker:{VERSION}
A valid licensing key is necessary for every aivis calculation in every engine and every component.
It has to be set (exported) as environment variable AIVIS_ENGINE_V2_API_KEY
.
aivis will send HTTPS requests to https://v3.aivis-engine-v2.vernaio-licensing.com
(before release 2.7: https://v2.aivis-engine-v2.vernaio-licensing.com
, before release 2.3: https://aivis-engine-v2.perfectpattern-licensing.de
) to check if your licensing key is valid. Therefore, requirements are an active internet connection as well as no firewall blocking an application other than the browser calling this url.
If aivis returns a licensing error, please check the following items before contacting aivis Support:
<FirstPartOfKey>.<SecondPartOfKey>
with first and second part being UUIDs. In particular, there must be no whitespace. https://v3.aivis-engine-v2.vernaio-licensing.com
in your browser. The expected outcome is "Method Not Allowed". In that case, at least the url is not generally blocked.
First, we need to train the model (workflow step 1: Training) using the Training Worker.
At the beginning, we create a folder docker
, a subfolder training-config
and add the configuration file config.yaml
:
data:
folder: /srv/data
dataTypes:
defaultType: FLOAT
training:
target:
signal: "L-1"
output:
folder: /srv/output
Keys data and output are relevant for input and output, and more information can be found in the docker reference manual. Possible subkeys of training will be explained in the training section, and the docker reference manual.
As a next step, we create a second folder data
and add the Training Data CSV file train_ad.csv
to the folder.
Afterwards, we create a blank folder output
.
Our folder structure should now look like this:
+- docker
| +- training-config
| +- config.yaml
|
+- data
| +- train_ad.csv
|
+- output
Finally, we can start our training via:
docker run --rm -it \
-v $(pwd)/docker/training-config:/srv/conf \
-v $(pwd)/data/train_ad.csv:/srv/data/train_ad.csv \
-v $(pwd)/output:/srv/output \
-e AIVIS_ENGINE_V2_API_KEY={LICENSE_KEY} \
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-training-worker:{VERSION}
docker run --rm -it `
-v ${PWD}/docker/training-config:/srv/conf `
-v ${PWD}/data/train_ad.csv:/srv/data/train_ad.csv `
-v ${PWD}/output:/srv/output `
-e AIVIS_ENGINE_V2_API_KEY={LICENSE_KEY} `
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-training-worker:{VERSION}
After a short time, this should lead to two output files in the output
folder:
training-report.json
can be inspected to get information about dependencies between the various signals.model.json
holds all model information for the following Inference.
After the training has finished, we can evaluate it by running a historical evaluation (bulk inference) on the second data file. For evaluation, we use a new data set in order to check how the engine performs on unseen data ("out-of-sample" data). The config file below instructs the Inference Worker to create a prediction each minute between start time and end time. All times are expressed as UNIX Timestamps in milliseconds. With this config file we obtain a continuous stream of values — exactly as it would be desired by the machine operator.
For this, we create a second subfolder inference-config
of the data
folder and add the configuration file config.yaml
:
data:
folder: /srv/data
dataTypes:
defaultType: FLOAT
inference:
config:
skipOnInsufficientData: true
modelFile: /srv/output/model.json
timestamps:
- _type: Equidistant
startTime: 1465632000000
endTime: 1466786700000
interval: 60000
output:
folder: /srv/output
Note that there is also a different, experimental prediction method infer with next normal to help identifying the causes of observed anomalies, see section Infer With Next Normal.
After that, we add the Inference Data CSV file eval_ad.csv
to the data
folder.
Our folder structure should now look like this:
+- docker
| +- training-config
| +- config.yaml
| +- inference-config
| +- config.yaml
|
+- data
| +- train_ad.csv
| +- eval_ad.csv
|
+- output
Finally, we can run the Inference via:
docker run --rm -it \
-v $(pwd)/docker/inference-config:/srv/conf \
-v $(pwd)/data/eval_ad.csv:/srv/data/eval_ad.csv \
-v $(pwd)/output:/srv/output \
-e AIVIS_ENGINE_V2_API_KEY={LICENSE_KEY} \
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-inference-worker:{VERSION}
docker run --rm -it `
-v ${PWD}/docker/inference-config:/srv/conf `
-v ${PWD}/data/eval_ad.csv:/srv/data/eval_ad.csv `
-v ${PWD}/output:/srv/output `
-e AIVIS_ENGINE_V2_API_KEY={LICENSE_KEY} `
{REGISTRY}/{NAMESPACE}/aivis-engine-v2-ad-inference-worker:{VERSION}
Successful execution should lead to the file scores.json
in the output
folder, which holds the calculated anomaly scores.
The results correspond to the trend plotted above.
Such historical evaluation can best be achieved with the Inference Worker.
For continuous live monitoring the Inference Service may be preferable as it offers a RESTful API to trigger predictions via HTTP in contrast to the Inference Worker, which uses a file based API.
In the course of using aivis, large amounts of data are ingested. This chapter explains the terminology as well as the required format, quality and quantity.
Most aivis engines work on time series data that is made up of signals. Every signal consists of two things, these being
timestamp
and availability
. The ID needs to be unique within the data.The data points for one or more signals for a certain detection time range are called a history.
The values of a signal can be boolean values, 64-bit Floating Point numbers or Strings. Non-finite numbers (NAN
and infinity) and empty strings are regarded as being unknown and are therefore skipped.
Points in time are represented by UNIX Timestamps in milliseconds (64-bit Integer). This means the number of milliseconds that have passed since 01.01.1970 00:00:00 UTC.
The point in time that a signal value belongs to is called the Detection Timestamp. This usually is the timestamp when the measurement originally has taken place. If the measurement is a longer offline process, it should refer to the point in time at which the measured property was established, e.g. the time point of sample drawing or the production time for delayed sampling. In case of the target signal, the Detection Timestamp should be set to the time you would have liked to have measured the signal online. In the aivis Signal Prediction example use case, the paper quality is such a signal. It is measured around 2 hours after the production of the paper in a laboratory and must be backdated to a fictitious, but instantaneous quality measurement in the process.
Different signals may have different Detection Timestamps. Some might have a new value every second, some every minute, some just when a certain event happens. aivis automates the process of synchronizing them internally. This includes dealing with holes in the data.
When doing a historical evaluation, we want to know what the engine would have inferred/predicted for a list of Inference Timestamps that lie in the past (Inference Timestamps are the moments for which you want to get an inference). For a realistic inference, the engine must ignore all signal values that were not yet available to the database at the Inference Timestamp. A good example for such a case is a measurement, that is recorded by a human. The value of this measurement will be backdated by him/her to the Detection Timestamp, but it took e.g. 5 minutes to extract the value and report it to the system. So, it would be wrong to assume that one minute after this fictitious Detection Timestamp, the value would have been already available to the Inference. Another example case is the fully automated lagged data ingestion of distributed systems (especially cloud systems).
There are multiple ways to handle availability. Which strategy you use depends an the concrete use case.
To allow for these different strategies, every data point can have an additional Availability Timestamp that tells the system when this value became available or would have been available. Signal values for which the Availability Timestamp lies after the Inference Timestamp are not taken it into account for an inference at this Inference Timestamp.
If there is no knowledge about when data became available, the Availability Timestamp can be set to the Detection Timestamp — but then you must keep in mind that your historical evaluation might look better as it could have been in reality.
aivis works best on raw, unprocessed data. It is important to keep the following rules in mind:
SignalInactive
trigger, since the engine learns how often the signal gets a new data point.SignalInactive
).There is the possibility of filtering the Training Data in multiple ways:
The filtering is configurable:
This means that two models could be trained on the same data set, but on different time windows or signal sets. Alternatively, the user can of course also restrict the data that enters the engine beforehand.
aivis uses data at two distinct points in the workflow:
When making an inference, aivis must know the current state of the real system by including a small portion of history.
In Training, the engine calculates which signals among the many signals in the Training Data will be relevant for the Inference. Furthermore, for each relevant signal a time window is specified relative to the Inference Timestamp. This time window determines which values of the signal must be included in order to make a prediction for said timestamp. This doesn't only include the values within the time window, but also either the value right at the start, or the last value before the time window (see "Nearest Predecessor"). This information is called the Inference Data Specification and must be obeyed strictly when triggering Inference, as the engine relies on this data.
You can inspect a model for its Inference Data Specification.
It is possible to set the maximum amount of time to be included in the local history. This is done in the configuration of the Training via the parameter Maximal Lag.
The following diagram gives you a visual representation of how an Inference Data Specification could look like:
In the diagram you see that a start lag and end lag is specified for every signal. For the Inference, this means that for each signal we need all data points whose detection timestamps lie in the window [ inference timestamp - start lag; inference timestamp - end lag ]
as well as the nearest predecessor (see below).
As previously mentioned, it is essential that you provide data for the whole time window. Especially, it must be clear what the value at the beginning is, i.e. at inference timestamp - start lag
.
Typically, there is no measurement for exactly this point in time. Then, you must provide the nearest predecessor. This is the last value before the beginning of the time window. Then, the engine can at least take this value as an estimate. Of course this first data point must also be available at the Inference Timestamp (regarding the Availability Timestamp).
Depending on the configuration, the engine will either throw an error or ignore timestamps for which you provide neither a value at the beginning of the time window nor a nearest predecessor. This implies that you always need at least one available value per relevant signal. Sending more data outside the demanded time window will have no effect on the inference, though.
For docker users, the only option to read data is providing data files following the CSV Format Specification, which are processed by the built-in files reader.
Since 2.10, SDK users also have the option to use the built-in files reader, which we recommend: The reader has been highly optimized and is e.g. capable of deterministic parallel reading. It is also equipped with various filtering options, so that unneeded data is not even added to the data context, which saves memory.
We assume, that the folder path/to/input/folder/
contains csvs following the CSV Format Specification. Two of them (with the id product_type
and color
) are string signals, which has to be explicitly listed in the file reader's config, since the default is to assume all signal types being float.
# create empty data context, replace <<Engine>> by the engine you're using
data_context = <<Engine>>Data.create()
# create config for files reader
files_reader_config = json.dumps(
{
"folder": "path/to/input/folder/",
"dataTypes": {
"stringSignals": ["product_type", "color"]
}
}
)
# read data
data_context.read_files(files_reader_config)
// create empty data context, replace <<Engine>> by the engine you're using
try(final <<Engine>>Data dataContext = <<Engine>>Data.create()) {
// create config for files reader
final DtoTimeseriesFilesReaderConfig filesReaderConfig = new DtoTimeseriesFilesReaderConfig("path/to/input/folder/")
.withDataTypes(new DtoTimeseriesDataTypesConfig().withStringSignals({"product_type", "color"}));
// read data
dataContext.readFiles(filesReaderConfig);
} // auto-destroy data
// create empty data context
TimeseriesDataHandle data_context = aivis_timeseries_data_create(&err);
check_err(&err, "Create data context");
// create config for files reader
const char *reader_config = "{"
"\"folder\": \"path_to_input_folder\"",
"\"dataTypes"\: {"
"\"stringSignals\": [\"product_type\", \"color\"]"
"}"
"}";
// read data
aivis_timeseries_data_read_files(data_context, (uint8_t *) reader_config, strlen(reader_config), &err);
check_err(&err, "Read Files");
// ... use data ...
// destroy data context
aivis_timeseries_data_destroy(data_context, &err);
check_err(&err, "Destroy data context");
data_context = 0;
You can directly process any data source of your choice by writing your own reader. You build data points and add them to the data context, which looks like this:
# create empty data context, replace <<Engine>> by the engine you're using
data_context = <<Engine>>Data.create()
# add sample data
data_context.add_float_signal("signal-id", [
DtoFloatDataPoint(100, 1.0),
DtoFloatDataPoint(200, 2.0),
DtoFloatDataPoint(300, 4.0),
])
# ... use data context ...
// create empty data context, replace <<Engine>> by the engine you're using
try(final <<Engine>>Data dataContext = <<Engine>>Data.create()) {
// add sample data
dataContext.addFloatSignal("signal-id", Arrays.asList(
new DtoFloatDataPoint(100L, 1.0),
new DtoFloatDataPoint(200L, 2.0),
new DtoFloatDataPoint(300L, 3.0),
));
// ... use data context ...
} // auto-destroy data context
// create empty data context
TimeseriesDataHandle data_context = aivis_timeseries_data_create(&err);
check_err(&err, "Create data context");
const DtoFloatDataPoint points[] = {
{100, 1.0},
{200, 2.0},
{300, 4.0},
};
// add sample data
aivis_timeseries_data_add_float_signal(data_context, "signal-id", &points[0], sizeof points / sizeof *points, &err);
check_err(&err, "Adding signal");
// ... use data context ...
// destroy data context
aivis_timeseries_data_destroy(data_context, &err);
check_err(&err, "Destroy data context");
data_context = 0;
Above we have filled the data store with three hard coded data points to illustrate the approach. Usually you will read in the data from some other source with your own reading function.
Please also note that the data reading is the place where decisions about signal data types are made. The code above obviously adds a float signal, which is analogous for the other data types.
All artifacts (SDKs and docker images) have a built-in data reader which can process CSV files. As the CSV format is highly non-standardized, we will discuss it briefly in this section.
CSV files must be stored in a single folder specified in the config under data.folder. Within this folder the CSV files can reside in an arbitrary subfolder hierarchy. In some cases (e.g. for HTTP requests), the folder must be passed as a ZIP file.
General CSV rules:
CR LF
/LF
). In other words, each record must be on its own line.Special rules:
timestamp
and contain the Detection Timestamp as UNIX Timestamps in milliseconds (64-bit Integer)availability
. This contains the Availability Timestamp in the same format as the Detection Timestamp.timestamp
or availability
, are interpreted as signals.Boolean values must be written in one of the following ways:
true
/false
(case insensitive)1
/0
1.0
/0.0
with an arbitrary number of additional zeros at the endRegular expression: (?i:true)|(?i:false)|1(\.0+)?|0(\.0+)?
Numbers are stored as 64-bit Floating Point numbers. They are written in scientific notation like -341.4333e-44
, so they consist of the compulsory part Significand and an optional part Exponent that is separated by an e
or E
.
The Significand contains one or multiple figures and optionally a decimal separator .
. In such a case, figures before or after the separator can be ommited and are assumed to be 0
. It can be prefixed with a sign (+
or -
).
The Exponent contains one or multiple figures and can be prefixed with a sign, too.
The 64-bit Floating Point specification also allows for 3 non-finite values (not a number, positive infinity and negative infinity) that can be written as nan
, inf
/+inf
and -inf
(case insensitive). These values are valid, but the engine regards them as being unknown and they are therefore skipped.
Regular expression: (?i:nan)|[+-]?(?i:inf)|[+-]?(?:\d+\.?|\d*\.\d+)(?:[Ee][+-]?\d+)?
String values must be encoded as UTF-8. Empty strings are regarded as being unknown values and are therefore skipped.
timestamp,availability,SIGNAL_1,SIGNAL_2,SIGNAL_3,SIGNAL_4,SIGNAL_5
1580511660000,1580511661000,99.98,74.33,1.94,true,
1580511720000,1580511721000,95.48,71.87,-1.23,false,MODE A
1580511780000,1580511781000,100.54,81.19,,1e-5,MODE A
1580511840000,1580511841000,76.48,90.01,2.46,0.0,MODE C
...
Previous sections gave an introduction on how to use aivis Anomaly Detection and also shed some light on how it works. The following sections will provide a more profound background. It is not necessary to know this background to use aivis Anomaly Detection! However, you may find convenient solutions for specific problems, such as special data kinds or restrictions, or ways to reduce training computation time. The following sections are organized in the natural order of the workflow. By workflow, we mean the cycle of data preparation, model training, and finally utilizing it by making inferences with it. It will become clear that only minimal user input is required for this workflow. Nevertheless, the user has the option to control the process with several input parameters which will be presented below.
An aivis Anomaly Detection model is always specific to a certain signal, called the target signal. For a complicated system, it is usually advisable to have several anomaly models in place. This is no problem given the low computational demands. Therefore, you should have some idea about which signals you consider most important to monitor. As a rule of thumb, you should pick signals from each part of your system, and include signals of main interest such as quality of your product. As it is not of major importance which signals exactly to pick, you may choose those which are most easy to interpret. For each target signal, the relationships of other signals to this signal are monitored for unusual behavior. This means that several signals are included in an anomaly model with a focus on those related to the target. After model training, training results can be inspected to decide whether there are parts of the system not yet sufficiently covered by the anomaly models, and additional models are necessary.
The training comprises all the steps to reach an anomaly model. aivis relieves the user of many steps that are typically associated with the machine learning process. Some of the steps are performed automatically, others are rendered superfluous. While domain knowledge can be employed, this is not a necessity. The following section will illuminate these topics in more detail. It is explained how the raw data is transformed, additional features are extracted automatically or manually, and relevant features are identified. In the following sections, the model configuration parameters are explained in some detail. Here we focus on explaining the underlying concepts. When it comes to actually making the inputs, syntactical questions will be answered by the aivis reference manuals, which define the exact inputs that can be made depending on whether you are using one of the SDKs or a Docker container.
The requirements for the input data that were formulated in section Data Specification serve the purpose of making it possible for aivis to read the data. Typically, several additional steps are required to make the data appropriate to be fed into a machine learning algorithm. Among others, these include:
All of the above is handled by aivis automatically. Here, data preparation steps that go beyond anything described in the “Data Specification” section are not necessary and even discouraged as they may alter the underlying information. Synchronization is not necessary as aivis treats each signal separately as a time series, and this also eliminates the need for imputation of missing values. Standardization is a routine task that is executed internally (if necessary at all). Outliers don’t need to be removed beforehand or cut back thanks to the way the aivis engine builds its models. Some minimal outlier handling might still be beneficial as will be explained below.
When the data is brought into a form that can directly be used for model building, it is referred to as “features”. Sometimes, signals are noisy and fluctuate a lot, while the change in what the signal actually is measuring may be only very small. To reduce such noise, a common approach would be to calculate a moving mean and use it as a feature. This is just one example for data aggregation over some time interval but there may be other cases, involving e.g. the maximum of some value in a certain time range, or similar. In aivis, such feature engineering using moving time windows is not necessary. Here, it pays off that aivis understands the data as time series and automatically extracts powerful features. Again, this is explained in more detail below.
As mentioned above, aivis takes over many time-consuming data preparation and feature engineering tasks. As the aivis training algorithm is very powerful, in most cases the automatically generated features already suffice to obtain an optimum model. For some special kinds of signals such as angles, audio data, or categorical data, there are built-in interpreters that take care of the relevant pecularities. Nevertheless, there may still be situations for which the model performance improves adding manually engineered features. This is facilitated by the expression language.
The philosophy of aivis is that the user does not need to care about which signals may be relevant for a specific target. Therefore, typically you may just always insert all signals. From each signal, several features are auto-engineered, e.g. to reflect the behavior of the signal within some relevant time window. This means that we have many features that can all potentially be used for model training. Sensitivity of the model, however, is somewhat reduced with a higher number of features included. Therefore, the relevant features are selected automatically, thus not requiring any domain knowledge.
Finding the relevant signals is done with the help of calculating the distance correlations of the features to the target. The distance correlation is a measure of how much two vectors are related to each other. For example, if the distance correlation between some feature and the target signal equals 0, then the feature and the target signal are statistically independent. Therefore, nothing can be concluded from this feature. On the other hand, if the distance correlation between some feature and the target signal equals 1, the target signal could be perfectly predicted by this signal alone. Therefore, in this way signals which are (practically) independent of the target are cleaned out automatically in the analysis.
Even if some feature (distance) correlates with the target signal, it does not necessarily mean that it adds relevant information. Often, two or several features behave so similarly that they effectively become redundant. In this case, the more important ones are kept, thus enabling a more effective training stage.
Hereinafter, we will go back to using the word “signal” instead of “feature”. While there are subtle differences, in most cases the terms can be used interchangeably.
In addition to feature engineering and cleaning, another preparative step is segmentation. Segmentation is a distinctive virtue of aivis, and – depending on the data – can prove to be very advantageous. During segmentation, the data is checked for homogeneity of the relationship between signals and the target. If it is observed that the relation between data and target differs notably on subsets of the data, different segments are formed. Maybe you produce three different products using the same machines. The machines may then run under different operating conditions. This is an obvious example where segmentation may prove useful. Note, however, that in general things are more intricate. On the one hand, a single segment might suffice even for this case: Even though the signals may take different values for the different operating modes, their interrelations may still be the same. On the other hand, having several segments quite frequently turns out to be helpful. This fully applies also to cases for which there is no a priori reason to expect several different behaviors.
After having generated and selected the best features and clustered the data into different segments, the last step of the training is model building. This step finally produces a model: something that is capable of calculating an anomaly score for given live signal histories. In a mathematical sense, it's a multi-dimensional function from a set of signal and target features into the interval from 0 to 1. If the signal histories are similar to those in the training set, the anomaly score will be small, close to 0. If the signal histories strongly deviate from anything seen in the training set, the anomaly score will be large, close to 1. However, as the live data will practically never exactly reproduce the training set, the challenge is to benchmark the deviation. In aivis Anomaly Detection deviations are benchmarked based on the density of training samples and the expected impact of the deviation on the target signal. In practice, how exactly the model works, is not relevant for the the user, as all that stays behind the scenes.
While the previous section was concerned with what aivis does automatically, the following section explains the parameters the user may configure to control the training. We start with an overview of all kinds of possible configuration keys. We stress that the vast majority of the keys is optional. A minimal training configuration was used above in SDK training, respectively, in Docker training. This example may mainly serve as a quick reference. The meaning of the different keys is explained in the following sections, and a definition of the syntax is given in the reference manuals.
training:
dataFilter:
startTime: 1464732000000
endTime: 1465631880000
excludeSignals:
- signal: L-8
startTime: 1465000000000
endTime: 1465631880000
# includeSignals: ... similar
includeRanges:
- startTime: 1464732000000
endTime: 1464863188000
- startTime: 1464932000000
endTime: 1465631880000
# excludeRanges: ... similar
target:
signal: L-1
removeOutliers: false
lagging:
maximalLag: 300000
minimalLag: 180000
mesh: 60000
sampling:
additionalSampleMesh: 30000
maximalSampleCount: 100000
operativePeriods:
signal: MY_BOOLEAN_OPERATIVE_SIGNAL
signals:
- signal: B-1
forceRetain: true
- signal: L-2
lagging:
maximalLag: 300000
minimalLag: 0
mesh: 30000
- signal: B-2
interpreter:
_type: Cyclic
cycleLength: 5.0
- signal: B-3
interpreter:
_type: Oscillatory
windowLength: 1200000
mesh: 60000
- signal: MY_CATEGORICAL_SIGNAL
interpreter:
_type: Categorical
lagging:
maximalLag: 300000
minimalLag: 60000
mesh: 60000
modeling:
checkTargetAtInference: true
controlPointCount: 2500
enableIncremental: true
training_config = json.dumps({
"dataFilter": {
"startTime": 1464732000000,
"endTime": 1465631880000,
"excludeSignals": [{
"signal": "L-8",
"startTime": 1465000000000,
"endTime": 1465631880000
# "includeSignals" : ... similar
}],
"includeRanges" : [{
"startTime": 1464732000000,
"endTime": 1464863188000
},{
"startTime": 1464932000000,
"endTime": 1465631880000
}],
# "excludeRanges" : ... similar
},
"target": {
"signal": "L-1",
"removeOutliers": False,
"lagging": {
"maximalLag": 300000,
"minimalLag": 180000,
"mesh": 60000
}
},
"sampling": {
"additionalSampleMesh": 30000,
"maximalSampleCount": 100000
},
"operativePeriods": {
"signal": "MY_BOOLEAN_OPERATIVE_SIGNAL"
},
"signals": [{
"signal": "B-1",
"forceRetain": True
}, {
"signal": "L-2",
"lagging": {
"maximalLag": 300000,
"minimalLag": 0,
"mesh": 30000
}}, {
"signal": "B-2",
"interpreter": {
"_type": "Cyclic",
"cycleLength": 5.0
}}, {
"signal": "B-3",
"interpreter": {
"_type": "Oscillatory",
"windowLength": 1200000,
"mesh": 60000
}}, {
"signal": "MY_CATEGORICAL_SIGNAL",
"interpreter": {
"_type": "Categorical"
}}],
"lagging": {
"maximalLag": 300000,
"minimalLag": 60000,
"mesh": 60000
},
"modeling": {
"checkTargetAtInference": True,
"controlPointCount": 2500,
"enableIncremental": True
}
})
final DtoTrainingConfig trainingConfig = new DtoTrainingConfig(
new DtoTargetConfig("L-1")
.withRemoveOutliers(false)
.withLagging(new DtoTargetLaggingConfig(300000L, 180000L, 60000L))
)
.withDataFilter(new DtoDataFilter()
.withStartTime(1464732000000L)
.withEndTime(1465631880000L)
.withExcludeSignals(new DtoDataFilterRange[] {
new DtoDataFilterRange("L-8")
.withStartTime(1465000000000)
.withEndTime(1465631880000L)
})
// .withIncludeSignals ... similar
.withIncludeRanges(new DtoInterval[]{
new DtoInterval()
.withStartTime(1464732000000L)
.withEndTime(1464863188000L),
new DtoInterval()
.withStartTime(1464932000000L)
.withEndTime(1465631880000L)
})
// .withExcludeRanges ... similar
)
.withSampling(new DtoSamplingConfig()
.withAdditionalSampleMesh(30000L)
.withMaximalSampleCount(100000)
)
.withOperativePeriods(new DtoOperativePeriodsConfig("MY_BOOLEAN_OPERATIVE_SIGNAL"))
.withSignals(new DtoSignalConfig[] {
new DtoSignalConfig("B-1")
.withForceRetain(true),
new DtoSignalConfig("L-2")
.withLagging(new DtoSignalLaggingConfig(300000L, 30000L)
.withMinimalLag(0L)),
new DtoSignalConfig("B-2")
.withInterpreter(new DtoCyclicSignalInterpreter(5.0)),
new DtoSignalConfig("B-3")
.withInterpreter(new DtoOscillatorySignalInterpreter(1200000L,60000L)),
new DtoSignalConfig("MY_CATEGORICAL_SIGNAL")
.withInterpreter(new DtoCategoricalSignalInterpreter()),
})
.withLagging(new DtoLaggingConfig(300000L, 60000L)
.withMinimalLag(60000L))
.withModeling(new DtoModelingConfig()
.withCheckTargetAtInference(true)
.withControlPointCount(2500)
.withEnableIncremental(true));
const char *training_config = "{"
"\"dataFilter\": {"
"\"startTime\": 1464732000000,"
"\"endTime\": 1465631880000,"
"\"excludeSignals\": [{"
"\"signal\": \"L-8\","
"\"startTime\": 1465000000000,"
"\"endTime\": 1465631880000"
"}],"
// "\"includeSignals\": ... similar
"\"includeRanges\": [{"
"\"startTime\": 1464732000000,"
"\"endTime\": 1464863188000"
"}, {"
"\"startTime\": 1464932000000,"
"\"endTime\": 1465631880000"
"}]"
// "\"excludeRanges\": ... similar
"},"
"\"target\": {"
"\"signal\": \"L-1\","
"\"removeOutliers\": false,"
"\"lagging\": {"
"\"maximalLag\": 300000,"
"\"minimalLag\": 180000,"
"\"mesh\": 60000"
"}"
"},"
"\"sampling\": {"
"\"additionalSampleMesh\": 30000,"
"\"maximalSampleCount\": 100000"
"},"
"\"operativePeriods\": {"
"\"signal\": \"MY_BOOLEAN_OPERATIVE_SIGNAL\""
"},"
"\"signals\": [{"
"\"signal\": \"B-1\","
"\"forceRetain\": true"
"}, {"
"\"signal\": \"L-2\","
"\"lagging\": {"
"\"maximalLag\": 300000,"
"\"minimalLag\": 0,"
"\"mesh\": 30000"
"}}, {"
"\"signal\": \"B-2\","
"\"interpreter\": {"
"\"_type\": \"Cyclic\","
"\"cycleLength\": 5.0"
"}}, {"
"\"signal\": \"B-3\","
"\"interpreter\": {"
"\"_type\": \"Oscillatory\","
"\"windowLength\": 1200000,"
"\"mesh\": 60000"
"}}, {"
"\"signal\": \"MY_CATEGORICAL_SIGNAL\","
"\"interpreter\": {"
"\"_type\": \"Categorical\""
"}}],"
"\"lagging\": {"
"\"maximalLag\": 300000,"
"\"minimalLag\": 60000,"
"\"mesh\": 60000"
"},"
"\"modeling\": {"
"\"checkTargetAtInference\": true,"
"\"controlPointCount\": 2500,"
"\"enableIncremental\": true"
"}"
}";
The following sections list and explain the parameters the user may configure to control the training. The sections are organized along the structure of the configuration classes.
The data filter allows you define the signals and time range that are used for training. Concretely, the data filter allows you to choose signals for training. This can be done by either of the following ways: exclude signals, or, alternatively, provide a list of signal names to include (include signals). Beyond that, the data filter allows you to determine the time range of data that is used for training, and even to include or exclude separate time ranges for specific signals.
There are several situations for which data filters come in handy. The global end time may be used to split the data set into a training set and an inference data set. As the name suggests, the training set is used to train a model. The inference data set may afterwards be used to assess the model’s quality. Typically, 10 – 15 % of the data are reserved for evaluation. It is important to make sure that the model does not use any data from the inference data set during training. In fact, it should at no point have access to any additional information that is not present under productive circumstances. This is necessary to optimize the model to the real situation, and to assure that model performance tests are meaningful.
As for selecting the global start time, having more data from a longer time range is almost always advantageous. This allows aivis to get a clear picture of the various ways of how signal behaviors influence the target. That being said, the time range that is chosen should be representative for the time period for which predictions are to be made. If some process was completely revised at your industry, this may have affected signal values or relationships between the signals. In such a case, it is advisable to include sufficient data from the time after revision. For major revisions, it might even be preferable to restrict only to data after that revision. Such restriction can easily be done using the start time.
It is also possible to include/exclude several time intervals globally, instead of just selecting/excluding one global time interval. This is carried out using the fields include ranges and exclude ranges 2.5. It is important to understand the way in which the global and signal based includes/excludes interact. When include signals and include ranges are set to given values different than 'None', first the signals available in include signals are taken with their respective time ranges, and only then those time ranges are intersected with the global time ranges defined in include ranges. The opposite is valid for exclude ranges and exclude signals, where we first consider the time ranges excluded globally, and the time ranges of the signals contained in exclude signals are united to the global ones.
Analogous to the global start time and end time, such intervals can also be specified for individual signals. Note, however, that it is usually advisable to apply time constraints globally.
Data excluded by data filters is still available for the expression language but does not directly enter the model building. Control on signal selection for model building is provided by the signal configuration. Finally, note that complementary to the data filter, training periods can conveniently be defined by operative periods.
Perhaps the most important input parameter is the signal you want to monitor. We call this signal the target signal. In most cases, it will simply be one of the signals in your data. However, it is also possible to define a target that is a combination of multiple signals using various mathematical functions and operators using the Expression Language, explained in detail in “Appendix 1: Expression Language”. The target signal is used to inform the engine about which part of the data to focus on. In particular, aivis Anomaly Detection monitors whether the relations between the target and other signals have changed. Note, that signals that are not related to the target are rejected and are therefore not monitored. Therefore, it is necessary to train and apply several anomaly models to monitor a complex system, see also the sections on preparation and training output. Alternatively, it is also possible to prohibit rejection of individual signals, see the section on signal configuration.
By default, the relation of the target to itself is not monitored. However, for some applications it is relevant to monitor the most recent value of the target to its values in the past. This can be achieved by setting a target lagging configuration. More details on lagging configurations can be found below. If the target was synthesized via the Expression Language, signals that enter the expression are treated similar to the target itself. In particular, the histories of these signals are not taken into account by default. This can be overruled by setting signal specific lagging configurations, see the section on signal configurations.
As mentioned before, there is no need to remove outliers in the features, since the way that aivis Anomaly Detection operates on the data provides little leverage to outliers. However, this does not apply to the target signal. By default, outliers in the target signal are ignored. This makes sense if outliers relate to measurement problems or special situations which are not of interest. On the other hand, if outliers are to be considered as part of normal behavior, they need to be included in the analysis and the default behavior can be switched off.
Usually, all data points are used for training for which there is a target value in the data. This may be more than necessary. Setting the maximal sample count helps you to prevent excessive training durations by limiting the number of target points that are used. For example, say you have a data set with a target signal that contains 2 million values. Using all of these values for training could lead to an excessive computational effort. By default, this number is reduced to 500,000. To speed up the computations, this can further be reduced. Good results are often achieved already using just 100,000 sample points or even less. This works thanks to aivis Anomaly Detection automatically choosing the most informative sample points. This is done by making sure that the whole spectrum of behavioral patterns is covered. If the maximum sample count is set to a number that is larger than the actual number of target signal values in the data, all sample points are considered.
On the other hand, the number of training points may also be too low. In general, this is only possible by acquiring more data. However, there is a notable exception for which additional sample points can easily be added: if you know that the target hardly changes between the timestamps for which target values are provided. In this case, additional samples can be generated. The distance in time between these additional samples is controlled by the additional sample mesh. Note, however, that the new samples only add information if some relevant signal changes between samples. If the data contains information on the target for timestamps t1, t2, t3, ... and information on other signals exists only for those timestamps, it does make little sense to calculate additional samples.
Sometimes, the target includes data points for which a prediction is not desired. A typical situation is the time during which a machine is off, possibly including warm up and cool down phases. Also for maintenance time, typically no prediction is desired. To restrict the model training to times of interest, a signal may be assigned to be the operative signal. An operative signal must be boolean. Then, training is restricted to target timestamps for which the operative signal is “true”. Often, there is no such signal in the raw data but it may be easy to derive the operative times from other signals. For example some motor may be stopped when production is off. If motor speed is above a certain threshold, this may be used to define operative periods. For such situations, an operative signal may easily be created with help of the Expression Language.
The signal configuration is the place to pass additional information about feature signals in order to enforce a special treatment. Each signal configuration refers to one specific signal.
At the core of the signal configuration is the interpreter. The interpreter defines which features are built from a signal and how these enter the engine. The features produced by an interpreter we call the different aspects of the signal. Very often the default configuration is the best choice and you don't need to set any interpreter. However, it is important to configure here any non-standard behavior as it may strongly affect the outcome. Below you find a table on the different interpreters, followed by some more in-depth explanations.
Interpreter | Short Explanation | Examples |
---|---|---|
Default | Corresponds to a numerical interpreter for float signals, and to a categorical one for string and boolean signals | |
Numerical | No special aspect generation. The signal is taken as it is. | Speed, temperature, weight,... |
Categorical | Each signal value corresponds to some category. Categories have no order. | Color, operation mode, on/off,... |
Cyclic | Signal values can be mapped to a finite interval. Lower and upper bound of this interval are identified with each other. | Angles (0° to 360°), time of the day (0:00 to 24:00),... |
Oscillatory | Signal contains periodically recurrent parts. Interest is rather in the frequency of recurrences than the actual signal values. | Audio data, vibrations,... |
By default, all float signals are interpreted as numerical. This interpreter should be used for all signals for which the order of numbers is meaningful and which don't require some special treatment. A thermometer, for example, generates numerical data: the smaller the number the colder the temperature. It is irrelevant whether the scale is continuous, or whether the thermometer’s reading precision is limited to integer degrees. The numerical signal kind is quite common for float signals but there are also situations, for which it does not fit. Therefore, float signals may be also declared any of the other signal kinds.
String and boolean signals are always interpreted as categorical. Categorical data has nominal scale, i.e. it takes only specific levels and does not necessarily follow any order. In practice, this would express the information about certain states, such as “green”, “red”, or “blue”. This information may be present in form of strings, booleans, or also encoded in numbers. An example could be a signal for which "1.0" stands for "pipe open", "2.0" for "pipe blocked", and "3.0" for "pipe sending maintenance alarm".
For a cyclic signal only the residue from division by the cycle length is accounted for. This means the order of numbers is meaningful but it wraps at the cycle length. A common example are angles. Angles are usually defined in the interval \(0\) to \(2 \pi\). This means a cycle length of \(2 \pi\). If the signal takes a value outside this range, it is automatically mapped therein. For example, \(2.1 \pi\) is identified with \(0.1 \pi\). And, of course, 0 and \(1.99 \pi\) are considered to be close to each other. Another example can be derived from a continuous time signal. Let's say time is measured in the unit of hours. Then, applying an interpreter with cycle length 24, yields an aspect that describes the time of the day.
Finally, audio, vibration, or any other data that oscillates with some periodicity may best be interpreted as oscillatory. Oscillatory signals are interpreted in the frequency domain. In order to calculate a frequency spectrum and automatically derive the most relevant aspects, two configuration parameters are necessary. The mesh describes the shortest timespan to consider, the inverse sampling frequency. For example, a mesh of 2 milliseconds means a sample rate of 0.5 kHz. Within this documentation, the unit of the timestamps is usually assumed to be milliseconds to keep explanations concise. However, the unit of timestamps is irrelevant internally. Oscillatory signals may well have sample rates above 1 kHz for which a more fine-grained time unit is necessary. For example, a 32 kHz audio signal has a signal value each 0.031250 milliseconds. In this case, the usual notion of timestamps as milliseconds does not work anymore. Instead, timestamps may be provided in units of nanoseconds and full information is retained for a mesh of 31250. (Alternatively, timestamps may also be provided in units of a thirty second part of a millisecond, and full information is retained for a mesh of 1.) If the highest frequencies of the signal are expected not to be relevant, i.e. if the microphone or detector records with a higher rate than actually needed, the mesh may be chosen larger than the difference between timestamps in the data. In the above example, a mesh of 62500 nanoseconds would retain only each second value. The other parameter is the window length. It describes the longest time span to consider for a frequency spectrum. Therefore, it should reflect some reasonable time to "listen" to the signal before trying to get information out of it. The window length defines the period of the lowest frequency that can be analysed. Therefore, at the very least it should be as long as the period of the lowest relevant frequency. If no data are provided during some interval larger than twice the mesh, no frequency spectrum is calculated for this gap. Instead, the frequency spectrum is calculated for the last period for which there is no gap over the window length. This behavior allows for discontinuous signal transmission. To reduce the amount of data, for example, the signal may be provided in bunches which are sent each 2 minutes, each covering a time period of 10 seconds. The drawbacks are some delay of results, up to 2 minutes in the above example, and loss of information about anything happening within the gap.
Sometimes, it is not clear which interpreter to choose. As an example, take a signal for which "0.0" may stand for "no defects", "1.0" for "isolated microscopic defects" and "3.0" for "microscopic connected defects". A priori, one may assume that the effect of isolated defects may be somewhere in between no defects and connected defects, and thus assign the numerical scale. On the other hand, isolated microscopic defects may have no relevant effect: It may be irrelevant whether there are no or isolated defects. In this case, a categorical scale would be preferable. Such a situation can easily be dealt with: create a duplicate of the signal with the help of the expression language, configure one of the signals as numerical and the other as categorical, and let aivis make the best out of both.
This configuration option was mentioned already in the section on data filter. If force retain is set to true, the corresponding signal will not be cleaned and thus definitely enters the model building. As aivis performs very well in feature selection, you may not expect to improve the model performance by forcing some signals to be included. Nevertheless you may be interested in how the model or its performance changes by inclusion of some signal. In a first run, you may have denoted that some signal has been cleaned away although you expect or even know from previous data analysis that it is related to the target. In this case, you may force the engine to retain it in order to calculate and retrieve its correlation to the target and its relevance compared to other signals. Often the same information is contained in different signals and therefore even signals with high correlation to the target may be excluded because they are redundant.
2.3
The use of a global lagging configuration is explained below. Sometimes it can make sense to adjust the lagging specifically to some signals. This helps, among other things, to keep calculational effort low and can be done if you have prior information for some signals about which (range of) lags are most relevant: For a few signals a very high maximal lag may be useful, or a very small mesh. For example, in a production line, it may take a few seconds or minutes from one process step to the next one. Let's assume you are interested in the third process step. Then signals that refer to this step may be configured with zero lag, while signals that describe the first two process steps may be configured with some lag > 0.
A typical case for the application of signal configurations arises when the target is synthesized by the expression language. In this case, all signals that are part of the expression are automatically excluded, see the section on the target configuration. However, this exclusion is not performed for signals with a specific signal configuration. In this scenario, the minimal lag may be of particular relevance, to allow modeling of delayed information availability in the database, see the sections on the lagging and on the target configuration.
In many situations, predictions can be improved by taking into account parts of the history of some signal values, and not only the most recent value. However, signal values from very far in the past are unlikely to be relevant, and their inclusion would unnecessarily increase computation time. Therefore, the maximum time window to be included for each signal can be controlled by the parameter maximal lag [milliseconds]. The maximal lag tells aivis how far it should look into the past to find dependencies. For example, if data from 10 minutes ago affects the current target value, then the maximal lag should be at least that high. Consequently, the best value for maximal lag depends on the process at hand. Start out with lower values if possible, as this value has a large impact on the calculation effort. If this configuration is not set, the target signal is predicted only from the current signal values (or, as always, their nearest predecessors).
The number of sample points within this past time window is controlled by the mesh [milliseconds]. aivis analyzes said time window by slicing it up and taking a sample data point from each slice. The width and therefore the amount of these slices is determined by the mesh. This means that for a target at time t, signal values are retrieved at times t, t – mesh, t – 2 * mesh, and so on until you reach the maximal lag constraint. A smaller mesh is associated with higher computing time and memory demand but may lead to better predictions. It should be kept in mind that if the mesh is larger than the sample interval of the data, some information is lost. Therefore, it is important to choose the mesh according to the time scale during which relevant signals may change.
Simplified, you can imagine that maximal lag and mesh take data points from the past and add them as an additional “lagged signal” to the original problem. For example, a maximal lag of 10 hours and a mesh of 1 hour would require as much additional computational effort as setting a maximal lag of 5 hours, but a mesh of 30 minutes, as both would add up to 10 “lagged” signals per original signal. For each lagged signal, the distance correlation is calculated. The results constitute a correlation trend, and an example is depicted in the figure below. In the example the correlation is highest shortly before the evaluation time. This may be indicative that a shorter maximal lag might suffice for this signal. The correlations are part of the report. This way, the lagging configuration can be checked after the training.
These different lagged signals are almost always strongly correlated with each other and thus partially redundant. However, you don't need to care about this problem. The different lagged signals are automatically combined and selected to distill full information with a low number of final features.
In analogy to the maximal lag, there is also the option to define a minimal lag 2.3. If this parameter is set to some positive value, no information is used from the most recent past. This parameter is mainly useful for special cases: First, if you know that some range of lags is particularly important, it allows you to pinpoint aivis to exactly this range of lags. The second case applies to delayed information during live prediction. If you expect signals to be available only with some delay in your database, this situation can be emulated by setting a minimal lag. Then aivis will not use any information more recent than the minimal lag, and performance will therefore be independent of any delay shorter than this minimal lag. In the regard of information delays also note the availability timestamps.
The control point count can be set to tweak model building.It controls the granularity of the model and has a strong influence on the required computing time. The larger this number, the more details may potentially be included. However, training computing time scales approximately with the third power of the control point count. By default, 5,000 control points are used. Just like with the maximal sample counts, a higher maximal control point count should technically yield better results. However, there is always a ceiling above which increasing the value will not lead to significant improvements, but only longer computation times.
As already mentioned in the in the beginning of this documentation, there are two distinct use cases of aivis Anomaly Detection. This can be controlled by the flag check target at inference. For usual production monitoring, of course, the target needs to be monitored: it was chosen because it is an important signal. Therefore, the flag defaults to true. However, there is another application of aivis Anomaly Detection: to complement aivis Signal Prediction with an estimate of prediction validity. The purpose of aivis Signal Prediction is to estimate some signal (called also target for aivis Signal Prediction). This is relevant for signals that are not readily available, for example because signal values can only be retrieved after an expensive or protracted measurement process. Therefore, in an aivis Signal Prediction setting, the current target is not available in general. Goal of aivis Signal Prediction is to predict the best possible value for this target based on information on other signals. Inevitably, the more these other signals differ from what has been learned in the training, the more uncertain becomes the prediction. For this purpose, aivis Anomaly Detection can be used to retrieve some validity score: the anomaly score for a model run with the same data and configuration as the corresponding signal prediction model. When switching off to check the target at inference, the anomaly detection model is optimized for such signal prediction health check.
There are two outputs that can be retrieved from the training. First, the report lists the most relevant information. This includes a list of all signals considered for model building together with their distance correlation to the target. Distance correlations are also presented for different aspects and lags. And the report includes a list of all other signals together with a justification for their exclusion from model building. Both information are relevant to understand which part of your system is monitored by the model. Excluded signals are not monitored at all. The other signals are tracked as closer the larger the distance correlation is. A single anomaly model usually does not cover your system completely but it is necessary to have several anomaly models in place. The report can help you detecting parts of your system that might be covered insufficiently. Note, however, that usually it is enough to include only some subset of your signals because problems show up at the same time in correlated signals. Such signals that were excluded due to redundancy with other signals are marked to be of low importance. In contrast, signals marked as independent have no impact on the considered target. Changes in your system that are reflected only by changes in such signals will not be detected. This way, the anomaly detection model(s) focus only on the relevant parts of your system. You may add additional models to cover signals marked as independent. But usually it turns out that those signals are not used for monitoring, anyway.
Second, a model is retrieved, which will later be used for predicting the target values, i.e., for inference. The model also specifies the Inference Data Specification and the inference output type.
2.4
Upon the completion of the initial training phase, it is typical for new data to be gathered and stored. This data can be incorporated in two ways. One could either retrain the model from scratch, employing the full dataset, or undertake an Incremental Learning process, focusing solely on the data amassed after the initial training. Frequently, the latter approach proves to be the only practical option, because of the inaccessibility of older data, the difficulties associated with retrieving it or the desire to keep the old model for the most part unchanged and only tweak it in a configurable way.
In the Incremental Learning phase, the pre-existing model is retrained on this new data. The process keeps a record of the optimal parameters identified in the previous training stage, which are then optimized once more to reflect the new data samples.
Subsequent sections will explain the steps integral to the incremental process, as well as the necessary components to execute this process. These components include the incremental configuration, the new incremental data, and the already trained model, which was discussed in prior sections.
To execute an incremental learning step, the model at the user's disposal should have been initially trained with the enable incremental option in the training configuration. This setting ensures that along with the standard model information, the requisite quantities for retraining the model are also stored.
In terms of data, the same general specifications apply as those for the initial training, while it is enough to only provide data for the target signal and the signals contained in the model, i.e. listed in the inference data specification. These data can be then interpreted in parts using the data filter.
The workflow for incremental learning is much like the original training process, but with an important distinction. Rather than recalculating all the data preparation steps - such as feature engineering, signal cleaning, segmentation, and model building - incremental learning applies the data preparation steps that were already computed during training to the new data. This way, it recreates the same features used in the training phase. Consequently, incremental learning only utilizes datapoints where it's possible to reproduce these features.
A general workflow representation is shown in the previous figure. The procedure requires three main components: input data (see Data Specification), an already trained model (see Incremental Learning Requirements) and the incremental configurations.
Below is an example illustrating the utilization of two distinct configurations for Incremental Learning (Learning and Update Configuration). The explanation of the different keys will be provided after the subsequent tables.
label: TARGET
incremental_learning_config = json.dumps({
"label": "TARGET"
})
final DtoIncrementalLearningConfig incrementalLearningConfig = new DtoIncrementalLearningConfig("TARGET");
const char *incremental_learning_config = "{"
"\"label\": \"TARGET\""
"}";
The key label indicates which signal contains the information of the incremental data being normal (all label values 0) or abnormal (all label values 1).
dataFilter:
autoUpdate:
tolerance: 0.2
maximalCount: 10
incremental_update_config = json.dumps({
"dataFilter": {},
"autoUpdate": {
"tolerance": 0.2,
"maximalCount": 10
}
})
final DtoIncrementalUpdateConfig incrementalUpdateConfig = new DtoIncrementalUpdateConfig()
.withDataFilter(new DtoDataFilter())
.withAutoUpdate(new DtoIncrementalAutoUpdateConfig(0.2).withMaximalCount(10));
const char *incremental_update_config = "{"
"\"dataFilter\": {},"
"\"autoUpdate\": {"
"\"tolerance\": 0.2,"
"\"maximalCount\": 10"
"}"
"}";
The optional key data filter allows you to restrict time ranges that are used for the incremental retraining (for a detailed explanation, see Data Filter ). Excluding signals in the data filter is not advised as it has no impact and might cause an error. The needed signals for incremental retraining are only given by the inference data specification.
The optional key auto update helps to ensure that a given inference quality on the incrementally added data is met. If specified, it triggers a series of iterative incremental training sessions on the new data. These successive training steps will continue until the difference between the score inferred by the incremental model and the provided label value is smaller than the specified tolerance for all timestamps in the incrementally added data, or until the maximum number of iterations maximal count is reached.
Below is a simple code snippet for one model update using the configurations defined above.
data:
folder: /srv/incremental-data
dataTypes:
defaultType: FLOAT
incrementalLearning:
modelFile: /srv/model.json
learningConfig:
label: TARGET
updateConfigs:
- dataFilter:
autoUpdate:
tolerance: 0.2
maximalCount: 10
output:
folder: /srv/output
learning = AnomalyDetectionIncrementalLearning.create(model, incremental_learning_config)
learning.update(incremental_data, incremental_update_config)
updated_model = learning.get_model()
final AnomalyDetectionIncrementalLearning learning = AnomalyDetectionIncrementalLearning.create(model, incrementalLearningConfig);
learning.update(incrementalData, incrementalUpdateConfig);
final IDtoModel updatedModel = learning.getModel();
AnomalyDetectionIncrementalLearningHandle learning_handle = aivis_anomaly_detection_incremental_learning_create(
(uint8_t *) incremental_learning_config,
strlen(incremental_learning_config),
&err
);
check_err(&err, "Create learning handle");
aivis_anomaly_detection_incremental_learning_update(
learning_handle,
incremental_data_handle,
(uint8_t *) incremental_update_config,
strlen(incremental_update_config),
&err);
check_err(&err, "Update");
const List_u8 *updated_model = aivis_anomaly_detection_incremental_learning_get_model(
learning_handle,
&err);
check_err(&err, "Save model");
When the model has been trained, it is ready for the ultimate goal, which is inference. Inference means, the model is provided with new (usually unseen) data around a certain timestamp and is asked for some value/estimation at said timestamp: That value is simply the prediction of the target in aivis Signal Prediction. In aivis Anomaly Detection the inferences are scores and in aivis State Detection the inferences provide scores per segment.
In general, there are two main scenarios in which you would want to make inferences. The first one is performance evaluation of the model. Here, inferences are made on historical data, i.e., some test data set. Again, it is important that this data was not part of the training set, as this would lead to unrealistic and therefore non-representative predictions. These historical inferences are used for determining model performance, see section “Measuring Model Performance”.
The second typical scenario for making inferences is using them in a productive setting. Here, the true target value or process state is technically not known at the time of the inference. This is called live inference. For live inference, inferences are usually made on an ongoing basis, as this is typically what you would want for most productive use cases. This is contrary to performance evaluation, where all inferences are made in one go.
For each of the above scenarios, there is a dedicated docker image. The Inference Worker creates predictions for a predefined time window in a bulk manner for model evaluation. In contrast, the Inference Service is optimized for live inference. It offers a RESTful web API that allows the triggering of individual predictions for a specified time via an HTTP call. Due to the different application modes, APIs differ between the different docker images and the SDK. These differences will be noted in the following sections.
In the SDK, in order to make inferences, it is necessary to pass a list of timestamps for which the inferences are to be made. This allows for the request of a single live inference result but also for bulk inference for model evaluation on historic data. Typically it is easy to generate such lists of timestamps in the programming language that calls the SDK. On the other hand, docker images are not necessarily called from within a powerful programming language. This is not an issue for the Inference Service. For live inference, typically inference is requested only for a single timestamp, the most recent one. However, it could be cumbersome to derive a list of timestamps for the Inference Worker. Therefore, for the Inference Worker, timestamps are selected via a list of timestamps configs. There are two different methods:
Timestamp Config | Short Explanation | Typical Use Case |
---|---|---|
Equidistant | Provides equidistant inference timestamps with fixed interval (for example a inference each minute). | Obtain continuous inferences in some time interval. |
AtNextSignalValue | Selects those timestamps for inference for which there are data points for some specified signal. | For model validation it is necessary to make inferences for timestamps for which target values are known. |
For both timestamps configs, there are a start time and an end time. An operative signal can be used to further restrict the timestamps. Finally, a minimum interval may be set to avoid calculating too many inferences in a short time interval. This can speed up the computation, or may be useful to balance the distribution of inferences. Finally note that further flexibility can be gained by providing several timestamps configs in which case all timestamps are combined. An example was already provided in the Getting Started section.
Regarding the data, the same general specifications hold as for the training. In addition, the data is expected to obey the Inference Data Specification that can be inspected from the training output. The Inference Data Specification ensures that all data needed for inference has been provided. Note that this may include also signals synthesized by the expression language, which then need to be synthesized by the same expression. If the Inference Data Specification is not satisfied for some timestamp, this timestamp is either skipped on insufficient data, or an error is thrown. This behavior needs to be configured for the SDK and the Inference Worker. For the Inference Service, an error is always returned.
Data filtering can be performed as for the training. As described in the subsection on training data filtering, evaluating the performance of the model is done best by splitting the available data with the data filter. That way, it is not necessary to provide different data files for training and inference. Instead, some training end timestamp is specified, and the same timestamp for inference start. With this approach, the Inference Data Specification is automatically satisfied. For the Inference Service, data filtering is not possible.
Signal availabilities are an important concept for checking model performance with historical data. Only those data are to be used for inference that would be available if the inference would be performed online. However, the availability timestamp does not always correspond to the timestamp the data would have become available. For example, availability timestamps may have been overridden during some migration of the data base. For such cases, the Inference Worker provides the option to ignore signal availabilities. This option may also come in handy to check the impact of the delay of information on the model performance.
2.7
For computing the anomaly score in both docker inference and SDK inference two methods are given: infer and infer with next normal. The standard ìnfer function returns the anomaly scores for the specified timestamps.
The function infer with next normal requires an additional next normal config. In this configuration a threshold and optionally a feature filter can be set. Anomaly scores below this threshold are considered normal and no additional computation is performed. For any timestamp at which the feature values yield an anomaly score above the threshold, the next normal point is sought after. The next normal point is the collection of feature values which is the closest to the observed feature values whose anomaly score is below the threshold. Moreover, a rating is calculated from the difference between observed and next normal feature values. It reflects the influence of each feature on the score, i.e. the feature with the highest rating is the most influential factor causing the score being above the provided threshold. The feature filter in the next normal config allows the user to exclude features from the above next normal seeking process or to fix the features which are allowed to vary in the next normal seeking process via the include features entry. If the algorithm can not find a collection of feature values below the threshold it will return the feature values with the smallest reachable score.
In summary, the function infer with next normal returns, in addition to the anomaly score at the timestamp of interest, the observed feature values, the next normal feature values, the score of the next normal feature values, and a rating of the difference between observed and next normal feature values.
In the output and in the feature filter the features are indexed by an integer id which can be looked up in the report to relate this id to the underlying signals, aspects and lags.
Warning: The infer with next normal function is experimental. It does an expensive optimization and should carefully be used for models with many features, in particular when a lagging config is present. Also in such cases the result might not yet be optimal.
We provide here a minimal example of the syntax of the infer with next normal method:
data:
...
inference:
config:
...
timestamps:
...
nextNormal:
threshold: 0.4
output:
...
# choose inference timestamps
timestamps = ...
# build next normal config
next_normal_config = json.dumps({"threshold": 0.4})
# infer anomaly scores and next normal point
scores_with_next_normal = inference.infer_with_next_normal(inference_data, timestamps, next_normal_config)
// choose inference timestamps
final List<Long> timestamps = ...
// build next normal config
final DtoNextNormalConfig nextNormalConfig = new DtoNextNormalConfig(0.4);
// infer anomaly scores and next normal point
final List<DtoFloatDataPointWithNextNormal> scoresWithNextNormal = inference.inferWithNextNormal(inferenceData, timestamps, nextNormalConfig);
// choose inference timestamps
Time *timestamps = ...
// build next normal config
const char *next_normal_config = "{\"threshold\": 0.4}";
// infer anomaly scores and next normal point
const List_DtoFloatDataPointWithNextNormal *scores_with_next_normal = aivis_anomaly_detection_inference_infer_with_next_normal(
inference_handle,
inference_data,
timestamps,
timestamps_len,
(uint8_t *) next_normal_config,
strlen(next_normal_config),
&err
);
check_err(&err, "Infer Scores with next normal");
// free scores_with_next_normal
aivis_free(scores_with_next_normal);
scores_with_next_normal = NULL;
// free timestamps
free(timestamps);
timestamps = NULL;
Regarding the inference configuration, we provide an overview of all kinds of possible configuration keys below. This overview is meant for quick reference. A minimal inference configuration is provided in SDK inference, respectively, in Docker inference. The meaning of the different keys is explained in other sections, and a definition of the syntax is given in the reference manuals. For the docker images, here we focus only on the Inference Worker as it features more configuration keys. Note that on the other side, the inference service can be asked to evaluate several models in one go.
config:
dataFilter:
startTime: 1465632000000
endTime: 1466786640000
excludeSignals:
- signal: L-8
startTime: 1465698800000
endTime: 1465720600000
# includeSignals: ... similar
includeRanges:
- startTime: 1465632000000
endTime: 1465698800000
- startTime: 1465720600000
endTime: 1466786640000
# excludeRanges: ... similar
skipOnInsufficientData: true
inference_config = json.dumps({
"dataFilter": {
"startTime": 1465632000000,
"endTime": 1466786640000,
"excludeSignals": [{
"signal": "L-8",
"startTime": 1465698800000,
"endTime": 1465720600000
}],
# "includeSignals": ... similar
"includeRanges": [{
"startTime": 1465632000000,
"endTime": 1465698800000
}, {
"startTime": 1465720600000,
"endTime": 1466786640000
}],
# "excludeRanges": ... similar
},
"skipOnInsufficientData": True
})
final DtoInferenceConfig inferenceConfig = new DtoInferenceConfig(true)
.withDataFilter(new DtoDataFilter()
.withStartTime(1465632000000L)
.withEndTime(1466786640000L)
.withExcludeSignals(new DtoDataFilterRange[] {
new DtoDataFilterRange("L-8")
.withStartTime(1465698800000L)
.withEndTime(1465720600000L)
})
// .withIncludeSignals ... similar
.withIncludeRanges( new DtoInterval [] {
new DtoInterval()
.withStartTime(1465632000000L)
.withEndTime(1465698800000L),
new DtoInterval()
.withStartTime(1465720600000L)
.withEndTime(1466786640000L)
})
// .withExcludeRanges ... similar
);
const char *inference_config = "{"
"\"dataFilter\": {"
"\"startTime\": 1465632000000,"
"\"endTime\": 1466786640000,"
"\"excludeSignals\": [{"
"\"signal\": \"L-8\","
"\"startTime\": 1465698800000,"
"\"endTime\": 1465720600000"
"}],"
// "\"includeSignals\": ... similar
"\"includeRanges\": [{"
"\"startTime\": 1464732000000,"
"\"endTime\": 1464863188000"
"}, {"
"\"startTime\": 1464932000000,"
"\"endTime\": 1465631880000"
"}]"
// "\"excludeRanges\": ... similar
"},"
"\"skipOnInsufficientData\": true"
"}";
Evaluating the performance on a test data set needs at least some knowledge of the presence or absence of anomalies in the test timeframe.
Even if you assume there are no anomalies in the test data, you may still see the anomaly score to deviate from zero for certain timestamps. This can have several reasons:
Things get even more interesting if there are a few known anomalies in the test data. Then you can check whether all anomalies would have been detected. If anomalies follow some detoriation process, you may observe a slow increase of the anomaly score reflecting this detoriation. You can learn about suitable thresholds, for example to trigger an alarm. Finally, if there are several anomaly models in place, you may observe that only specific scores peak at an anomaly: those for which the target is "closest" to the anomaly.
Before starting the workflow, sometimes there is the need to add a new signal to the dataset (a synthetic signal) that is derived from other signals already present. There are various reasons for this, especially if
Technically, you can add synthetic signals using the docker images or any SDK Data API
To create new synthetic signals in a flexible way, aivis Signal Prediction features a rich Expression Language to articulate the formula.
The Expression Language is an extension of the scripting language Rhai. We have mainly added support for handling signals natively. Information on the basic usage of the language can be found in the very helpful Language Reference of the Rhai Book. This documentation will mainly focus on the added features.
A signal consists of a list of data points that represents a time series (timestamps and values of the same type).
The following value types are supported:
bool
: Booleani64
: 64-bit Integerf64
: 64-bit Floating Pointstring
: UTF-8 StringA signal type and its value type are written generically as signal<T>
and specifically like e.g. signal<i64>
for an integer signal.
It is not possible to write down a signal literally, but you can refer to an already existing signal in your dataset.
Referring to an already existing signal is done via one of these two functions:
s(signal_id: string literal): signal<T>
s(signal_id: string literal, time_shift: integer literal): signal<T>
The optional time shift parameter shifts the data points into the future. For example, if the signal "a" takes the value 5.7 at timestamp 946684800000, then the following expression takes the same value 5.7 at timestamp 946684808000. The synthesized signal is therefore a lagged version of the original signal "a".
s("a", 8000)
These functions must be used exactly with the syntax above. It is not allowed to invoke them as methods on the signal id. Both parameters must be simple literals without any inner function invocation!
Examples:
s("my signal id") // OK
s("my signal id", 8000) // OK
s("my s" + "ignal id", 8000) // FAIL
"my signal id".s(8000) // FAIL
s("my signal id", 7000 + 1000) // FAIL
To begin with, let's start with a very simple example. Let "a" and "b" be the IDs of two float signals. Then
s("a") + s("b")
yields the sum of the two signals. The Rhai +
operator has been overloaded to work directly on signals (such as many other operators, see below). Therefore, the above expression yields a new signal. It contains data points for all timestamps of "a" and "b".
A more common application of the expression language may be the aim to interpolate over several timestamps. For example, "a" might fluctuate and we may therefore be interested in a local linear approximation of "a" rather than in "a" itself:
trend_intercept(s("a"), t, -1000, 0)
Here, the literal t
refers to the current timestamp. Therefore, the expression yields the present value as obtained from a linear approximation over the last second. As another example, the maximum within the last second:
max(slice(s("a"), t, -1000, 0))
A typical use of the expression language is synthesizing an operative signal. Assume you want to make inferences only when your production is running, and you are sure your production is off when some specific signal "speed" falls below a certain threshold, say 10. However, "speed" may also be above the threshold during maintenance. However, during maintenance "speed" exceeds the threshold only for a few hours. This is in contrast to production which usually runs stable for months. In this situation, an operative signal may thus be synthesized by adopting only intervals larger than one day, i.e. 86400000 ms:
set_sframe(s("speed") > 10, false, 86400000)
In the following, all functions are defined that operate directly on signals and do not have a Rhai counterpart (such as the +
operator). Some functions directly return a signal. The others can be used to create signals via the t
literal as will be explained below. Note that a timeseries is always defined on a finite number of timestamps: all timestamps of all signals involved in the expression are used for the synthesized signal. Time shifts specified in the signal function s(signal_id: string literal, time_shift: integer literal)
are taken into account. On the other hand, arguments of the functions below (in particular time
, from
, and to
) do not alter the evaluation timestamps. If you need more evaluation timestamps, please apply add_timestamps
to some signal in the expression (see below).
add_timestamps(signal_1: signal<T>, signal_2: signal<S>): signal<T>
– returns a new signal which extends signal_1 by the timestamps of signal_2. The signal values for the new timestamps are computed with respect to signal_1 using the latest predecessor similar to the above at()
function. The syntax for this expression is s("x1").add_timestamps(s("x2"))
. 2.4at(signal: signal<T>, time: i64): T
– returns the signal value at a given timeNAN
, 0
, false
or ""
set_lframe(signal: signal<bool>, new_value: bool, minimal_duration: i64) : signal<bool>
– returns a new boolean signal, where large same-value periods of at least duration minimal_duration
are set to new_value
. Note that the duration of a period is only known after end of the period. This affects the result of this function especially for live prediction.set_sframe(signal: signal<bool>, new_value: bool, maximal_duration: i64) : signal<bool>
– returns a new boolean signal, where small same-value periods of at most duration maximal_duration
are set to new_value
. Note that the duration of a period is only known after end of the period. This affects the result of this function especially for live prediction.slice(signal: signal<T>, time: i64, from: i64, to: i64): array<T>
– returns an array with all values within a time window of the given signal.[time + from; time + to]
steps(signal: signal<T>, time: i64, from: i64, to: i64, step: i64): array<T>
– returns an array with values extracted from the given signal using the at
function step by step.(time + from) + (0 * step)
, (time + from) + (1 * step)
, ... (until time + to
is reached inclusively)time_since_transition(signal: signal<bool>, time: i64, max_time: i64) : f64
– returns a new float signal, which gives time since last switch of signal
from false
to true
. If this time exceeds max_time
we return max_time
. Times before the first switch and times t
where the signal gives false
in [t - max_time , t]
are mapped to max_time
. 2.4times(signal: signal<T>): signal<i64>
– returns a new signal constructed from the given one, where the value of each data point is set to the timestamptrend_slope/trend_intercept(signal: signal<i64/f64>, time: i64, from: i64, to: i64): f64
– returns the slope/y-intercept of a simple linear regression modelNAN
value is ignored; returns NAN
if there are no data points available; the following timestamps are used: [time + from; time + to]
. The intercept at t = time
is returned.When combining several expressions which operate on time windows then, from a performance point of view, it might be better to build the expression step by step than writting the combination into one expression.
For example, if we want to exclude periods smaller than 30 minutes and periods bigger than 12 hours from an existing boolean signal with signal id "control" we may use the expression:
(s("control")).set_lframe(false, 12*60*60*1000).set_sframe(false, 30*60*1000)
When evaluating this expression at a timestamp t
the synthesizer scans trough the 30 minutes time window before t
and for each timestamp in there it scan through another 12 hour window before.
This means constructing the desired synthesized signal is of complexity 12 × 60 × 30 × # timestamps. However, splitting the above in two expressions, we first generate a signal "helper" via
(s("control")).set_lframe(false, 12*60*60*1000)
and then we apply on the result the expression
(s("helper")).set_sframe(false, 30*60*1000)
In this case we end up with complexity 12 × 60 × # timestamps + 30 × # timestamps which is considerably smaller than before.
In this section, we will briefly show the potential behind Rhai and what you can create out of it. Rhai supports many types including also collections. But Rhai does not have natively a signal type. Then, when working with signals, one approach involves extracting the primitive values from signals and converting the results back into a signal format. This process uses the literal
t: i64
– the current timestamp
together with the function s
to refer to some signal and some other function defined above to extract values from the signal. For example, the sum of two signals "a" and "b" could be written without use of the overloaded +
operator:
s("a").at(t) + s("b").at(t)
The results of such an expression are automatically translated into a new signal. In order to construct a signal from the results, the expression must not terminate with a ;
. Of course, the additional signal functions can be used as any other functions in Rhai, and may thus be combined with the rest of Rhai's tools, when applicable.
As such, you can script. A typical snippet would look like the following
let array = [[s("one").at(t), s("two").at(t)], [s("three").at(t), s("four").at(t)], [s("five").at(t), s("six").at(t)]];
let pair_avg = array.map(|sub| sub.avg());
pair_avg.filter(|x| !x.is_nan()).map(|cleaned| cleaned.abs().exp()).sum().ln()
Here, we used array functions (avg(), sum()) that will be clearly defined and presented in the following sections. The last line defines the result of the expression.
In the same spirit of many other languages, you can create and control flow using statements if, for, do, while, and more (read Language Reference of the Rhai Book). Here's an example demonstrating their usage
let val = s("one").at(t);
if (val >= 10.0) && (val <= 42.0) {
1.0 - (val - 42.0)/(10.0-60.0)
} else if (val <= 60.0) && (val > 42.0) {
1.0 - (val - 42.0)/(60.0-42.0)
} else {
0.0/0.0
}
In this code snippet, we determine a value to return based on the current state of the "one" signal. Different expressions are assigned depending on the signal's current value. Note that 0.0/0.0
will evaluate to NAN
.
Like most other languages, you can create your own functions and use them whenever needed.
fn add(x, y) {
x + y
}
fn sub(x, y,) { // trailing comma in parameters list is OK
x - y
}
Rhai allows you to do many more things than the ones here described. Careful reading of Language Reference of the Rhai Book brings numerous benefits in the usage of this programming language.
The following functions for arrays were additionally defined:
some(items: array<bool>): bool
– returns true
if at least one item is true
all(items: array<bool>): bool
– returns true
if all items are true
sum(items: array<i64/f64>): f64
– returns the sum of all items and 0.0
on an empty arrayproduct(items: array<i64/f64>): f64
– returns the product of all items and 1.0
on an empty arraymax(items: array<i64/f64>): f64
– returns the largest array item; any NAN
value is ignored; returns NAN
on an empty arraymin(items: array<i64/f64>): f64
– returns the smallest array item; any NAN
value is ignored; returns NAN
on an empty arrayavg(items: array<i64/f64>): f64
– returns the arithmetic average of all array items; any NAN
value is ignored; returns NAN
on an empty arraymedian(items: array<i64/f64>): f64
– returns the median of all array items; any NAN
value is ignored; returns NAN
on an empty arrayThe following constants are defined in Rhai:
PI(): f64
– the Archimedes' constant: 3.1415...
E(): f64
– the Euler's number: 2.718...
Signals can be used in all normal operators and functions that are designed for primitive values. You can even mix signals and primitive values in the same invocation. If at least one parameter is a signal, the result will also be a signal.
See:
The following operators were defined:
+(i64/f64): i64/f64
-(i64/f64): i64/f64
+(i64/f64, i64/f64): i64/f64
-(i64/f64, i64/f64): i64/f64
*(i64/f64, i64/f64): i64/f64
/(i64/f64, i64/f64): i64/f64
%(i64/f64, i64/f64): i64/f64
**(i64/f64, i64/f64): i64/f64
&(i64, i64): i64
|(i64, i64): i64
^(i64, i64): i64
<<(i64, i64): i64
>>(i64, i64): i64
!(bool): bool
&(bool, bool): bool
|(bool, bool): bool
^(bool, bool): bool
+(string, string): string
false
on different argument types):==(bool/i64/f64/string, bool/i64/f64/string): bool
!=(bool/i64/f64/string, bool/i64/f64/string): bool
<(i64/f64, i64/f64): bool
<=(i64/f64, i64/f64): bool
>(i64/f64, i64/f64): bool
>=(i64/f64, i64/f64): bool
Binary arithmetic and comparison operators can handle mixed i64
and f64
arguments properly, the other parameter is then implicitly converted beforehand via to_float
. Binary arithmetic operators will return f64
if at least one f64
argument is involved.
See:
The following functions were defined:
abs(i64/f64): i64/f64
sign(i64/f64): i64
sqrt(f64): f64
exp(f64): f64
ln(f64): f64
log(f64): f64
log(f64, f64): f64
sin(f64): f64
cos(f64): f64
tan(f64): f64
sinh(f64): f64
cosh(f64): f64
tanh(f64): f64
asin(f64): f64
acos(f64): f64
atan(f64): f64
asinh(f64): f64
acosh(f64): f64
atanh(f64): f64
hypot(f64, f64): f64
atan(f64, f64): f64
floor(f64): f64
ceiling(f64): f64
round(f64): f64
int(f64): f64
fraction(f64): f64
len(string): i64
trim(string): string
– with whitespace characters as defined in UTF-8to_upper(string): string
to_lower(string): string
sub_string(value: string, start: i64, end: i64): string
to_int(bool): i64
– returns 1
/0
to_float(bool): f64
– returns 1.0
/0.0
to_string(bool): string
– returns "true"
/"false"
to_float(i64): f64
to_string(i64): string
to_int(f64): i64
– returns 0
on NAN
; values beyond INTEGER_MAX
/INTEGER_MIN
are cappedto_string(f64): string
to_degrees(f64): f64
to_radians(f64): f64
parse_int(string): i64
– throws error if not parsableparse_float(string): f64
– throws error if not parsableis_zero(i64/f64): bool
is_odd(i64): bool
is_even(i64): bool
is_nan(f64): bool
is_finite(f64): bool
is_infinite(f64): bool
is_empty(string): bool
NAN
):max(i64/f64, i64/f64): i64/f64
min(i64/f64, i64/f64): i64/f64
Comparison operators can handle mixed i64
and f64
arguments properly, the other parameter is then implicitly converted beforehand via to_float
. It will return f64
if at least one f64
argument is involved.
The Boolean conversion and comparison functions were added and are not part of the official Rhai.
Usually the steps of the workflow will run as part of two different service applications: Training App and Inference App
The diagrams below display typical blueprints of these service aplications using different available components of the engine as well as where they might be located in the end-customer infrastructure landscape (execution environments).
Hereby the following color code was used:
The service application Training App covers the workflow step Training, as well as any bulk inference, e.g. for historical evaluation.
It is executed in the so-called Cold World, which means that it consists of long running tasks that are executed infrequently and have a high resource consumption. Training App works on historical data that was previously archived and thus needs to be retrieved in an extra step from the Data Lake / Cold Storage.
Because of its high resource consumption it is usually not located in the OT network, but is a good fit for the cloud or an on-premise datacenter.
The service application Inference App provides the means for live prediction.
In contrast to the Training App, it runs within the Hot World. Usually it is an ongoing process which serves to predict the current value and only needs minimal resources. Inference App works on live data that is easily available from the Historian / Hot Storage.
As the outcome often influences the production systems (e.g. Advanced Process Control), usually it runs in the OT network. Thanks to low resource consumption, it can run on practical any environment/device, be it in the cloud, on-premise, on-edge or even embedded.
aivis engine v2 toolbox is no official part of aivis engine v2 but an associated side project. It mainly provides tools to turn output artifacts of aivis engine v2 into technical, single-file HTML reports for data scientists. Its api and behaviour is subject to change and experimental. Users should already know the concepts of aivis engine v2 beforehand.
Caveats:
The aivis engine v2 toolbox does not need a licensing key. Its python code is free to look into or even adapt. The respective toolbox release belonging to a aivis engine v2 release {VERSION} is available as:
aivis_engine_v2_toolbox-{VERSION}-py3-none-any.whl
aivis-engine-v2-toolbox:{VERSION}
Each call to construct a toolbox HTML report for engine xy has the following structure:
from aivis_engine_v2_toolbox.api import build_xy_report
config = {
"title": "My Use Case Title",
...
"outputFile": "/path/to/my-use-case-report.html"}
build_xy_report(config)
Additionally, the config needs to contain references to the respective engine's output files, e.g. "analysisReportFile": "/path/to/analysis-report.json"
. The full call to create a report for any engine can for example be found in the python or argo examples of the respective engine.
There are many optional expert configurations to customize your HTML report. Some examples:
The aivis engine v2 toolbox always assumes timestamps to be unix and translates them to readable dates. This behaviour can be switched off via "advancedConfig": {"unixTime": False}
, so that timestamps always remain long
values.
By referring to a metadata file via "metadataFile": "/path/to/metadata.json"
, signals are not only described via their signal id but enriched with more information. The metadata json contains an array of signals with the keys id
(must) as well as name
, description
, unitSymbol
, unitType
(all optional):
{
"signals": [{
"id": "fa6c65bb-5cee-45fa-ab19-355ba94889e9",
"name": "et 1",
"description": "extruder temperature nr. 1",
"unitName": "Kelvin",
"unitSymbol": "K"
}, {
"id": "dc3477e5-a83c-4485-b7f4-7528d336d9c4",
"name": "abc 2"
},
...
]}
To every HTML report which contains a timeseries plot, additional signals can be added to also be displayed. It is however not an automatism to include all signals of the dataset for display, since a full dataset is typically an amount of data which should not be put into a single-file HTML.
All custom configuration options can be seen in the api.py
file in src/aivis_engine_v2_toolbox
.