diff --git a/config.yaml b/config.yaml
index 8c6372b3..3270b0c1 100644
--- a/config.yaml
+++ b/config.yaml
@@ -27,6 +27,7 @@ packages:
- database-postgres
- reprocess-mediator
- fhir-ig-importer
+ - openfn
- datalake
profiles:
diff --git a/documentation/packages/openfn/README.md b/documentation/packages/openfn/README.md
new file mode 100644
index 00000000..cc20b32c
--- /dev/null
+++ b/documentation/packages/openfn/README.md
@@ -0,0 +1,40 @@
+# Package Documentation
+
+## Introduction
+
+Welcome to the documentation for the `openfn` package! This package is designed to provide a platform for seamless integration and automation of data workflows. Whether you are a developer, data analyst, or data scientist, this package will help you streamline your data processing tasks.
+
+## Usage
+
+Once you have added the `openfn` package, you can start using it in your projects. Here is how to instantiate the package
+
+`instant package init -n openfn --dev`
+
+## Demo
+
+To get a hands-on experience with the `openfn` package, try out the demo. The demo showcases the package's capabilities and provides a sample project used to export data from CDR to NDR with transformations. It utilizes a Kafka queue and a custom adapter to map Bundles to be compliant with the FHIR Implementation Guide (IG).
+
+### Getting Started
+
+To access the demo, follow these steps:
+
+1. Visit the [OpenFn Demo](http://localhost:4000) website.
+2. Use the following demo credentials
+
+```
+username: root@openhim.org
+password: instant101
+```
+
+3. Configure the Kafka trigger
+ Change the trigger type from webhook to “Kafka Consumer”
+ Enter in configuration details → see [docs](https://docs.google.com/document/d/1cefHnp6IS6zvFwqQs8EsMQo5D4npsoE-S33R4OBpQjI/edit?usp=sharing)
+ Kafka topic: {whichever you want to use} (e.g., “cdr-ndr”)
+ Hosts: {cdr host name}
+ Initial offset reset policy: earliest
+ Connection timeout: 30 (default value, but can be adjusted)
+ Warning: Check Disable this trigger to ensure that consumption doesn’t start until you are ready to run the workflow! Once unchecked, it will immediately start consuming messages off the topic.
+
+### Documentation
+
+For more detailed information on the `openfn` package and its functionalities, please refer to the [official documentation](https://github.com/openfn/docs). The documentation covers various topics, including installation instructions, usage guidelines, and advanced features.
diff --git a/documentation/packages/openfn/environment-variables.md b/documentation/packages/openfn/environment-variables.md
new file mode 100644
index 00000000..7ce16508
--- /dev/null
+++ b/documentation/packages/openfn/environment-variables.md
@@ -0,0 +1,184 @@
+## Environment Variables
+
+
+
+
+ Variable Name |
+ Description |
+ Type |
+ Relevance |
+ Required |
+ Default |
+
+
+
+
+ DATABASE_URL |
+ The URL of the PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ DISABLE_DB_SSL |
+ Whether to disable SSL for the database connection |
+ |
+ |
+ |
+ |
+
+
+ IS_RESETTABLE_DEMO |
+ Whether the application is running in resettable demo mode |
+ |
+ |
+ |
+ |
+
+
+ LISTEN_ADDRESS |
+ The IP address to listen on |
+ |
+ |
+ |
+ |
+
+
+ LOG_LEVEL |
+ The log level for the application |
+ |
+ |
+ |
+ |
+
+
+ ORIGINS |
+ The allowed origins for CORS |
+ |
+ |
+ |
+ |
+
+
+ PRIMARY_ENCRYPTION_KEY |
+ The primary encryption key |
+ |
+ |
+ |
+ |
+
+
+ SECRET_KEY_BASE |
+ The secret key base |
+ |
+ |
+ |
+ |
+
+
+ WORKER_RUNS_PRIVATE_KEY |
+ The private key for worker runs |
+ |
+ |
+ |
+ |
+
+
+ POSTGRES_USER |
+ The username for the PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ POSTGRES_SERVICE |
+ The service name for the PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ POSTGRES_DATABASE |
+ The name of the PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ POSTGRES_PASSWORD |
+ The password for the PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ POSTGRES_PORT |
+ The port number for the PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ OpenFn_POSTGRESQL_DB |
+ The name of the OpenFn PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ OpenFn_POSTGRESQL_USERNAME |
+ The username for the OpenFn PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ OpenFn_POSTGRESQL_PASSWORD |
+ The password for the OpenFn PostgreSQL database |
+ |
+ |
+ |
+ |
+
+
+ WORKER_LIGHTNING_PUBLIC_KEY |
+ The public key for the worker lightning |
+ |
+ |
+ |
+ |
+
+
+ WORKER_SECRET |
+ The secret key for the worker |
+ |
+ |
+ |
+ |
+
+
+ OpenFn_IMAGE |
+ The image name for OpenFn |
+ |
+ |
+ |
+ |
+
+
+ OpenFn_WORKER_IMAGE |
+ The image name for OpenFn worker |
+ |
+ |
+ |
+ |
+
+
+
diff --git a/kafka-mapper-consumer/docker-compose.config.yml b/kafka-mapper-consumer/docker-compose.config.yml
index dd13d684..167bac85 100644
--- a/kafka-mapper-consumer/docker-compose.config.yml
+++ b/kafka-mapper-consumer/docker-compose.config.yml
@@ -27,7 +27,7 @@ services:
configs:
kafka-mapper-consumer-openhimConfig.js:
file: ./openhimConfig.js
- name: kafka-mapper-consumer-openhimConfig.js-${fhir_ig_importer_config_importer_openhimConfig_js_DIGEST:?err}
+ name: kafka-mapper-consumer-openhimConfig.js-${kafka_mapper_consumer_openhimConfig_js_DIGEST:?err}
labels:
name: kafka-mapper-consumer
kafka-mapper-consumer-consumer-ui-app.json:
diff --git a/openfn/docker-compose.dev.yml b/openfn/docker-compose.dev.yml
new file mode 100644
index 00000000..cf45c59e
--- /dev/null
+++ b/openfn/docker-compose.dev.yml
@@ -0,0 +1,13 @@
+version: '3.9'
+
+services:
+ openfn:
+ ports:
+ - target: 4000
+ published: 4000
+ mode: host
+ worker:
+ ports:
+ - target: 2222
+ published: 2222
+ mode: host
diff --git a/openfn/docker-compose.yml b/openfn/docker-compose.yml
new file mode 100644
index 00000000..3b55d1fe
--- /dev/null
+++ b/openfn/docker-compose.yml
@@ -0,0 +1,64 @@
+version: '3.9'
+
+services:
+ openfn:
+ image: ${OPENFN_IMAGE}
+ # This command runs a shell script that performs the following actions:
+ # 1. Executes the Lightning.Release.migrate() function to handle database migrations.
+ # 2. Sets up a user with the provided first name, last name, email, password, and role using the Lightning.Setup.setup_user function.
+ # - The user details are hardcoded with the first name "Test", last name "User", email "root@openhim.org", and password "instant101".
+ # - The role assigned to the user is "superuser".
+ # - The function also takes an API key from the environment variable ${OPENFN_API_KEY}.
+ # - Additionally, it sets up a schema with the name "openhim ndr" and type "http", including credentials and a base URL from environment variables.
+ # 3. Starts the Lightning application using the /app/bin/lightning start command.
+ command: >
+ sh -c "/app/bin/lightning eval 'Lightning.Release.migrate()' && /app/bin/lightning eval 'Lightning.Setup.setup_user(%{first_name: \"Test\",last_name: \"User\",email: \"root@openhim.org\",password: \"instant101\", role: :superuser}, \"${OPENFN_API_KEY}\" ,[%{name: \"openhim ndr\", schema: \"http\", body: %{username: \"${FHIR_SERVER_USERNAME}\", password: \"${FHIR_SERVER_PASSWORD}\", baseUrl: \"${FHIR_SERVER_BASE_URL}\"}}])' && /app/bin/lightning start"
+ deploy:
+ resources:
+ limits:
+ cpus: '${OPENFN_DOCKER_WEB_CPUS:-0}'
+ memory: '${OPENFN_DOCKER_WEB_MEMORY:-0}'
+ environment:
+ - DATABASE_URL=${OPENFN_DATABASE_URL}
+ - DISABLE_DB_SSL=${OPENFN_DISABLE_DB_SSL}
+ - IS_RESETTABLE_DEMO=${OPENFN_IS_RESETTABLE_DEMO}
+ - LISTEN_ADDRESS=${OPENFN_LISTEN_ADDRESS}
+ - LOG_LEVEL=${OPENFN_LOG_LEVEL}
+ - ORIGINS=${OPENFN_ORIGINS}
+ - PRIMARY_ENCRYPTION_KEY=${OPENFN_PRIMARY_ENCRYPTION_KEY}
+ - SECRET_KEY_BASE=${OPENFN_SECRET_KEY_BASE}
+ - WORKER_RUNS_PRIVATE_KEY=${OPENFN_WORKER_RUNS_PRIVATE_KEY}
+ - WORKER_SECRET=${OPENFN_WORKER_SECRET}
+ - KAFKA_TRIGGERS_ENABLED=${OPENFN_KAFKA_TRIGGERS_ENABLED}
+ healthcheck:
+ test: '${ DOCKER_WEB_HEALTHCHECK_TEST:-curl localhost:4000/health_check}'
+ interval: '10s'
+ timeout: '3s'
+ start_period: '5s'
+ retries: 3
+ networks:
+ - kafka_public
+ - postgres
+ worker:
+ image: ${OPENFN_WORKER_IMAGE}
+ deploy:
+ resources:
+ limits:
+ cpus: '${OPENFN_DOCKER_WORKER_CPUS:-0}'
+ memory: '${OPENFN_DOCKER_WORKER_MEMORY:-0}'
+ environment:
+ - WORKER_LIGHTNING_PUBLIC_KEY=${OPENFN_WORKER_LIGHTNING_PUBLIC_KEY}
+ - WORKER_SECRET=${OPENFN_WORKER_SECRET}
+ - NODE_ENV=production
+ command: [ 'pnpm', 'start:prod', '-l', 'ws://openfn:${URL_PORT-4000}/worker' ]
+ networks:
+ - kafka_public
+ - postgres
+
+networks:
+ kafka_public:
+ name: kafka_public
+ external: true
+ postgres:
+ name: postgres_public
+ external: true
diff --git a/openfn/importer/postgres/create-db.js b/openfn/importer/postgres/create-db.js
new file mode 100644
index 00000000..d95f4b10
--- /dev/null
+++ b/openfn/importer/postgres/create-db.js
@@ -0,0 +1,73 @@
+const { Pool } = require("pg");
+
+const user = process.env.POSTGRES_USER || "postgres";
+const host = process.env.POSTGRES_SERVICE || "postgres-1";
+const database = process.env.POSTGRES_DATABASE || "postgres";
+const password = process.env.POSTGRES_PASSWORD || "instant101";
+const port = process.env.POSTGRES_PORT || 5432;
+const newDb = process.env.NEW_DATABASE_NAME || "openfn";
+const newUser = process.env.NEW_DATABASE_USER || "openfn";
+const newUserPassword = process.env.NEW_DATABASE_PASSWORD || "instant101";
+
+const pool = new Pool({
+ user,
+ host,
+ database,
+ password,
+ port,
+});
+
+const tableQueries = [];
+const insertQueries = [];
+
+(async () => {
+ const client = await pool.connect();
+
+ const createDb = async (db) => {
+ //Check db exists before creating
+
+ const result = await client.query(
+ "SELECT 1 FROM pg_database WHERE datname = $1",
+ [db]
+ );
+
+ if (!result.rows.length) {
+ await client.query(`CREATE DATABASE ${db};`);
+
+ console.log(`Database '${db}' created successfully`);
+ } else {
+ console.log(`Database '${db}' already exists`);
+ }
+ };
+
+ const createUser = async () => {
+ const user = await client.query(
+ "SELECT 1 FROM pg_user WHERE usename = $1",
+ [newUser]
+ );
+
+ if (!user.rows.length) {
+ await client.query(
+ `CREATE USER ${newUser} WITH ENCRYPTED PASSWORD '${newUserPassword}';`
+ );
+ await client.query(
+ `GRANT ALL PRIVILEGES ON DATABASE ${newDb} TO ${newUser};`
+ );
+ console.log(`User ${newUser} created`);
+ }
+ };
+
+ try {
+ await createDb(newDb);
+
+ await createUser();
+ await Promise.all(tableQueries.map((query) => client.query(query)));
+
+ await Promise.all(insertQueries.map((query) => client.query(query)));
+ } catch (error) {
+ console.error("Error in db operations:", error.message);
+ } finally {
+ client.release();
+ pool.end();
+ }
+})();
diff --git a/openfn/importer/postgres/docker-compose.config.yml b/openfn/importer/postgres/docker-compose.config.yml
new file mode 100644
index 00000000..21ae873c
--- /dev/null
+++ b/openfn/importer/postgres/docker-compose.config.yml
@@ -0,0 +1,43 @@
+version: '3.9'
+
+services:
+ openfn_db_config:
+ image: node:erbium-alpine
+ command: sh -c "cd /importer && ls && npm i && ls && node create-db.js"
+ configs:
+ - target: /importer/package.json
+ source: package.json
+ - target: /importer/create-db.js
+ source: create-db.js
+ networks:
+ postgres:
+ environment:
+ POSTGRES_USER: ${POSTGRES_USER}
+ POSTGRES_SERVICE: ${POSTGRES_SERVICE}
+ POSTGRES_DATABASE: ${POSTGRES_DATABASE}
+ POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
+ POSTGRES_PORT: ${POSTGRES_PORT}
+ NEW_DATABASE_NAME: ${OPENFN_POSTGRESQL_DB}
+ NEW_DATABASE_USER: ${OPENFN_POSTGRESQL_USERNAME}
+ NEW_DATABASE_PASSWORD: ${OPENFN_POSTGRESQL_PASSWORD}
+ deploy:
+ replicas: 1
+ restart_policy:
+ condition: none
+
+networks:
+ postgres:
+ name: postgres_public
+ external: true
+
+configs:
+ package.json:
+ file: ./package.json
+ name: package.json-${package_json_DIGEST:?err}
+ labels:
+ name: openfn
+ create-db.js:
+ file: ./create-db.js
+ name: create_db.js-${create_db_js_DIGEST:?err}
+ labels:
+ name: openfn
diff --git a/openfn/importer/postgres/package-lock.json b/openfn/importer/postgres/package-lock.json
new file mode 100644
index 00000000..1daaed52
--- /dev/null
+++ b/openfn/importer/postgres/package-lock.json
@@ -0,0 +1,243 @@
+{
+ "name": "postgres-config",
+ "version": "0.0.1",
+ "lockfileVersion": 2,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "postgres-config",
+ "version": "0.0.1",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "pg": "^8.11.3"
+ }
+ },
+ "node_modules/pg": {
+ "version": "8.12.0",
+ "resolved": "https://registry.npmjs.org/pg/-/pg-8.12.0.tgz",
+ "integrity": "sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==",
+ "dependencies": {
+ "pg-connection-string": "^2.6.4",
+ "pg-pool": "^3.6.2",
+ "pg-protocol": "^1.6.1",
+ "pg-types": "^2.1.0",
+ "pgpass": "1.x"
+ },
+ "engines": {
+ "node": ">= 8.0.0"
+ },
+ "optionalDependencies": {
+ "pg-cloudflare": "^1.1.1"
+ },
+ "peerDependencies": {
+ "pg-native": ">=3.0.1"
+ },
+ "peerDependenciesMeta": {
+ "pg-native": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/pg-cloudflare": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.1.1.tgz",
+ "integrity": "sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==",
+ "optional": true
+ },
+ "node_modules/pg-connection-string": {
+ "version": "2.6.4",
+ "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.6.4.tgz",
+ "integrity": "sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA=="
+ },
+ "node_modules/pg-int8": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz",
+ "integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==",
+ "engines": {
+ "node": ">=4.0.0"
+ }
+ },
+ "node_modules/pg-pool": {
+ "version": "3.6.2",
+ "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.6.2.tgz",
+ "integrity": "sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==",
+ "peerDependencies": {
+ "pg": ">=8.0"
+ }
+ },
+ "node_modules/pg-protocol": {
+ "version": "1.6.1",
+ "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.6.1.tgz",
+ "integrity": "sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg=="
+ },
+ "node_modules/pg-types": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz",
+ "integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==",
+ "dependencies": {
+ "pg-int8": "1.0.1",
+ "postgres-array": "~2.0.0",
+ "postgres-bytea": "~1.0.0",
+ "postgres-date": "~1.0.4",
+ "postgres-interval": "^1.1.0"
+ },
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/pgpass": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/pgpass/-/pgpass-1.0.5.tgz",
+ "integrity": "sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==",
+ "dependencies": {
+ "split2": "^4.1.0"
+ }
+ },
+ "node_modules/postgres-array": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz",
+ "integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==",
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/postgres-bytea": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz",
+ "integrity": "sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/postgres-date": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz",
+ "integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/postgres-interval": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz",
+ "integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==",
+ "dependencies": {
+ "xtend": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/split2": {
+ "version": "4.2.0",
+ "resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz",
+ "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==",
+ "engines": {
+ "node": ">= 10.x"
+ }
+ },
+ "node_modules/xtend": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
+ "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
+ "engines": {
+ "node": ">=0.4"
+ }
+ }
+ },
+ "dependencies": {
+ "pg": {
+ "version": "8.12.0",
+ "resolved": "https://registry.npmjs.org/pg/-/pg-8.12.0.tgz",
+ "integrity": "sha512-A+LHUSnwnxrnL/tZ+OLfqR1SxLN3c/pgDztZ47Rpbsd4jUytsTtwQo/TLPRzPJMp/1pbhYVhH9cuSZLAajNfjQ==",
+ "requires": {
+ "pg-cloudflare": "^1.1.1",
+ "pg-connection-string": "^2.6.4",
+ "pg-pool": "^3.6.2",
+ "pg-protocol": "^1.6.1",
+ "pg-types": "^2.1.0",
+ "pgpass": "1.x"
+ }
+ },
+ "pg-cloudflare": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.1.1.tgz",
+ "integrity": "sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==",
+ "optional": true
+ },
+ "pg-connection-string": {
+ "version": "2.6.4",
+ "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.6.4.tgz",
+ "integrity": "sha512-v+Z7W/0EO707aNMaAEfiGnGL9sxxumwLl2fJvCQtMn9Fxsg+lPpPkdcyBSv/KFgpGdYkMfn+EI1Or2EHjpgLCA=="
+ },
+ "pg-int8": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz",
+ "integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw=="
+ },
+ "pg-pool": {
+ "version": "3.6.2",
+ "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.6.2.tgz",
+ "integrity": "sha512-Htjbg8BlwXqSBQ9V8Vjtc+vzf/6fVUuak/3/XXKA9oxZprwW3IMDQTGHP+KDmVL7rtd+R1QjbnCFPuTHm3G4hg==",
+ "requires": {}
+ },
+ "pg-protocol": {
+ "version": "1.6.1",
+ "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.6.1.tgz",
+ "integrity": "sha512-jPIlvgoD63hrEuihvIg+tJhoGjUsLPn6poJY9N5CnlPd91c2T18T/9zBtLxZSb1EhYxBRoZJtzScCaWlYLtktg=="
+ },
+ "pg-types": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz",
+ "integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==",
+ "requires": {
+ "pg-int8": "1.0.1",
+ "postgres-array": "~2.0.0",
+ "postgres-bytea": "~1.0.0",
+ "postgres-date": "~1.0.4",
+ "postgres-interval": "^1.1.0"
+ }
+ },
+ "pgpass": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/pgpass/-/pgpass-1.0.5.tgz",
+ "integrity": "sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==",
+ "requires": {
+ "split2": "^4.1.0"
+ }
+ },
+ "postgres-array": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz",
+ "integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA=="
+ },
+ "postgres-bytea": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz",
+ "integrity": "sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w=="
+ },
+ "postgres-date": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz",
+ "integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q=="
+ },
+ "postgres-interval": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz",
+ "integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==",
+ "requires": {
+ "xtend": "^4.0.0"
+ }
+ },
+ "split2": {
+ "version": "4.2.0",
+ "resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz",
+ "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg=="
+ },
+ "xtend": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
+ "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ=="
+ }
+ }
+}
diff --git a/openfn/importer/postgres/package.json b/openfn/importer/postgres/package.json
new file mode 100644
index 00000000..ebef53f8
--- /dev/null
+++ b/openfn/importer/postgres/package.json
@@ -0,0 +1,11 @@
+{
+ "name": "postgres-config",
+ "version": "0.0.1",
+ "main": "index.js",
+ "scripts": {},
+ "author": "Jembi",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "pg": "^8.11.3"
+ }
+}
diff --git a/openfn/importer/workflows/docker-compose.config.yml b/openfn/importer/workflows/docker-compose.config.yml
new file mode 100644
index 00000000..eb1395c5
--- /dev/null
+++ b/openfn/importer/workflows/docker-compose.config.yml
@@ -0,0 +1,37 @@
+version: '3.9'
+
+services:
+ openfn_workflow_config:
+ image: node:18-alpine3.20
+ command: sh -c "cd /workflows && ls && npm install -g @openfn/cli && cat config.json && openfn deploy -c config.json --no-confirm"
+ configs:
+ - target: /workflows/project.yaml
+ source: project.yaml
+ - target: /workflows/config.json
+ source: config.json
+ networks:
+ postgres:
+ environment:
+ OPENFN_API_KEY: ${OPENFN_API_KEY}
+ OPENFN_API_URL: ${OPENFN_ENDPOINT}
+ deploy:
+ replicas: 1
+ restart_policy:
+ condition: none
+
+networks:
+ postgres:
+ name: postgres_public
+ external: true
+
+configs:
+ config.json:
+ file: ./example/config.json
+ name: config.json-${config_json_DIGEST:?err}
+ labels:
+ name: configjson
+ project.yaml:
+ file: ./example/project.yaml
+ name: project.yaml-${project_yaml_DIGEST:?err}
+ labels:
+ name: project
diff --git a/openfn/importer/workflows/example/config.json b/openfn/importer/workflows/example/config.json
new file mode 100644
index 00000000..76a189df
--- /dev/null
+++ b/openfn/importer/workflows/example/config.json
@@ -0,0 +1,6 @@
+{
+ "specPath": "project.yaml",
+ "statePath": "state.json",
+ "apiKey": "apiKey",
+ "endpoint": "http://openfn:4000"
+}
diff --git a/openfn/importer/workflows/example/project.yaml b/openfn/importer/workflows/example/project.yaml
new file mode 100644
index 00000000..e85e8e98
--- /dev/null
+++ b/openfn/importer/workflows/example/project.yaml
@@ -0,0 +1,1310 @@
+name: jembi-cdc-ahie
+description: POC test project for ART patient use case
+credentials:
+ root@openhim.org-openhim-ndr:
+ name: openhim ndr
+ owner: root@openhim.org
+workflows:
+ 1-process-cdr-bundle:
+ name: 1-process-cdr-bundle
+ jobs:
+ parse-bundle:
+ name: parse bundle
+ adaptor: "@openfn/language-common@2.0.1"
+ credential: null
+ body: |
+ // this will take an incoming bundle and sort it into different resource types
+ fn((state) => {
+ let data = state.data.body;
+ if (typeof data === "string") {
+ data = JSON.parse(data)
+ }
+
+ const next = {
+ entry: data.entry
+ }
+
+ console.log(`Loading ${next.entry.length} bundle items from input`)
+
+ return next;
+ })
+
+ group($.entry, 'resource.resourceType')
+
+ fn((state) => {
+ const counts = Object.keys(state.data).map((k) => `${k} (${state.data[k].length})`);
+ console.log(`Sorted bundles into the following groups: ${counts.join('\n ')}`)
+ return {
+ inputs: state.data,
+ }
+ })
+
+ Register-Organization:
+ name: Register Organization
+ adaptor: "@openfn/language-http@6.4.3"
+ credential: root@openhim.org-openhim-ndr
+ body: |
+ //For the POC test only we need to register the Organization first
+ //... for each Bundle... and AFTER we can process the Bundle
+ fn(state => {
+ const orgId =
+ state.inputs.Patient[0].resource.managingOrganization.reference.replace(
+ 'Organization/',
+ ''
+ );
+ console.log('Setting up test organization', orgId);
+
+ const mappedOrganization = {
+ resourceType: 'Organization',
+ id: orgId,
+ meta: {
+ profile: ['http://moh.gov.et/fhir/hiv/StructureDefinition/organization'],
+ },
+ text: {
+ status: 'generated',
+ div: ' identifier: http://moh.gov.et/fhir/hiv/identifier/mohid
/MOH001, http://moh.gov.et/fhir/hiv/identifier/hfuid
/HFUID001
active: true
type: Current Healthcare Provider
name: Meshwalkiya Health Center
address: Meshulekia Kirkos Addis Ababa
',
+ },
+ identifier: [
+ {
+ system: 'http://moh.gov.et/fhir/hiv/identifier/mohid',
+ value: 'MOH001',
+ },
+ {
+ system: 'http://moh.gov.et/fhir/hiv/identifier/hfuid',
+ value: 'HFUID001',
+ },
+ ],
+ active: true,
+ type: [
+ {
+ coding: [
+ {
+ system: 'http://snomed.info/sct',
+ code: '257622000',
+ },
+ ],
+ text: 'Current Healthcare Provider',
+ },
+ ],
+ name: `Organization ${orgId}`,
+ address: [
+ {
+ line: ['Meshulekia'],
+ city: 'Kirkos',
+ district: 'Kirkos woreda 9',
+ state: 'Addis Ababa',
+ },
+ ],
+ };
+
+ return { ...state, mappedOrganization, orgId };
+ });
+
+ // basic HTTP put should do it
+ put(`/fhir/Organization/${$.orgId}`, {
+ body: $.mappedOrganization,
+ headers: { 'content-type': 'application/json' },
+ });
+
+ fn(state => {
+ console.log('Organization registered!');
+ return state;
+ });
+
+ map-encounters:
+ name: map encounters
+ adaptor: "@openfn/language-fhir-ndr-et@0.1.0"
+ credential: null
+ body: |
+ fn(state => {
+ console.log('Parsing encounters...');
+ state.encounters = [];
+ return state;
+ });
+
+ each($.inputs.Encounter, state => {
+ const input = state.data.resource;
+ console.log('Mapping encounter ', input.id);
+
+ util.setSystemMap({
+ 'http://cdr.aacahb.gov.et/Encounter':
+ 'http://moh.gov.et/fhir/hiv/identifier/encounter',
+ });
+
+ const visitType = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/visit-type'
+ );
+
+ const [_, patientId] = input.subject.reference.split('Patient/');
+ const patient = state.inputs.Patient.find(
+ e => e.resource.id === patientId
+ )?.resource;
+ if (!patient) {
+ console.log(
+ `WARNING: COULD NOT FIND PATIENT ${patientId ?? input.subject} IN BUNDLE`
+ );
+ }
+
+ // TODO: Future phase work out whether this is a target-facility-encounter or a entry-from-outside-target-facility-encounter
+ if (visitType) {
+ const serviceType = input.serviceType?.coding.find(
+ item =>
+ item.system === 'http://terminology.hl7.org/CodeSystem/service-type'
+ );
+
+ const result = builders.encounter('target-facility-encounter', {
+ id: input.id,
+ status: input.status,
+
+ class:
+ input.class ??
+ util.coding(
+ 'Absent-or-unknown',
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/absent-unknown-code-system'
+ ),
+ identifier: input.identifier,
+ serviceType: serviceType && {
+ coding: serviceType,
+ },
+ period: input.period,
+ subject: input.subject,
+ serviceProvider: patient.managingOrganization,
+ type: input.type,
+ });
+
+ // Handle the visit type extension manually
+ util.addExtension(
+ result.type[0],
+ 'http://moh.gov.et/fhir/hiv/StructureDefinition/encounter-visit-type',
+ util.concept([
+ visitType.valueString.toLowerCase(),
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/encounter-visit-type-code-system',
+ ])
+ ),
+ state.encounters.push(result);
+ } else {
+ console.log('WARNING: no visit type found for Encounter ', input.id);
+ }
+
+ return state;
+ });
+
+ fn(state => {
+ console.log(
+ `Finished parsing Encounters! Created ${state.encounters.length} Encounters`
+ );
+
+ return state;
+ });
+
+ map-patients:
+ name: map patients
+ adaptor: "@openfn/language-fhir-ndr-et@0.1.0"
+ credential: null
+ body: |
+ // Handle patients (and related persons)
+
+ // This import is needed for code assist in VSC
+ // It is harmless to include in job code, but not needed in lightning
+ import { b, util } from '@openfn/language-fhir-ndr-et';
+
+ const createId = (base, type) => {
+ let id = `${base}${type}`;
+ if (id.length <= 64) {
+ // If the id is too long, cut the difference out of the base id
+ id = `${base.substr(0, base.length - (id.length - 64))}${type}`;
+ }
+ return id;
+ };
+
+ // Flag a resource as being newly created
+ // This affects the final bundle
+ // the __new__ property will be removed at the end
+ const markNew = resource => {
+ resource.__new__ = true;
+ return resource;
+ };
+
+ // Util function to map residential type on the address
+ const mapAddress = a => {
+ if (/rural/i.test(a.text)) {
+ const { text, ...address } = a;
+ return {
+ ...address,
+ residentialType: util.concept('Rural', [
+ '224804009',
+ 'http://snomed.info/sct',
+ ]),
+ };
+ } else if (/urban/i.test(a.text)) {
+ const { text, ...address } = a;
+ return {
+ ...address,
+ residentialType: util.concept('Urban', [
+ '224807002',
+ 'http://snomed.info/sct',
+ ]),
+ };
+ }
+ return a;
+ };
+
+ const populationMap = {
+ 'general population': {
+ code: 'General-Population',
+ display: 'General Population',
+ },
+ prisoners: {
+ code: 'Prisoner',
+ display: 'Prisoner',
+ },
+ 'children of plhiv': {
+ code: 'Children-Of-PLHIV',
+ display: 'Children of PLHIV',
+ },
+ 'partners of plhiv': {
+ code: 'Partners-Of-PLHIV',
+ display: 'Partners of PLHIV',
+ },
+ 'discordant couples': {
+ code: 'Discordant-Couples',
+ display: 'Discordant Couples',
+ },
+ 'other marps': {
+ code: 'Other-MARPS',
+ display: 'Other MARPS',
+ },
+ 'female commercial sex workers': {
+ code: 'Female-Commercial-Sex-Workers',
+ display: 'Female Commercial Sex Workers',
+ },
+ 'sex worker': {
+ code: 'Female-Commercial-Sex-Workers',
+ display: 'Female Commercial Sex Workers',
+ },
+ 'distance drivers': {
+ code: 'Distance-Drivers',
+ display: 'Distance Drivers',
+ },
+ 'mobile or daily laborers': {
+ code: 'Mobile-Daily-Laborers',
+ display: 'Mobile or Daily Laborers',
+ },
+ 'ovc or aids orphaned': {
+ code: 'OVC-AIDS-Orphaned',
+ display: 'OVC or AIDS Orphaned',
+ },
+ tg: {
+ code: 'TG',
+ display: 'TG',
+ },
+ reguees: {
+ code: 'Refugee',
+ display: 'Refugee',
+ },
+ 'people who inject drug': {
+ code: 'People-Who-Inject-Drug',
+ display: 'People who Inject drug',
+ },
+ msm: {
+ code: 'MSM',
+ display: 'MSM',
+ },
+ };
+
+ const educationMap = {
+ 'no education': {
+ code: 'LA35-1',
+ display: 'No schooling',
+ },
+ primary: {
+ code: 'LA32466-7',
+ display: 'Primary school education',
+ },
+ secondary: {
+ code: 'LA32467-5',
+ display: 'High school education',
+ },
+ tertiary: {
+ code: 'LA15568-1',
+ display: 'Graduated from a college or university',
+ },
+ tvet: {
+ code: 'LA39-3',
+ display: 'Technical or trade school',
+ },
+ 'other, specify': {
+ code: 'LA46-8',
+ display: 'Other',
+ },
+ assoc: {
+ code: 'LA39-3',
+ display: 'Technical or trade school',
+ },
+ bd: {
+ code: 'LA15568-1',
+ display: 'Graduated from a college or university',
+ },
+ elem: {
+ code: 'LA32466-7',
+ display: 'Primary school education',
+ },
+ gd: {
+ code: 'LA15568-1',
+ display: 'Graduated from a college or university',
+ },
+ hs: {
+ code: 'LA32467-5',
+ display: 'High school education',
+ },
+ pb: {
+ code: 'LA15568-1',
+ display: 'Graduated from a college or university',
+ },
+ postg: {
+ code: 'LA15568-1',
+ display: 'Graduated from a college or university',
+ },
+ scol: {
+ code: 'LA32467-5',
+ display: 'High school education',
+ },
+ sec: {
+ code: 'LA32467-5',
+ display: 'High school education',
+ },
+ };
+
+ fn(state => {
+ console.log('Parsing patients...');
+ state.patients ??= [];
+ state.observations ??= [];
+ state.relations ??= [];
+
+ // This will map identifier systems for us automatically
+ util.setSystemMap({
+ 'http://cdr.aacahb.gov.et/SmartCareID':
+ 'http://moh.gov.et/fhir/hiv/identifier/SmartCareID',
+ 'http://cdr.aacahb.gov.et/MRN': 'http://moh.gov.et/fhir/hiv/identifier/MRN',
+ 'http://cdr.aacahb.gov.et/UAN': 'http://moh.gov.et/fhir/hiv/identifier/UAN',
+ 'http://terminology.hl7.org/CodeSystem/v2-0131':
+ 'http://terminology.hl7.org/CodeSystem/v2-0131',
+ });
+
+ return state;
+ });
+
+ // Create an NDR patient for each CDR patient
+ each($.inputs.Patient, state => {
+ const input = state.data.resource;
+
+ console.log(` Processing patient ${input.id}`);
+
+ const religion = util.findExtension(
+ input,
+ 'http://hl7.org/fhir/StructureDefinition/patient-religion'
+ )?.valueCodeableConcept.coding[0];
+
+ const result = b.patient('patient', {
+ id: input.id,
+ religion:
+ religion &&
+ util.concept(
+ religion.display,
+ util.coding(
+ religion.code,
+ 'http://terminology.hl7.org/CodeSystem/v3-ReligiousAffiliation'
+ )
+ ),
+ identifier: input.identifier,
+ name: input.name,
+ telecom: input.telecom,
+ gender: input.gender,
+ birthDate: input.birthDate,
+ maritalStatus: input.maritalStatus,
+ managingOrganization: input.managingOrganization,
+ address: input.address?.map(mapAddress),
+ });
+
+ state.patients.push(result);
+
+ return state;
+ });
+
+ // Map the observations for each patient
+ each($.inputs.Patient, state => {
+ const input = state.data.resource;
+ console.log(` Processing observations for patient ${input.id}`);
+ // Assumes one encounter per bundle
+ const encounter = state.inputs.Encounter[0].resource;
+
+ const occupation = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/Occupation'
+ );
+ if (occupation) {
+ state.observations.push(
+ markNew(
+ b.observation('patient-occupation-observation', {
+ id: createId(input.id, 'PatientOccupation'),
+ // what to do about id?
+ status: input.status ?? 'final',
+
+ effective: encounter.period?.start,
+ encounter,
+ performer: input.managingOrganization ?? state.orgId,
+
+ // Map the occupation string without a coding
+ value: util.concept(occupation.valueString, [
+ occupation.valueString,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/occupation-code-system',
+ ]),
+
+ subject: input,
+ })
+ )
+ );
+ }
+
+ const education = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/EducationalLevel'
+ );
+ if (education) {
+ const ed =
+ education.valueString ?? education.valueCodeableConcept?.coding[0].code;
+ const edLabel =
+ education.valueString ??
+ education.valueCodeableConcept?.coding[0].display;
+ const mapped = educationMap[ed.toLowerCase()];
+ if (!mapped) {
+ console.error('ERROR: could not find mapped education for ', ed);
+ }
+ const value = util.concept(edLabel, [mapped.code, 'http://loinc.org']);
+
+ state.observations.push(
+ markNew(
+ b.observation('highest-education-observation', {
+ id: createId(input.id, 'HighestEducation'),
+ status: input.status ?? 'final',
+
+ effective: encounter.period?.start,
+ encounter,
+ performer: input.managingOrganization ?? state.orgId,
+ subject: input,
+ value,
+ })
+ )
+ );
+ }
+
+ const popGroup = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/TargetPopulationGroup'
+ );
+ if (popGroup) {
+ const group =
+ popGroup.valueString ?? popGroup.valueCodeableConcept?.coding[0].text;
+ const mapped = populationMap[group.toLowerCase()];
+ if (!mapped) {
+ console.error(
+ 'ERROR: could not find mapped population group for ',
+ group
+ );
+ }
+ const value = util.concept(mapped.display, [
+ mapped.code,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/target-population-category-code-system',
+ ]);
+
+ state.observations.push(
+ markNew(
+ b.observation('target-population-observation', {
+ id: createId(input.id, 'TargetPopulation'),
+ status: input.status ?? 'final',
+ effective: encounter.period?.start,
+ encounter,
+ performer: input.managingOrganization ?? state.orgId,
+ subject: input,
+ value,
+ })
+ )
+ );
+ }
+
+ return state;
+ });
+
+ // Find related people
+ each($.inputs.RelatedPerson, state => {
+ const input = state.data.resource;
+
+ console.log(` Processing related person ${input.id}`);
+
+ let patient;
+ if (input.patient) {
+ // Extract the actual ID out of the reference
+ const [_, patientId] = input.patient.reference.split('Patient/');
+ patient = state.inputs.Patient.find(
+ e => e.resource.id === patientId
+ )?.resource;
+ if (patient) {
+ console.log(` Found patient ${input.patient.reference} in bundle`);
+ } else {
+ console.log(
+ 'WARNING: COULD NOT FIND MATCHING PATIENT FOR ',
+ input.patient.reference
+ );
+ return state;
+ }
+ }
+
+ const r = b.relatedPerson('related-person', {
+ ...input, // this should cover the basics
+ id: input.id,
+ extension: input.extension,
+
+ address: input.address?.map(mapAddress),
+
+ relationship: input.relationship.reduce(
+ (obj, next) => Object.assign(obj, next),
+ {}
+ ),
+ });
+
+ state.relations.push(r);
+
+ return state;
+ });
+
+ fn(state => {
+ console.log(
+ `Finished parsing patients! Created ${state.patients.length} Patients, ${state.observations.length} Observations and ${state.relations.length} Related Persons`
+ );
+
+ return state;
+ });
+
+ map-medications:
+ name: map medications
+ adaptor: "@openfn/language-fhir-ndr-et@0.1.0"
+ credential: null
+ body: |
+ const createId = (base, type) => {
+ let id = `${base}${type}`;
+ if (id.length > 64) {
+ // If the id is too long, cut the difference out of the base id
+ id = `${base.substr(0, base.length - (id.length - 64))}${type}`;
+ }
+ return id;
+ };
+
+ // Flag a resource as being newly created
+ // This affects the final bundle
+ // the __new__ property will be removed at the end
+ const markNew = resource => {
+ resource.__new__ = true;
+ };
+
+ const findServiceProvider = (encounter, patient) =>
+ encounter?.serviceProvider ?? patient?.managingOrganization;
+
+ const categoryTypeMap = {
+ '1st switch': 'First-Switch',
+ '2nd switch': 'Second-Switch',
+ '3rd switch': 'Third-Switch',
+ '1st substitution': 'First-Substitute',
+ '2nd substitution': 'Second-Substitute',
+ '3rd substitution': 'Third-Substitute',
+ '4th substitution': 'Fourth-Substitute',
+ '5th substitution': 'Fifth-Substitute',
+ '6th substitution': 'Sixth-Substitute',
+ };
+
+ const treatmentFailureMap = {
+ 'Virologic failure': 'Virologic-Failure',
+ 'Clinical failure': 'Clinical-Failure',
+ 'Immunologic failure': 'Immunologic-Failure',
+ };
+
+ const substituteReasonMap = {
+ 'Risk of pregnancy': 'LA6531-3',
+ Pregnancy: 'LA6530-5',
+ 'New drug available': 'LA6533-9',
+ 'Due to new TB': 'LA6532-1',
+ 'Drug stock-out': 'LA6534-7',
+ 'Toxicity/Side effects': 'LA6529-7',
+ Other: 'LA6535-4',
+ Anemia: 'LA6535-4',
+ };
+
+ const stoppedReasonMap = {
+ 'toxicity/side effects': {
+ code: 'LA6529-7',
+ text: 'Toxicity/Side Effects',
+ },
+ pregnancy: {
+ code: 'A6530-5',
+ text: 'Pregnancy',
+ },
+ 'planned treatment interruption': {
+ code: '18733-6',
+ text: 'Attending Progress note',
+ },
+ 'other patient decisions': {
+ code: '18733-6',
+ text: 'Attending Progress note',
+ },
+ 'poor adherence': {
+ code: 'LA31432-0',
+ text: 'Not adherent to treatment',
+ },
+ 'planned treatment interruption': {
+ code: 'LP56812-8',
+ text: 'Medication regimen.status',
+ },
+ 'patient lack finances': {
+ code: 'LA28316-0',
+ text: 'Financial concerns',
+ },
+ 'other patient decisions': {
+ code: '51855-5',
+ text: 'Patient Note',
+ },
+ 'illness/hospitalization': {
+ code: '34130-5',
+ text: 'Hospital Progress note',
+ },
+ 'drugs out of stock': {
+ code: 'LA6534-7',
+ text: 'Drug Out of Stock',
+ },
+ other: {
+ code: 'LA46-8',
+ text: 'Other',
+ },
+ };
+
+ fn(state => {
+ console.log('Parsing medications...');
+ state.medicationDispense = [];
+ state.medication = [];
+ state.observations ??= [];
+ state.careplans ??= [];
+ state.requests ??= [];
+ state.admin ??= [];
+
+ return state;
+ });
+
+ const extractMedication = (state, input) => {
+ const { coding, text } = input.medicationCodeableConcept;
+
+ const codes = coding.find(
+ c => c.system === 'http://cdr.aacahb.gov.et/hiv-regimen-codes'
+ );
+ const c = [
+ codes.code,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-treatment-code-system',
+ ];
+
+ let value;
+ if (text) {
+ value = util.concept(text, c);
+ } else {
+ value = util.concept(codes.code, c);
+ }
+
+ const m = b.medication('arv-regimen-medication', {
+ id: `${input.id}Medication`,
+ status: 'active',
+ code: value,
+ });
+ markNew(m);
+ return m;
+ };
+
+ const extractAdmin = (state, input, encounter, medication, request) => {
+ const end = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/dose-end-date'
+ );
+ if (end) {
+ const ma = b.medicationAdministration('arv-medication-administration', {
+ id: createId(input.id, 'ARVMedicationAdministration'),
+ status: 'completed',
+ medication: util.ref(medication),
+ subject: input.subject,
+ context: input.context,
+ request,
+ effective: end.valueDateTime,
+ //effective: encounter.period,
+ });
+ markNew(ma);
+ return ma;
+ }
+ };
+
+ const extractRequest = (
+ state,
+ input,
+ careplan,
+ medication,
+ dispense,
+ encounter,
+ observation
+ ) => {
+ const r = b.medicationRequest('arv-medication-request', {
+ id: createId(input.id, 'ARVMedicationRequest'),
+ status: 'completed',
+ intent: 'order',
+ medication,
+ basedOn: careplan,
+ dispenseRequest: {
+ quantity: {
+ value: parseInt(input.quantity.value),
+ unit: 'TAB',
+ system: 'http://terminology.hl7.org/CodeSystem/v3-orderableDrugForm',
+ code: 'TAB',
+ },
+ },
+ subject: input.subject,
+ encounter: input.context,
+ authoredOn: encounter.period?.start,
+ identifier: {
+ value: dispense.id,
+ system: 'http://moh.gov.et/fhir/hiv/identifier/medication',
+ },
+ reasonReference: util.reference(observation),
+ });
+
+ markNew(r);
+
+ return r;
+ };
+
+ const extractCareplan = (state, input, encounter, patient) => {
+ const cp = b.carePlan('art-follow-up-careplan', {
+ id: createId(input.id, 'ARTFollowUpCareplan'),
+ status: 'active',
+ intent: 'order',
+ subject: input.subject,
+ encounter: input.context,
+ created: encounter.period?.start,
+ performer: findServiceProvider(encounter, patient),
+ category: [
+ {
+ coding: [
+ {
+ system:
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/care-plan-category-code-system',
+ code: 'art-follow-up-care-plan',
+ },
+ ],
+ text: 'ART',
+ },
+ ],
+ });
+ markNew(cp);
+ return cp;
+ };
+ const extractCareplanExtensions = (state, input, plan, encounter) => {
+ const nextVisit = util.findExtension(
+ encounter,
+ 'http://cdr.aacahb.gov.et/next-visit'
+ );
+ if (nextVisit) {
+ util.addExtension(
+ plan.activity[0],
+ 'http://moh.gov.et/fhir/hiv/StructureDefinition/care-plan-next-visit',
+ nextVisit.valueDateTime
+ );
+ }
+ };
+
+ // This maps the actual dispense object
+ const mapDispense = (_state, input /* Dispense resource*/, encounter) => {
+ // Create a matching dispense
+ const handover = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/dose-end-date'
+ );
+
+ return b.medicationDispense('arv-medication-dispense', {
+ id: input.id,
+ status: input.status,
+ subject: input.subject,
+ context: input.context,
+ quantity: input.quantity && {
+ ...input.quantity,
+ value: parseInt(input.quantity.value),
+ unit: 'TAB',
+ code: 'TAB',
+ },
+ daysSupply: input.daysSupply && {
+ ...input.daysSupply,
+ value: parseInt(input.daysSupply.value),
+ unit: 'Days',
+ code: 'd',
+ },
+ whenHandedOver: handover?.valueDateTime,
+ medication: input.reference,
+ // missing properties will be added later in the script
+ });
+ };
+
+ // This pulls out a bunch of observations
+ const extractObservations = (state, input, encounter, patient) => {
+ const result = {};
+ let stopReasonObs;
+
+ const stopReason = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/medication-stopped-reason'
+ );
+ if (stopReason) {
+ console.log(
+ `creating art-followup-stopped-reasons-observation for ${input.id}`
+ );
+
+ //console.log('stopReason::', JSON.stringify(stopReason, null, 2));
+
+ const stopReasonObs = b.observation(
+ 'art-followup-stopped-reasons-observation',
+ {
+ id: createId(input.id, 'ARTFollowupStatusStoppedReasonObservation'),
+ status: 'final',
+ subject: input.subject,
+ encounter,
+ effective: encounter?.period?.start,
+ performer: findServiceProvider(encounter, patient),
+ }
+ );
+
+ let text = stopReason.valueString ?? stopReason.valueCodeableConcept?.text;
+
+ //('stopReason text::', JSON.stringify(text, null, 2));
+ if (!text) {
+ console.error(`No stop reason text found for `, input.id);
+ }
+
+ const value = {};
+ const mapped = stoppedReasonMap[text.toLowerCase()];
+ //console.log('mapped stopReasonObs::', JSON.stringify(mapped, null, 2));
+ if (mapped) {
+ util.addExtension(
+ value,
+ 'http://moh.gov.et/fhir/hiv/StructureDefinition/reason-art-stopped',
+ util.concept(mapped.text, [mapped.code, 'http://loinc.org'])
+ );
+
+ //console.log('stopReasonObs::', JSON.stringify(stopReasonObs, null, 2));
+
+ stopReasonObs['valueCodeableConcept'] = value;
+
+ // console.log(
+ // 'NEW stopReasonObs::',
+ // JSON.stringify(stopReasonObs, null, 2)
+ // );
+
+ markNew(stopReasonObs);
+
+ // We have to write the concept like this because it's a strange structure
+ // the adaptor should handle this a lot better really
+ state.observations.push(stopReasonObs);
+ console.log('Stopped Obs pushed...');
+ } else {
+ console.error('WARNING: NO MAPPED VALUE FOR STOP-REASON: ', text);
+ }
+ }
+
+ if (input.statusReasonCodeableConcept) {
+ const reason = Array.isArray(input.statusReasonCodeableConcept)
+ ? input.statusReasonCodeableConcept[0]
+ : input.statusReasonCodeableConcept;
+ const o = b.observation('art-followup-status-observation', {
+ id: createId(input.id, 'ARTFollowupStatusObservation'),
+ status: 'final',
+ subject: input.subject,
+ encounter,
+ effective: encounter?.period?.start,
+ value: util.concept([
+ reason.coding[0].code,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/art-follow-up-status-code-system',
+ ]),
+ hasMember: stopReasonObs ? util.ref(stopReasonObs) : undefined,
+ performer: findServiceProvider(encounter, patient),
+ });
+ result.followUpStatus = o;
+ state.observations.push(o);
+ }
+
+ const extSwitchType = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/switch-type'
+ );
+ const extSwitchReason = util.findExtension(
+ input,
+ 'http://cdr.aacahb.gov.et/switch-reason'
+ );
+
+ let changeObs;
+ let changeTypeObs;
+ let changeReasonObs;
+
+ if (input.substitution || extSwitchType) {
+ changeObs = b.observation('arv-regimen-changed-observation', {
+ id: createId(input.id, 'ARVRegimenChange'),
+ status: 'final',
+ subject: input.subject,
+ encounter,
+ effective: encounter?.period,
+ performer: findServiceProvider(encounter, patient),
+
+ value: util.concept([
+ input.substitution ? 'Substituted' : 'Switched',
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-regimen-change-category-code-system',
+ ]),
+ // hasMember will be set at the end
+ });
+ markNew(changeObs);
+ state.observations.push(changeObs);
+ }
+
+ if (input.substitution?.type || extSwitchType) {
+ let value;
+ if (extSwitchType) {
+ const code = categoryTypeMap[extSwitchType.valueString];
+ if (!code) {
+ console.error('No mapped switch code found for ', input.id);
+ console.log(extSwitchType);
+ }
+ value = util.concept([
+ code,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-regimen-switch-type-code-system',
+ ]);
+ } else {
+ const code = categoryTypeMap[input.substitution?.type.text];
+ if (!code) {
+ console.error('No mapped switch code found for ', input.id);
+ console.log(input.substitution);
+ }
+ value = util.concept([
+ code,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-regimen-substitute-type-code-system',
+ ]);
+ }
+
+ changeTypeObs = b.observation('arv-change-category-type-observation', {
+ id: createId(input.id, 'ARVRegimenChangeCategoryType'),
+ status: 'final',
+ subject: input.subject,
+ encounter,
+ effective: encounter?.period?.start,
+ performer: findServiceProvider(encounter, patient),
+ value,
+ // hasmember is added later
+ });
+ markNew(changeTypeObs);
+ state.observations.push(changeTypeObs);
+ }
+
+ if (extSwitchReason ?? input.substitution?.reason) {
+ let reason;
+ if (extSwitchReason ?? input.substitution?.reason) {
+ reason =
+ extSwitchReason?.valueCodeableConcept?.coding[0].code ??
+ extSwitchReason?.valueString;
+ } else {
+ reason = input.substitution?.reason[0].text;
+ }
+
+ let reasonLabel;
+ if (extSwitchReason) {
+ reasonLabel =
+ extSwitchReason?.valueCodeableConcept?.coding[0].display ??
+ extSwitchReason?.valueString;
+ } else {
+ reasonLabel = input.substitution?.reason[0].text;
+ }
+ //console.log('reasonLabel ::', reasonLabel);
+
+ if (!reason) {
+ console.error('No reason observation found for ', input.id);
+ console.log(extSwitchReason || input.substitution);
+ }
+
+ reason = treatmentFailureMap[reason] || 'LA46-8' /* Other */;
+ const reasonCode = substituteReasonMap[reasonLabel] || 'LA6535-4'; /*Other*/
+ //console.log('reasonCode ::', reasonCode);
+
+ const value = util.concept(reasonLabel, ['http://loinc.org', reasonCode]);
+
+ // Add an extension to the new value concept
+ util.addExtension(
+ value,
+ 'http://moh.gov.et/fhir/hiv/StructureDefinition/cd4-vl-classification-for-treatment-failure',
+ util.concept([
+ reason,
+ 'http://moh.gov.et/fhir/hiv/CodeSystem/cd4-vl-classification-for-treatment-failure-code-system',
+ ])
+ );
+
+ changeReasonObs = b.observation('arv-regimen-change-reason-observation', {
+ id: createId(input.id, 'ARVRegimenChangeReason'),
+ status: 'final',
+ subject: input.subject,
+ encounter,
+ effective: encounter?.period?.start,
+ performer: findServiceProvider(encounter, patient),
+ value,
+ //hasMember: changeObs, //reference moved to Obs arv-regimen-changed-observation
+ });
+ markNew(changeReasonObs);
+ state.observations.push(changeReasonObs);
+ }
+
+ if (changeObs) {
+ changeObs.hasMember = [changeTypeObs, changeReasonObs]
+ .filter(Boolean)
+ .map(util.ref);
+ }
+
+ return result;
+ };
+
+ each($.inputs.MedicationDispense, state => {
+ const input = state.data.resource;
+
+ console.log(`Processing medication dispense ${input.id}`);
+
+ let encounter;
+ if (input.context) {
+ // Extract the actual ID out of the reference
+ const [_, encounterId] = input.context.reference.split('Encounter/');
+ encounter = state.inputs.Encounter.find(
+ e => e.resource.id === encounterId
+ )?.resource;
+ if (encounter) {
+ console.log(` Found encounter ${input.context.reference} in bundle`);
+ } else {
+ console.log(
+ 'WARNING: COULD NOT FIND MATCHING ENCOUNTER FOR ',
+ input.context
+ );
+ }
+ }
+
+ const [_, patientId] = input.subject.reference.split('Patient/');
+ const patient = state.inputs.Patient.find(
+ e => e.resource.id === patientId
+ )?.resource;
+
+ const dispense = mapDispense(state, input, encounter);
+
+ state.medicationDispense.push(dispense);
+
+ const med = extractMedication(state, input);
+ dispense.medicationReference = util.reference(med);
+ state.medication.push(med);
+
+ const plan = extractCareplan(state, input, encounter, patient);
+ state.careplans.push(plan);
+
+ const o = extractObservations(state, input, encounter, patient);
+
+ const request = extractRequest(
+ state,
+ input,
+ plan,
+ med,
+ dispense,
+ encounter,
+ o.followUpStatus
+ );
+ dispense.authorizingPrescription = [util.ref(request)];
+ state.requests.push(request);
+
+ // care plan and request have circular dependencies, so sort out plan.activity here
+ plan.activity = [{ reference: util.reference(request) }];
+ extractCareplanExtensions(state, input, plan, encounter);
+
+ const admin = extractAdmin(state, input, encounter, med, request);
+ if (admin) {
+ state.admin.push(admin);
+ }
+
+ return state;
+ });
+
+ fn(state => {
+ console.log(
+ `Finished parsing medications! Created ${state.medicationDispense.length} Patients`
+ );
+
+ return state;
+ });
+
+ send-to-NDR:
+ name: send to NDR
+ adaptor: "@openfn/language-http@6.4.3"
+ credential: root@openhim.org-openhim-ndr
+ body: |
+ fn(state => {
+ console.log('Sending final bundle to NDR');
+ return state;
+ });
+
+ post('/fhir', {
+ body: $.bundle,
+ headers: { 'content-type': 'application/json' },
+ parseAs: 'json',
+ }).catch(async (response, state) => {
+ const err = JSON.parse(response.body)
+ console.error('Error from NDR FHIR')
+ console.error(err)
+ state.issue = err.issue;
+ return state
+ })
+
+ fn(state => {
+ console.log('Done!');
+ return {
+ issue: state.issue,
+ bundle: state.bundle,
+ data: state.data,
+ response: state.response,
+ };
+ });
+
+ send-errors-to-kafka-topic:
+ name: send errors to kafka topic
+ adaptor: "@openfn/language-http@6.4.3"
+ credential: root@openhim.org-openhim-ndr
+ body: |
+ fn(state => {
+ console.log('Catching errors to send to kafka topic fhir-errors...');
+ console.log(state.issue)
+ return state;
+ });
+
+ //send error message from FHIR API
+ post('/openfn/fhir-errors', {
+ body: JSON.stringify($.issue),
+ headers: { 'content-type': 'application/json' },
+ });
+
+ fn(state => {
+ console.log('Sending failed bundle to kafka topic dead-queue...');
+ return state;
+ });
+
+ //send final Bundle we mapped
+ post('/openfn/dead-queue', {
+ body: $.bundle,
+ headers: { 'content-type': 'application/json' },
+ });
+
+ build-bundle:
+ name: build bundle
+ adaptor: "@openfn/language-common@2.0.1"
+ credential: null
+ body: |
+ const wrapResource = (res) => {
+ const { __new__, ...resource } = res;
+ return {
+ request: {
+ method: __new__ ? "POST" : "PUT",
+ url: __new__ ? res.resourceType : `${res.resourceType}/${res.id}`
+ },
+ resource,
+ }
+ }
+
+ fn((state) => {
+ console.log('Building final bundle...')
+
+ delete state.inputs
+
+ const bundle = {
+ "resourceType": "Bundle",
+ "type": "transaction",
+ "entry": []
+ };
+
+ // Write all data onto state
+ for (const k in state) {
+ const arr = state[k]
+ if (Array.isArray(arr)) {
+ bundle.entry.push(...arr.filter(a => a.resourceType).map(wrapResource))
+ }
+ }
+
+ console.log(`Bundle built with ${bundle.entry.length} items`)
+
+ const counts = {}
+ for (const res of bundle.entry ) {
+ try {
+ const type = res.resource.meta.profile[0].split('StructureDefinition/')[1]
+ counts[type] ??= 0
+ counts[type] += 1
+ } catch(e) {
+ console.log(res)
+ throw e
+ }
+ }
+
+ console.log(counts)
+
+ return { bundle };
+
+ })
+
+ triggers:
+ kafka:
+ type: webhook
+ enabled: true
+ edges:
+ kafka->parse-bundle:
+ source_trigger: kafka
+ target_job: parse-bundle
+ condition_type: always
+ enabled: true
+ parse-bundle->Register-Organization:
+ source_job: parse-bundle
+ target_job: Register-Organization
+ condition_type: on_job_success
+ enabled: true
+ Register-Organization->map-encounters:
+ source_job: Register-Organization
+ target_job: map-encounters
+ condition_type: on_job_success
+ enabled: true
+ map-encounters->map-patients:
+ source_job: map-encounters
+ target_job: map-patients
+ condition_type: on_job_success
+ enabled: true
+ map-patients->map-medications:
+ source_job: map-patients
+ target_job: map-medications
+ condition_type: on_job_success
+ enabled: true
+ send-to-NDR->send-errors-to-kafka-topic:
+ source_job: send-to-NDR
+ target_job: send-errors-to-kafka-topic
+ condition_type: js_expression
+ condition_label: import errors
+ condition_expression: state.issue
+ enabled: true
+ map-medications->build-bundle:
+ source_job: map-medications
+ target_job: build-bundle
+ condition_type: on_job_success
+ enabled: true
+ build-bundle->send-to-NDR:
+ source_job: build-bundle
+ target_job: send-to-NDR
+ condition_type: on_job_success
+ enabled: true
diff --git a/openfn/importer/workflows/example/state.json b/openfn/importer/workflows/example/state.json
new file mode 100644
index 00000000..1ff6f852
--- /dev/null
+++ b/openfn/importer/workflows/example/state.json
@@ -0,0 +1,155 @@
+{
+ "id": "891a9242-66d8-4089-813d-50676d33905a",
+ "name": "jembi-cdc-ahie",
+ "description": "POC test project for ART patient use case",
+ "inserted_at": "2024-09-26T10:04:11Z",
+ "updated_at": "2024-09-26T10:04:11Z",
+ "scheduled_deletion": null,
+ "project_credentials": {
+ "root@openhim.org-openhim-ndr": {
+ "id": "9ef53b55-f50b-4b3e-a610-5dd71ecf441d",
+ "name": "openhim ndr",
+ "owner": "root@openhim.org"
+ }
+ },
+ "history_retention_period": null,
+ "dataclip_retention_period": null,
+ "retention_policy": "retain_all",
+ "workflows": {
+ "1-process-cdr-bundle": {
+ "id": "5e077ab9-3956-4a40-9369-37442cdc072d",
+ "name": "1-process-cdr-bundle",
+ "edges": {
+ "kafka->parse-bundle": {
+ "enabled": true,
+ "id": "d7a60efd-b388-4f99-9d98-68e39fd54954",
+ "source_trigger_id": "11013279-3676-45d8-8e33-b492257d69e2",
+ "condition_type": "always",
+ "target_job_id": "00a86a87-46d4-4d83-83bb-3c0829b869d2"
+ },
+ "parse-bundle->Register-Organization": {
+ "enabled": true,
+ "id": "2486e44c-b126-45e2-843b-2e155feb5c70",
+ "source_job_id": "00a86a87-46d4-4d83-83bb-3c0829b869d2",
+ "condition_type": "on_job_success",
+ "target_job_id": "d9f273dc-5233-403a-afaa-c923e4c1d647"
+ },
+ "Register-Organization->map-encounters": {
+ "enabled": true,
+ "id": "0e79825e-a357-4c2c-8d0d-13cfc720a9a8",
+ "source_job_id": "d9f273dc-5233-403a-afaa-c923e4c1d647",
+ "condition_type": "on_job_success",
+ "target_job_id": "f56eb274-64f6-4f2f-bfcf-58f1face3778"
+ },
+ "map-encounters->map-patients": {
+ "enabled": true,
+ "id": "2f6780a8-983f-43f7-85ca-382e83a2e557",
+ "source_job_id": "f56eb274-64f6-4f2f-bfcf-58f1face3778",
+ "condition_type": "on_job_success",
+ "target_job_id": "95ff2b28-b077-4331-9970-7aaa7ca2ce33"
+ },
+ "map-patients->map-medications": {
+ "enabled": true,
+ "id": "4656e3d9-92c6-4986-ab17-f9258573539a",
+ "source_job_id": "95ff2b28-b077-4331-9970-7aaa7ca2ce33",
+ "condition_type": "on_job_success",
+ "target_job_id": "a26e338f-4ec8-4f62-9093-61d7f5c24497"
+ },
+ "send-to-NDR->send-errors-to-kafka-topic": {
+ "enabled": true,
+ "id": "ffea6d08-27f5-458c-8f66-375a0b84367c",
+ "source_job_id": "1d30372e-18fe-4950-a40d-aa08dbf80c7f",
+ "condition_expression": "state.issue",
+ "condition_type": "js_expression",
+ "condition_label": "import errors",
+ "target_job_id": "80d8b3f2-089d-42ab-908c-d3a3b6f85a64"
+ },
+ "map-medications->build-bundle": {
+ "enabled": true,
+ "id": "b92675c2-61af-49b1-aa8e-14135cd4b824",
+ "source_job_id": "a26e338f-4ec8-4f62-9093-61d7f5c24497",
+ "condition_type": "on_job_success",
+ "target_job_id": "db6a52fc-caac-4607-81df-caa5a28ffd1c"
+ },
+ "build-bundle->send-to-NDR": {
+ "enabled": true,
+ "id": "58c74730-1eb7-4dc6-8e7d-5e31056f9142",
+ "source_job_id": "db6a52fc-caac-4607-81df-caa5a28ffd1c",
+ "condition_type": "on_job_success",
+ "target_job_id": "1d30372e-18fe-4950-a40d-aa08dbf80c7f"
+ }
+ },
+ "concurrency": null,
+ "inserted_at": "2024-09-26T10:04:11Z",
+ "updated_at": "2024-09-26T10:04:11Z",
+ "jobs": {
+ "parse-bundle": {
+ "id": "00a86a87-46d4-4d83-83bb-3c0829b869d2",
+ "name": "parse bundle",
+ "body": "// this will take an incoming bundle and sort it into different resource types\nfn((state) => {\n let data = state.data.body;\n if (typeof data === \"string\") {\n data = JSON.parse(data)\n }\n\n const next = {\n entry: data.entry\n }\n\n console.log(`Loading ${next.entry.length} bundle items from input`)\n\n return next;\n})\n\ngroup($.entry, 'resource.resourceType')\n\nfn((state) => {\n const counts = Object.keys(state.data).map((k) => `${k} (${state.data[k].length})`);\n console.log(`Sorted bundles into the following groups: ${counts.join('\\n ')}`)\n return {\n inputs: state.data,\n }\n})\n",
+ "adaptor": "@openfn/language-common@2.0.1",
+ "project_credential_id": null
+ },
+ "Register-Organization": {
+ "id": "d9f273dc-5233-403a-afaa-c923e4c1d647",
+ "name": "Register Organization",
+ "body": "//For the POC test only we need to register the Organization first\n//... for each Bundle... and AFTER we can process the Bundle\nfn(state => {\n const orgId =\n state.inputs.Patient[0].resource.managingOrganization.reference.replace(\n 'Organization/',\n ''\n );\n console.log('Setting up test organization', orgId);\n\n const mappedOrganization = {\n resourceType: 'Organization',\n id: orgId,\n meta: {\n profile: ['http://moh.gov.et/fhir/hiv/StructureDefinition/organization'],\n },\n text: {\n status: 'generated',\n div: ' identifier: http://moh.gov.et/fhir/hiv/identifier/mohid
/MOH001, http://moh.gov.et/fhir/hiv/identifier/hfuid
/HFUID001
active: true
type: Current Healthcare Provider
name: Meshwalkiya Health Center
address: Meshulekia Kirkos Addis Ababa
',\n },\n identifier: [\n {\n system: 'http://moh.gov.et/fhir/hiv/identifier/mohid',\n value: 'MOH001',\n },\n {\n system: 'http://moh.gov.et/fhir/hiv/identifier/hfuid',\n value: 'HFUID001',\n },\n ],\n active: true,\n type: [\n {\n coding: [\n {\n system: 'http://snomed.info/sct',\n code: '257622000',\n },\n ],\n text: 'Current Healthcare Provider',\n },\n ],\n name: `Organization ${orgId}`,\n address: [\n {\n line: ['Meshulekia'],\n city: 'Kirkos',\n district: 'Kirkos woreda 9',\n state: 'Addis Ababa',\n },\n ],\n };\n\n return { ...state, mappedOrganization, orgId };\n});\n\n// basic HTTP put should do it\nput(`/fhir/Organization/${$.orgId}`, {\n body: $.mappedOrganization,\n headers: { 'content-type': 'application/json' },\n});\n\nfn(state => {\n console.log('Organization registered!');\n return state;\n});\n",
+ "adaptor": "@openfn/language-http@6.4.3",
+ "project_credential_id": "9ef53b55-f50b-4b3e-a610-5dd71ecf441d"
+ },
+ "map-encounters": {
+ "id": "f56eb274-64f6-4f2f-bfcf-58f1face3778",
+ "name": "map encounters",
+ "body": "fn(state => {\n console.log('Parsing encounters...');\n state.encounters = [];\n return state;\n});\n\neach($.inputs.Encounter, state => {\n const input = state.data.resource;\n console.log('Mapping encounter ', input.id);\n\n util.setSystemMap({\n 'http://cdr.aacahb.gov.et/Encounter':\n 'http://moh.gov.et/fhir/hiv/identifier/encounter',\n });\n\n const visitType = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/visit-type'\n );\n\n const [_, patientId] = input.subject.reference.split('Patient/');\n const patient = state.inputs.Patient.find(\n e => e.resource.id === patientId\n )?.resource;\n if (!patient) {\n console.log(\n `WARNING: COULD NOT FIND PATIENT ${patientId ?? input.subject} IN BUNDLE`\n );\n }\n\n // TODO: Future phase work out whether this is a target-facility-encounter or a entry-from-outside-target-facility-encounter\n if (visitType) {\n const serviceType = input.serviceType?.coding.find(\n item =>\n item.system === 'http://terminology.hl7.org/CodeSystem/service-type'\n );\n\n const result = builders.encounter('target-facility-encounter', {\n id: input.id,\n status: input.status,\n\n class:\n input.class ??\n util.coding(\n 'Absent-or-unknown',\n 'http://moh.gov.et/fhir/hiv/CodeSystem/absent-unknown-code-system'\n ),\n identifier: input.identifier,\n serviceType: serviceType && {\n coding: serviceType,\n },\n period: input.period,\n subject: input.subject,\n serviceProvider: patient.managingOrganization,\n type: input.type,\n });\n\n // Handle the visit type extension manually\n util.addExtension(\n result.type[0],\n 'http://moh.gov.et/fhir/hiv/StructureDefinition/encounter-visit-type',\n util.concept([\n visitType.valueString.toLowerCase(),\n 'http://moh.gov.et/fhir/hiv/CodeSystem/encounter-visit-type-code-system',\n ])\n ),\n state.encounters.push(result);\n } else {\n console.log('WARNING: no visit type found for Encounter ', input.id);\n }\n\n return state;\n});\n\nfn(state => {\n console.log(\n `Finished parsing Encounters! Created ${state.encounters.length} Encounters`\n );\n\n return state;\n});\n",
+ "adaptor": "@openfn/language-fhir-jembi@0.0.18",
+ "project_credential_id": null
+ },
+ "map-patients": {
+ "id": "95ff2b28-b077-4331-9970-7aaa7ca2ce33",
+ "name": "map patients",
+ "body": "// Handle patients (and related persons)\n\n// This import is needed for code assist in VSC\n// It is harmless to include in job code, but not needed in lightning\nimport { b, util } from '@openfn/language-fhir-jembi';\n\nconst createId = (base, type) => {\n let id = `${base}${type}`;\n if (id.length <= 64) {\n // If the id is too long, cut the difference out of the base id\n id = `${base.substr(0, base.length - (id.length - 64))}${type}`;\n }\n return id;\n};\n\n// Flag a resource as being newly created\n// This affects the final bundle\n// the __new__ property will be removed at the end\nconst markNew = resource => {\n resource.__new__ = true;\n return resource;\n};\n\n// Util function to map residential type on the address\nconst mapAddress = a => {\n if (/rural/i.test(a.text)) {\n const { text, ...address } = a;\n return {\n ...address,\n residentialType: util.concept('Rural', [\n '224804009',\n 'http://snomed.info/sct',\n ]),\n };\n } else if (/urban/i.test(a.text)) {\n const { text, ...address } = a;\n return {\n ...address,\n residentialType: util.concept('Urban', [\n '224807002',\n 'http://snomed.info/sct',\n ]),\n };\n }\n return a;\n};\n\nconst populationMap = {\n 'general population': {\n code: 'General-Population',\n display: 'General Population',\n },\n prisoners: {\n code: 'Prisoner',\n display: 'Prisoner',\n },\n 'children of plhiv': {\n code: 'Children-Of-PLHIV',\n display: 'Children of PLHIV',\n },\n 'partners of plhiv': {\n code: 'Partners-Of-PLHIV',\n display: 'Partners of PLHIV',\n },\n 'discordant couples': {\n code: 'Discordant-Couples',\n display: 'Discordant Couples',\n },\n 'other marps': {\n code: 'Other-MARPS',\n display: 'Other MARPS',\n },\n 'female commercial sex workers': {\n code: 'Female-Commercial-Sex-Workers',\n display: 'Female Commercial Sex Workers',\n },\n 'sex worker': {\n code: 'Female-Commercial-Sex-Workers',\n display: 'Female Commercial Sex Workers',\n },\n 'distance drivers': {\n code: 'Distance-Drivers',\n display: 'Distance Drivers',\n },\n 'mobile or daily laborers': {\n code: 'Mobile-Daily-Laborers',\n display: 'Mobile or Daily Laborers',\n },\n 'ovc or aids orphaned': {\n code: 'OVC-AIDS-Orphaned',\n display: 'OVC or AIDS Orphaned',\n },\n tg: {\n code: 'TG',\n display: 'TG',\n },\n reguees: {\n code: 'Refugee',\n display: 'Refugee',\n },\n 'people who inject drug': {\n code: 'People-Who-Inject-Drug',\n display: 'People who Inject drug',\n },\n msm: {\n code: 'MSM',\n display: 'MSM',\n },\n};\n\nconst educationMap = {\n 'no education': {\n code: 'LA35-1',\n display: 'No schooling',\n },\n primary: {\n code: 'LA32466-7',\n display: 'Primary school education',\n },\n secondary: {\n code: 'LA32467-5',\n display: 'High school education',\n },\n tertiary: {\n code: 'LA15568-1',\n display: 'Graduated from a college or university',\n },\n tvet: {\n code: 'LA39-3',\n display: 'Technical or trade school',\n },\n 'other, specify': {\n code: 'LA46-8',\n display: 'Other',\n },\n assoc: {\n code: 'LA39-3',\n display: 'Technical or trade school',\n },\n bd: {\n code: 'LA15568-1',\n display: 'Graduated from a college or university',\n },\n elem: {\n code: 'LA32466-7',\n display: 'Primary school education',\n },\n gd: {\n code: 'LA15568-1',\n display: 'Graduated from a college or university',\n },\n hs: {\n code: 'LA32467-5',\n display: 'High school education',\n },\n pb: {\n code: 'LA15568-1',\n display: 'Graduated from a college or university',\n },\n postg: {\n code: 'LA15568-1',\n display: 'Graduated from a college or university',\n },\n scol: {\n code: 'LA32467-5',\n display: 'High school education',\n },\n sec: {\n code: 'LA32467-5',\n display: 'High school education',\n },\n};\n\nfn(state => {\n console.log('Parsing patients...');\n state.patients ??= [];\n state.observations ??= [];\n state.relations ??= [];\n\n // This will map identifier systems for us automatically\n util.setSystemMap({\n 'http://cdr.aacahb.gov.et/SmartCareID':\n 'http://moh.gov.et/fhir/hiv/identifier/SmartCareID',\n 'http://cdr.aacahb.gov.et/MRN': 'http://moh.gov.et/fhir/hiv/identifier/MRN',\n 'http://cdr.aacahb.gov.et/UAN': 'http://moh.gov.et/fhir/hiv/identifier/UAN',\n 'http://terminology.hl7.org/CodeSystem/v2-0131':\n 'http://terminology.hl7.org/CodeSystem/v2-0131',\n });\n\n return state;\n});\n\n// Create an NDR patient for each CDR patient\neach($.inputs.Patient, state => {\n const input = state.data.resource;\n\n console.log(` Processing patient ${input.id}`);\n\n const religion = util.findExtension(\n input,\n 'http://hl7.org/fhir/StructureDefinition/patient-religion'\n )?.valueCodeableConcept.coding[0];\n\n const result = b.patient('patient', {\n id: input.id,\n religion:\n religion &&\n util.concept(\n religion.display,\n util.coding(\n religion.code,\n 'http://terminology.hl7.org/CodeSystem/v3-ReligiousAffiliation'\n )\n ),\n identifier: input.identifier,\n name: input.name,\n telecom: input.telecom,\n gender: input.gender,\n birthDate: input.birthDate,\n maritalStatus: input.maritalStatus,\n managingOrganization: input.managingOrganization,\n address: input.address?.map(mapAddress),\n });\n\n state.patients.push(result);\n\n return state;\n});\n\n// Map the observations for each patient\neach($.inputs.Patient, state => {\n const input = state.data.resource;\n console.log(` Processing observations for patient ${input.id}`);\n // Assumes one encounter per bundle\n const encounter = state.inputs.Encounter[0].resource;\n\n const occupation = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/Occupation'\n );\n if (occupation) {\n state.observations.push(\n markNew(\n b.observation('patient-occupation-observation', {\n id: createId(input.id, 'PatientOccupation'),\n // what to do about id?\n status: input.status ?? 'final',\n\n effective: encounter.period?.start,\n encounter,\n performer: input.managingOrganization ?? state.orgId,\n\n // Map the occupation string without a coding\n value: util.concept(occupation.valueString, [\n occupation.valueString,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/occupation-code-system',\n ]),\n\n subject: input,\n })\n )\n );\n }\n\n const education = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/EducationalLevel'\n );\n if (education) {\n const ed =\n education.valueString ?? education.valueCodeableConcept?.coding[0].code;\n const edLabel =\n education.valueString ??\n education.valueCodeableConcept?.coding[0].display;\n const mapped = educationMap[ed.toLowerCase()];\n if (!mapped) {\n console.error('ERROR: could not find mapped education for ', ed);\n }\n const value = util.concept(edLabel, [mapped.code, 'http://loinc.org']);\n\n state.observations.push(\n markNew(\n b.observation('highest-education-observation', {\n id: createId(input.id, 'HighestEducation'),\n status: input.status ?? 'final',\n\n effective: encounter.period?.start,\n encounter,\n performer: input.managingOrganization ?? state.orgId,\n subject: input,\n value,\n })\n )\n );\n }\n\n const popGroup = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/TargetPopulationGroup'\n );\n if (popGroup) {\n const group =\n popGroup.valueString ?? popGroup.valueCodeableConcept?.coding[0].text;\n const mapped = populationMap[group.toLowerCase()];\n if (!mapped) {\n console.error(\n 'ERROR: could not find mapped population group for ',\n group\n );\n }\n const value = util.concept(mapped.display, [\n mapped.code,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/target-population-category-code-system',\n ]);\n\n state.observations.push(\n markNew(\n b.observation('target-population-observation', {\n id: createId(input.id, 'TargetPopulation'),\n status: input.status ?? 'final',\n effective: encounter.period?.start,\n encounter,\n performer: input.managingOrganization ?? state.orgId,\n subject: input,\n value,\n })\n )\n );\n }\n\n return state;\n});\n\n// Find related people\neach($.inputs.RelatedPerson, state => {\n const input = state.data.resource;\n\n console.log(` Processing related person ${input.id}`);\n\n let patient;\n if (input.patient) {\n // Extract the actual ID out of the reference\n const [_, patientId] = input.patient.reference.split('Patient/');\n patient = state.inputs.Patient.find(\n e => e.resource.id === patientId\n )?.resource;\n if (patient) {\n console.log(` Found patient ${input.patient.reference} in bundle`);\n } else {\n console.log(\n 'WARNING: COULD NOT FIND MATCHING PATIENT FOR ',\n input.patient.reference\n );\n return state;\n }\n }\n\n const r = b.relatedPerson('related-person', {\n ...input, // this should cover the basics\n id: input.id,\n extension: input.extension,\n\n address: input.address?.map(mapAddress),\n\n relationship: input.relationship.reduce(\n (obj, next) => Object.assign(obj, next),\n {}\n ),\n });\n\n state.relations.push(r);\n\n return state;\n});\n\nfn(state => {\n console.log(\n `Finished parsing patients! Created ${state.patients.length} Patients, ${state.observations.length} Observations and ${state.relations.length} Related Persons`\n );\n\n return state;\n});\n",
+ "adaptor": "@openfn/language-fhir-jembi@0.0.18",
+ "project_credential_id": null
+ },
+ "map-medications": {
+ "id": "a26e338f-4ec8-4f62-9093-61d7f5c24497",
+ "name": "map medications",
+ "body": "const createId = (base, type) => {\n let id = `${base}${type}`;\n if (id.length > 64) {\n // If the id is too long, cut the difference out of the base id\n id = `${base.substr(0, base.length - (id.length - 64))}${type}`;\n }\n return id;\n};\n\n// Flag a resource as being newly created\n// This affects the final bundle\n// the __new__ property will be removed at the end\nconst markNew = resource => {\n resource.__new__ = true;\n};\n\nconst findServiceProvider = (encounter, patient) =>\n encounter?.serviceProvider ?? patient?.managingOrganization;\n\nconst categoryTypeMap = {\n '1st switch': 'First-Switch',\n '2nd switch': 'Second-Switch',\n '3rd switch': 'Third-Switch',\n '1st substitution': 'First-Substitute',\n '2nd substitution': 'Second-Substitute',\n '3rd substitution': 'Third-Substitute',\n '4th substitution': 'Fourth-Substitute',\n '5th substitution': 'Fifth-Substitute',\n '6th substitution': 'Sixth-Substitute',\n};\n\nconst treatmentFailureMap = {\n 'Virologic failure': 'Virologic-Failure',\n 'Clinical failure': 'Clinical-Failure',\n 'Immunologic failure': 'Immunologic-Failure',\n};\n\nconst substituteReasonMap = {\n 'Risk of pregnancy': 'LA6531-3',\n Pregnancy: 'LA6530-5',\n 'New drug available': 'LA6533-9',\n 'Due to new TB': 'LA6532-1',\n 'Drug stock-out': 'LA6534-7',\n 'Toxicity/Side effects': 'LA6529-7',\n Other: 'LA6535-4',\n Anemia: 'LA6535-4',\n};\n\nconst stoppedReasonMap = {\n 'toxicity/side effects': {\n code: 'LA6529-7',\n text: 'Toxicity/Side Effects',\n },\n pregnancy: {\n code: 'A6530-5',\n text: 'Pregnancy',\n },\n 'planned treatment interruption': {\n code: '18733-6',\n text: 'Attending Progress note',\n },\n 'other patient decisions': {\n code: '18733-6',\n text: 'Attending Progress note',\n },\n 'poor adherence': {\n code: 'LA31432-0',\n text: 'Not adherent to treatment',\n },\n 'planned treatment interruption': {\n code: 'LP56812-8',\n text: 'Medication regimen.status',\n },\n 'patient lack finances': {\n code: 'LA28316-0',\n text: 'Financial concerns',\n },\n 'other patient decisions': {\n code: '51855-5',\n text: 'Patient Note',\n },\n 'illness/hospitalization': {\n code: '34130-5',\n text: 'Hospital Progress note',\n },\n 'drugs out of stock': {\n code: 'LA6534-7',\n text: 'Drug Out of Stock',\n },\n other: {\n code: 'LA46-8',\n text: 'Other',\n },\n};\n\nfn(state => {\n console.log('Parsing medications...');\n state.medicationDispense = [];\n state.medication = [];\n state.observations ??= [];\n state.careplans ??= [];\n state.requests ??= [];\n state.admin ??= [];\n\n return state;\n});\n\nconst extractMedication = (state, input) => {\n const { coding, text } = input.medicationCodeableConcept;\n\n const codes = coding.find(\n c => c.system === 'http://cdr.aacahb.gov.et/hiv-regimen-codes'\n );\n const c = [\n codes.code,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-treatment-code-system',\n ];\n\n let value;\n if (text) {\n value = util.concept(text, c);\n } else {\n value = util.concept(codes.code, c);\n }\n\n const m = b.medication('arv-regimen-medication', {\n id: `${input.id}Medication`,\n status: 'active',\n code: value,\n });\n markNew(m);\n return m;\n};\n\nconst extractAdmin = (state, input, encounter, medication, request) => {\n const end = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/dose-end-date'\n );\n if (end) {\n const ma = b.medicationAdministration('arv-medication-administration', {\n id: createId(input.id, 'ARVMedicationAdministration'),\n status: 'completed',\n medication: util.ref(medication),\n subject: input.subject,\n context: input.context,\n request,\n effective: end.valueDateTime,\n //effective: encounter.period,\n });\n markNew(ma);\n return ma;\n }\n};\n\nconst extractRequest = (\n state,\n input,\n careplan,\n medication,\n dispense,\n encounter,\n observation\n) => {\n const r = b.medicationRequest('arv-medication-request', {\n id: createId(input.id, 'ARVMedicationRequest'),\n status: 'completed',\n intent: 'order',\n medication,\n basedOn: careplan,\n dispenseRequest: {\n quantity: {\n value: parseInt(input.quantity.value),\n unit: 'TAB',\n system: 'http://terminology.hl7.org/CodeSystem/v3-orderableDrugForm',\n code: 'TAB',\n },\n },\n subject: input.subject,\n encounter: input.context,\n authoredOn: encounter.period?.start,\n identifier: {\n value: dispense.id,\n system: 'http://moh.gov.et/fhir/hiv/identifier/medication',\n },\n reasonReference: util.reference(observation),\n });\n\n markNew(r);\n\n return r;\n};\n\nconst extractCareplan = (state, input, encounter, patient) => {\n const cp = b.carePlan('art-follow-up-careplan', {\n id: createId(input.id, 'ARTFollowUpCareplan'),\n status: 'active',\n intent: 'order',\n subject: input.subject,\n encounter: input.context,\n created: encounter.period?.start,\n performer: findServiceProvider(encounter, patient),\n category: [\n {\n coding: [\n {\n system:\n 'http://moh.gov.et/fhir/hiv/CodeSystem/care-plan-category-code-system',\n code: 'art-follow-up-care-plan',\n },\n ],\n text: 'ART',\n },\n ],\n });\n markNew(cp);\n return cp;\n};\nconst extractCareplanExtensions = (state, input, plan, encounter) => {\n const nextVisit = util.findExtension(\n encounter,\n 'http://cdr.aacahb.gov.et/next-visit'\n );\n if (nextVisit) {\n util.addExtension(\n plan.activity[0],\n 'http://moh.gov.et/fhir/hiv/StructureDefinition/care-plan-next-visit',\n nextVisit.valueDateTime\n );\n }\n};\n\n// This maps the actual dispense object\nconst mapDispense = (_state, input /* Dispense resource*/, encounter) => {\n // Create a matching dispense\n const handover = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/dose-end-date'\n );\n\n return b.medicationDispense('arv-medication-dispense', {\n id: input.id,\n status: input.status,\n subject: input.subject,\n context: input.context,\n quantity: input.quantity && {\n ...input.quantity,\n value: parseInt(input.quantity.value),\n unit: 'TAB',\n code: 'TAB',\n },\n daysSupply: input.daysSupply && {\n ...input.daysSupply,\n value: parseInt(input.daysSupply.value),\n unit: 'Days',\n code: 'd',\n },\n whenHandedOver: handover?.valueDateTime,\n medication: input.reference,\n // missing properties will be added later in the script\n });\n};\n\n// This pulls out a bunch of observations\nconst extractObservations = (state, input, encounter, patient) => {\n const result = {};\n let stopReasonObs;\n\n const stopReason = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/medication-stopped-reason'\n );\n if (stopReason) {\n console.log(\n `creating art-followup-stopped-reasons-observation for ${input.id}`\n );\n\n //console.log('stopReason::', JSON.stringify(stopReason, null, 2));\n\n const stopReasonObs = b.observation(\n 'art-followup-stopped-reasons-observation',\n {\n id: createId(input.id, 'ARTFollowupStatusStoppedReasonObservation'),\n status: 'final',\n subject: input.subject,\n encounter,\n effective: encounter?.period?.start,\n performer: findServiceProvider(encounter, patient),\n }\n );\n\n let text = stopReason.valueString ?? stopReason.valueCodeableConcept?.text;\n\n //('stopReason text::', JSON.stringify(text, null, 2));\n if (!text) {\n console.error(`No stop reason text found for `, input.id);\n }\n\n const value = {};\n const mapped = stoppedReasonMap[text.toLowerCase()];\n //console.log('mapped stopReasonObs::', JSON.stringify(mapped, null, 2));\n if (mapped) {\n util.addExtension(\n value,\n 'http://moh.gov.et/fhir/hiv/StructureDefinition/reason-art-stopped',\n util.concept(mapped.text, [mapped.code, 'http://loinc.org'])\n );\n\n //console.log('stopReasonObs::', JSON.stringify(stopReasonObs, null, 2));\n\n stopReasonObs['valueCodeableConcept'] = value;\n\n // console.log(\n // 'NEW stopReasonObs::',\n // JSON.stringify(stopReasonObs, null, 2)\n // );\n\n markNew(stopReasonObs);\n\n // We have to write the concept like this because it's a strange structure\n // the adaptor should handle this a lot better really\n state.observations.push(stopReasonObs);\n console.log('Stopped Obs pushed...');\n } else {\n console.error('WARNING: NO MAPPED VALUE FOR STOP-REASON: ', text);\n }\n }\n\n if (input.statusReasonCodeableConcept) {\n const reason = Array.isArray(input.statusReasonCodeableConcept)\n ? input.statusReasonCodeableConcept[0]\n : input.statusReasonCodeableConcept;\n const o = b.observation('art-followup-status-observation', {\n id: createId(input.id, 'ARTFollowupStatusObservation'),\n status: 'final',\n subject: input.subject,\n encounter,\n effective: encounter?.period?.start,\n value: util.concept([\n reason.coding[0].code,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/art-follow-up-status-code-system',\n ]),\n hasMember: stopReasonObs ? util.ref(stopReasonObs) : undefined,\n performer: findServiceProvider(encounter, patient),\n });\n result.followUpStatus = o;\n state.observations.push(o);\n }\n\n const extSwitchType = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/switch-type'\n );\n const extSwitchReason = util.findExtension(\n input,\n 'http://cdr.aacahb.gov.et/switch-reason'\n );\n\n let changeObs;\n let changeTypeObs;\n let changeReasonObs;\n\n if (input.substitution || extSwitchType) {\n changeObs = b.observation('arv-regimen-changed-observation', {\n id: createId(input.id, 'ARVRegimenChange'),\n status: 'final',\n subject: input.subject,\n encounter,\n effective: encounter?.period,\n performer: findServiceProvider(encounter, patient),\n\n value: util.concept([\n input.substitution ? 'Substituted' : 'Switched',\n 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-regimen-change-category-code-system',\n ]),\n // hasMember will be set at the end\n });\n markNew(changeObs);\n state.observations.push(changeObs);\n }\n\n if (input.substitution?.type || extSwitchType) {\n let value;\n if (extSwitchType) {\n const code = categoryTypeMap[extSwitchType.valueString];\n if (!code) {\n console.error('No mapped switch code found for ', input.id);\n console.log(extSwitchType);\n }\n value = util.concept([\n code,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-regimen-switch-type-code-system',\n ]);\n } else {\n const code = categoryTypeMap[input.substitution?.type.text];\n if (!code) {\n console.error('No mapped switch code found for ', input.id);\n console.log(input.substitution);\n }\n value = util.concept([\n code,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/arv-regimen-substitute-type-code-system',\n ]);\n }\n\n changeTypeObs = b.observation('arv-change-category-type-observation', {\n id: createId(input.id, 'ARVRegimenChangeCategoryType'),\n status: 'final',\n subject: input.subject,\n encounter,\n effective: encounter?.period?.start,\n performer: findServiceProvider(encounter, patient),\n value,\n // hasmember is added later\n });\n markNew(changeTypeObs);\n state.observations.push(changeTypeObs);\n }\n\n if (extSwitchReason ?? input.substitution?.reason) {\n let reason;\n if (extSwitchReason ?? input.substitution?.reason) {\n reason =\n extSwitchReason?.valueCodeableConcept?.coding[0].code ??\n extSwitchReason?.valueString;\n } else {\n reason = input.substitution?.reason[0].text;\n }\n\n let reasonLabel;\n if (extSwitchReason) {\n reasonLabel =\n extSwitchReason?.valueCodeableConcept?.coding[0].display ??\n extSwitchReason?.valueString;\n } else {\n reasonLabel = input.substitution?.reason[0].text;\n }\n //console.log('reasonLabel ::', reasonLabel);\n\n if (!reason) {\n console.error('No reason observation found for ', input.id);\n console.log(extSwitchReason || input.substitution);\n }\n\n reason = treatmentFailureMap[reason] || 'LA46-8' /* Other */;\n const reasonCode = substituteReasonMap[reasonLabel] || 'LA6535-4'; /*Other*/\n //console.log('reasonCode ::', reasonCode);\n\n const value = util.concept(reasonLabel, ['http://loinc.org', reasonCode]);\n\n // Add an extension to the new value concept\n util.addExtension(\n value,\n 'http://moh.gov.et/fhir/hiv/StructureDefinition/cd4-vl-classification-for-treatment-failure',\n util.concept([\n reason,\n 'http://moh.gov.et/fhir/hiv/CodeSystem/cd4-vl-classification-for-treatment-failure-code-system',\n ])\n );\n\n changeReasonObs = b.observation('arv-regimen-change-reason-observation', {\n id: createId(input.id, 'ARVRegimenChangeReason'),\n status: 'final',\n subject: input.subject,\n encounter,\n effective: encounter?.period?.start,\n performer: findServiceProvider(encounter, patient),\n value,\n //hasMember: changeObs, //reference moved to Obs arv-regimen-changed-observation\n });\n markNew(changeReasonObs);\n state.observations.push(changeReasonObs);\n }\n\n if (changeObs) {\n changeObs.hasMember = [changeTypeObs, changeReasonObs]\n .filter(Boolean)\n .map(util.ref);\n }\n\n return result;\n};\n\neach($.inputs.MedicationDispense, state => {\n const input = state.data.resource;\n\n console.log(`Processing medication dispense ${input.id}`);\n\n let encounter;\n if (input.context) {\n // Extract the actual ID out of the reference\n const [_, encounterId] = input.context.reference.split('Encounter/');\n encounter = state.inputs.Encounter.find(\n e => e.resource.id === encounterId\n )?.resource;\n if (encounter) {\n console.log(` Found encounter ${input.context.reference} in bundle`);\n } else {\n console.log(\n 'WARNING: COULD NOT FIND MATCHING ENCOUNTER FOR ',\n input.context\n );\n }\n }\n\n const [_, patientId] = input.subject.reference.split('Patient/');\n const patient = state.inputs.Patient.find(\n e => e.resource.id === patientId\n )?.resource;\n\n const dispense = mapDispense(state, input, encounter);\n\n state.medicationDispense.push(dispense);\n\n const med = extractMedication(state, input);\n dispense.medicationReference = util.reference(med);\n state.medication.push(med);\n\n const plan = extractCareplan(state, input, encounter, patient);\n state.careplans.push(plan);\n\n const o = extractObservations(state, input, encounter, patient);\n\n const request = extractRequest(\n state,\n input,\n plan,\n med,\n dispense,\n encounter,\n o.followUpStatus\n );\n dispense.authorizingPrescription = [util.ref(request)];\n state.requests.push(request);\n\n // care plan and request have circular dependencies, so sort out plan.activity here\n plan.activity = [{ reference: util.reference(request) }];\n extractCareplanExtensions(state, input, plan, encounter);\n\n const admin = extractAdmin(state, input, encounter, med, request);\n if (admin) {\n state.admin.push(admin);\n }\n\n return state;\n});\n\nfn(state => {\n console.log(\n `Finished parsing medications! Created ${state.medicationDispense.length} Patients`\n );\n\n return state;\n});\n",
+ "adaptor": "@openfn/language-fhir-jembi@0.0.18",
+ "project_credential_id": null
+ },
+ "send-to-NDR": {
+ "id": "1d30372e-18fe-4950-a40d-aa08dbf80c7f",
+ "name": "send to NDR",
+ "body": "fn(state => {\n console.log('Sending final bundle to NDR');\n return state;\n});\n\npost('/fhir', {\n body: $.bundle,\n headers: { 'content-type': 'application/json' },\n parseAs: 'json',\n}).catch(async (response, state) => {\n const err = JSON.parse(response.body)\n console.error('Error from NDR FHIR')\n console.error(err)\n state.issue = err.issue;\n return state\n})\n\nfn(state => {\n console.log('Done!');\n return {\n issue: state.issue,\n bundle: state.bundle,\n data: state.data,\n response: state.response,\n };\n});\n",
+ "adaptor": "@openfn/language-http@6.4.3",
+ "project_credential_id": "9ef53b55-f50b-4b3e-a610-5dd71ecf441d"
+ },
+ "send-errors-to-kafka-topic": {
+ "id": "80d8b3f2-089d-42ab-908c-d3a3b6f85a64",
+ "name": "send errors to kafka topic",
+ "body": "fn(state => {\n console.log('Catching errors to send to kafka topic fhir-errors...');\n console.log(state.issue)\n return state;\n});\n\n //send error message from FHIR API\npost('/openfn/fhir-errors', {\n body: JSON.stringify($.issue),\n headers: { 'content-type': 'application/json' },\n});\n\nfn(state => {\n console.log('Sending failed bundle to kafka topic dead-queue...');\n return state;\n});\n\n//send final Bundle we mapped\npost('/openfn/dead-queue', {\n body: $.bundle, \n headers: { 'content-type': 'application/json' },\n});\n",
+ "adaptor": "@openfn/language-http@6.4.3",
+ "project_credential_id": "9ef53b55-f50b-4b3e-a610-5dd71ecf441d"
+ },
+ "build-bundle": {
+ "id": "db6a52fc-caac-4607-81df-caa5a28ffd1c",
+ "name": "build bundle",
+ "body": "const wrapResource = (res) => {\n const { __new__, ...resource } = res;\n return {\n request: {\n method: __new__ ? \"POST\" : \"PUT\",\n url: __new__ ? res.resourceType : `${res.resourceType}/${res.id}`\n },\n resource,\n }\n}\n\nfn((state) => {\n console.log('Building final bundle...')\n\n delete state.inputs\n\n const bundle = {\n \"resourceType\": \"Bundle\",\n \"type\": \"transaction\",\n \"entry\": []\n };\n\n // Write all data onto state\n for (const k in state) {\n const arr = state[k]\n if (Array.isArray(arr)) {\n bundle.entry.push(...arr.filter(a => a.resourceType).map(wrapResource))\n }\n }\n\n console.log(`Bundle built with ${bundle.entry.length} items`)\n\n const counts = {}\n for (const res of bundle.entry ) {\n try {\n const type = res.resource.meta.profile[0].split('StructureDefinition/')[1]\n counts[type] ??= 0\n counts[type] += 1\n } catch(e) {\n console.log(res)\n throw e\n }\n }\n\n console.log(counts)\n\n return { bundle };\n\n})\n",
+ "adaptor": "@openfn/language-fhir-jembi@0.0.18",
+ "project_credential_id": null
+ }
+ },
+ "deleted_at": null,
+ "lock_version": 1,
+ "triggers": {
+ "kafka": {
+ "enabled": true,
+ "id": "11013279-3676-45d8-8e33-b492257d69e2",
+ "type": "webhook"
+ }
+ }
+ }
+ },
+ "requires_mfa": false
+}
diff --git a/openfn/package-metadata.json b/openfn/package-metadata.json
new file mode 100644
index 00000000..fda1bd9a
--- /dev/null
+++ b/openfn/package-metadata.json
@@ -0,0 +1,41 @@
+{
+ "id": "openfn",
+ "name": "OpenFn",
+ "description": "Lightning ⚡ (OpenFn v2) is a workflow automation platform that's used to automate critical business processes and integrate information systems.",
+ "type": "infrastructure",
+ "version": "0.0.1",
+ "dependencies": ["database-postgres"],
+ "environmentVariables": {
+ "OPENFN_DATABASE_URL": "postgresql://openfn:instant101@postgres-1:5432/lightning_dev",
+ "OPENFN_DISABLE_DB_SSL": "true",
+ "OPENFN_IS_RESETTABLE_DEMO": "true",
+ "OPENFN_LISTEN_ADDRESS": "0.0.0.0",
+ "OPENFN_LOG_LEVEL": "debug",
+ "OPENFN_ORIGINS": "http://localhost:4000",
+ "OPENFN_PRIMARY_ENCRYPTION_KEY": "KLu/IoZuaf+baDECd8wG4Z6auwNe6VAmwh9N8lWdJ1A=",
+ "OPENFN_SECRET_KEY_BASE": "jGDxZj2O+Qzegm5wcZ940RfWO4D6RyU8thNCr5BUpHNwa7UNV52M1/Sn+7RxiP+f",
+ "OPENFN_WORKER_RUNS_PRIVATE_KEY": "LS0tLS1CRUdJTiBQUklWQVRFIEtFWS0tLS0tCk1JSUV2Z0lCQURBTkJna3Foa2lHOXcwQkFRRUZBQVNDQktnd2dnU2tBZ0VBQW9JQkFRREVtR3drUW5pT0hqVCsKMnkyRHFvRUhyT3dLZFI2RW9RWG9DeDE4MytXZ3hNcGthTFZyOFViYVVVQWNISGgzUFp2Z2UwcEIzTWlCWWR5Kwp1ajM1am5uK2JIdk9OZGRldWxOUUdpczdrVFFHRU1nTSs0Njhldm5RS0h6R29DRUhabDlZV0s0MUd5SEZCZXppCnJiOGx2T1A1NEtSTS90aE5pVGtHaUIvTGFLMldLcTh0VmtoSHBvaFE3OGIyR21vNzNmcWtuSGZNWnc0ZE43d1MKdldOamZIN3QwSmhUdW9mTXludUxSWmdFYUhmTDlnbytzZ0thc0ZUTmVvdEZIQkYxQTJjUDJCakwzaUxad0hmdQozTzEwZzg0aGZlTzJqTWlsZlladHNDdmxDTE1EZWNrdFJGWFl6V0dWc25FcFNiOStjcWJWUXRvdEU4QklON09GClRmaEx2MG9uQWdNQkFBRUNnZ0VBV3dmZyt5RTBSVXBEYThiOVdqdzNKdUN4STE1NzFSbmliRUhKVTZzdzNyS0EKck9HM0w5WTI0cHhBdlVPSm5GMFFzbThrUVQ4RU1MU3B6RDdjdDVON2RZMngvaGY4TThhL0VSWXM4cFlYcXI5Vwpnbnh3NldGZ0R6elFHZ0RIaW0raXNudk5ucFdEbTRGVTRObG02d2g5MzVSZlA2KzVaSjJucEJpZjhFWDJLdE9rCklOSHRVbFcwNFlXeDEwS0pIWWhYNFlydXVjL3MraXBORzBCSDZEdlJaQzQxSWw0N1luaTg1OERaL0FaeVNZN1kKWTlTamNKQ0QvUHBENTlNQjlSanJDQjhweDBjWGlsVXBVZUJSYndGalVwbWZuVmhIa1hiYlM1U0hXWWM4K3pLRQp2ajFqSEpxc2UyR0hxK2lHL1V3NTZvcHNyM2x3dHBRUXpVcEJGblhMMFFLQmdRRDM5bkV3L1NNVGhCallSd1JGCkY2a2xOYmltU2RGOVozQlZleXhrT0dUeU5NSCtYckhsQjFpOXBRRHdtMit3V2RvcWg1ZFRFbEU5K1crZ0FhN0YKbXlWc2xPTW4wdnZ2cXY2Wkp5SDRtNTVKU0lWSzBzRjRQOTRMYkpNSStHUW5VNnRha3Y0V0FSMkpXaURabGxPdAp3R01EQWZqRVIrSEFZeUJDKzNDL25MNHF5d0tCZ1FESzk3NERtV0c4VDMzNHBiUFVEYnpDbG9oTlQ2UldxMXVwCmJSWng4ZGpzZU0vQ09kZnBUcmJuMnk5dVc3Q1pBNFVPQ2s4REcxZ3ZENVVDYlpEUVdMaUp5RzZGdG5OdGgvaU8KT1dJM0UyczZOS0VMMU1NVzh5QWZwNzV4Ung5cnNaQzI2UEtqQ0pWL2lTVjcyNlQ1ZTFzRG5sZUtBb0JFZnlDRgpvbEhhMmhybWxRS0JnUURHT1YyOWd1K1NmMng1SVRTWm8xT1ZxbitGZDhlZno1d3V5YnZ3Rm1Fa2V1YUdXZDh1CnJ4UFM3MkJ6K0Y1dUJUWngvMWtLa0w4Zm94TUlQN0FleW1zOWhUeWVybnkyMk9TVlBJSmN3dExqMUxTeDN3L0kKK0kyaVpsYVl1akVlZXpXbHY1S2R0cUNORjk3Zzh0ck1NTnMySVZKa1h1NXFwUk82V0ZXRzZGL2h4d0tCZ0hnNApHYUpFSFhIT204ekZTU2lYSW5FWGZKQmVWZmJIOUxqNzFrbVRlR3RJZTdhTlVHZnVxY1BYUGRiZUZGSHRsY2ZsCkx6dWwzS3V6VFExdEhGTnIyWkl5MTlQM1o1TSs4R2c5Y1FFeVRWYmlpV2xha2x0cmttRnRtQTI4bE0zVEZPWmkKUUNWMUZpZStjaWRVeC9qRnFma1F0c1VXQ2llSUxSazZOY1d0WGpXcEFvR0JBTGN6Y210VGlUUEFvWnk0MFV1QQpTOXpUd3RsamhmUWJEVTVjb21EcnlKcnFRU0VOdmQ2VW5HdW0zYVNnNk13dDc0NGxidDAyMC9mSGI0WTJkTGhMCmx4YWJ5b1dQUElRRUpLL1NNOGtURFEvYTRyME5tZzhuV3h5bGFLcHQ5WUhmZ2NYMkYzSzUrc0VSUGNFcVZlWFMKdWZkYXdYQVlFampZK3V2UHZ2YzU3RU1aCi0tLS0tRU5EIFBSSVZBVEUgS0VZLS0tLS0K",
+ "OPENFN_WORKER_SECRET": "secret_here",
+ "POSTGRES_USER": "postgres",
+ "POSTGRES_SERVICE": "postgres-1",
+ "POSTGRES_DATABASE": "postgres",
+ "POSTGRES_PASSWORD": "instant101",
+ "POSTGRES_PORT": "5432",
+ "OPENFN_POSTGRESQL_DB": "lightning_dev",
+ "OPENFN_POSTGRESQL_USERNAME": "openfn",
+ "OPENFN_POSTGRESQL_PASSWORD": "instant101",
+ "OPENFN_WORKER_LIGHTNING_PUBLIC_KEY": "LS0tLS1CRUdJTiBQVUJMSUMgS0VZLS0tLS0KTUlJQklqQU5CZ2txaGtpRzl3MEJBUUVGQUFPQ0FROEFNSUlCQ2dLQ0FRRUF4SmhzSkVKNGpoNDAvdHN0ZzZxQgpCNnpzQ25VZWhLRUY2QXNkZk4vbG9NVEtaR2kxYS9GRzJsRkFIQng0ZHoyYjRIdEtRZHpJZ1dIY3ZybzkrWTU1Ci9teDd6alhYWHJwVFVCb3JPNUUwQmhESURQdU92SHI1MENoOHhxQWhCMlpmV0ZpdU5Sc2h4UVhzNHEyL0piemoKK2VDa1RQN1lUWWs1Qm9nZnkyaXRsaXF2TFZaSVI2YUlVTy9HOWhwcU85MzZwSngzekdjT0hUZThFcjFqWTN4Kwo3ZENZVTdxSHpNcDdpMFdZQkdoM3kvWUtQcklDbXJCVXpYcUxSUndSZFFObkQ5Z1l5OTRpMmNCMzd0enRkSVBPCklYM2p0b3pJcFgyR2JiQXI1UWl6QTNuSkxVUlYyTTFobGJKeEtVbS9mbkttMVVMYUxSUEFTRGV6aFUzNFM3OUsKSndJREFRQUIKLS0tLS1FTkQgUFVCTElDIEtFWS0tLS0tCg",
+ "OPENFN_IMAGE": "openfn/lightning:v2.9.5",
+ "OPENFN_WORKER_IMAGE": "openfn/ws-worker:latest",
+ "OPENFN_KAFKA_TRIGGERS_ENABLED": "true",
+ "OPENFN_API_KEY": "apiKey",
+ "OPENFN_ENDPOINT": "http://localhost:4000",
+ "OPENFN_DOCKER_WEB_CPUS": "2",
+ "OPENFN_DOCKER_WEB_MEMORY": "4G",
+ "OPENFN_DOCKER_WORKER_CPUS": "2",
+ "OPENFN_DOCKER_WORKER_MEMORY": "4G",
+ "FHIR_SERVER_BASE_URL": "http://openhim-core:5001",
+ "FHIR_SERVER_USERNAME": "openfn_client",
+ "FHIR_SERVER_PASSWORD": "openfn_client_password"
+ }
+}
diff --git a/openfn/swarm.sh b/openfn/swarm.sh
new file mode 100644
index 00000000..42de3f7b
--- /dev/null
+++ b/openfn/swarm.sh
@@ -0,0 +1,91 @@
+#!/bin/bash
+
+declare ACTION=""
+declare MODE=""
+declare COMPOSE_FILE_PATH=""
+declare UTILS_PATH=""
+declare STACK="openfn"
+
+function init_vars() {
+ ACTION=$1
+ MODE=$2
+
+ COMPOSE_FILE_PATH=$(
+ cd "$(dirname "${BASH_SOURCE[0]}")" || exit
+ pwd -P
+ )
+
+ UTILS_PATH="${COMPOSE_FILE_PATH}/../utils"
+
+ readonly ACTION
+ readonly MODE
+ readonly COMPOSE_FILE_PATH
+ readonly UTILS_PATH
+ readonly STACK
+
+}
+
+# shellcheck disable=SC1091
+function import_sources() {
+ source "${UTILS_PATH}/docker-utils.sh"
+ source "${UTILS_PATH}/log.sh"
+}
+
+function initialize_package() {
+ local package_dev_compose_filename=""
+ if [[ "${MODE}" == "dev" ]]; then
+ log info "Running package in DEV mode"
+ package_dev_compose_filename="docker-compose.dev.yml"
+ else
+ log info "Running package in PROD mode"
+ fi
+
+ (
+ log info "Configuring postgres database"
+ docker::await_service_status "postgres" "postgres-1" "Running"
+
+ if [[ "${ACTION}" == "init" ]]; then
+ docker::deploy_config_importer $STACK "$COMPOSE_FILE_PATH/importer/postgres/docker-compose.config.yml" "openfn_db_config" "openfn"
+ fi
+
+ docker::deploy_service $STACK "${COMPOSE_FILE_PATH}" "docker-compose.yml" "$package_dev_compose_filename"
+
+ if docker service ps -q openfn_openfn &>/dev/null; then
+ echo "Service 'openfn_openfn' is running."
+ docker::deploy_config_importer $STACK "$COMPOSE_FILE_PATH/importer/workflows/docker-compose.config.yml" "openfn_workflow_config" "openfn_workflow"
+ else
+ echo "Service 'openfn_openfn' is not running."
+ log warn "Service 'interoperability-layer-openhim' does not appear to be running... skipping configuring of async/sync JeMPI channels"
+ fi
+ ) || {
+ log error "Failed to deploy package"
+ exit 1
+ }
+}
+
+function destroy_package() {
+ docker::stack_destroy $STACK
+}
+
+main() {
+ init_vars "$@"
+ import_sources
+
+ if [[ "${ACTION}" == "init" ]] || [[ "${ACTION}" == "up" ]]; then
+ log info "Running package in Single node mode"
+
+ initialize_package
+ elif [[ "${ACTION}" == "down" ]]; then
+ log info "Scaling down package"
+
+ docker::scale_services $STACK 0
+ elif [[ "${ACTION}" == "destroy" ]]; then
+ log info "Destroying package"
+
+ destroy_package
+ else
+ log error "Valid options are: init, up, down, or destroy"
+ fi
+}
+
+main "$@"
diff --git a/test/cucumber/features/single-mode/openfn.feature b/test/cucumber/features/single-mode/openfn.feature
new file mode 100644
index 00000000..0b61ab77
--- /dev/null
+++ b/test/cucumber/features/single-mode/openfn.feature
@@ -0,0 +1,24 @@
+Feature: OpenFn?
+ Does the OpenFn package work as expected
+
+ Scenario: Init OpenFn
+ Given I use parameters "package init -n=openfn --env-file=.env.local"
+ When I launch the platform with params
+ Then The service "openfn" should be started with 1 replica
+ And There should be 3 service
+ And The service "postgres-1" should be started with 1 replica
+ And The service "worker" should be started with 1 replica
+ And The service "openfn" should be connected to the networks
+ | postgres_public | kafka_public |
+
+ Scenario: Destroy OpenFn
+ Given I use parameters "package destroy -n=openfn --env-file=.env.local"
+ When I launch the platform with params
+ Then The service "openfn" should be removed
+ And The service "postgres-1" should be removed
+ And The service "worker" should be removed
+ And There should be 0 service
+ And There should be 0 volume
+ And There should be 0 config
+ And There should not be network
+ | postgres_public | kafka_public |
diff --git a/test/cucumber/features/single-mode/recipe.feature b/test/cucumber/features/single-mode/recipe.feature
index 93490536..7431601b 100644
--- a/test/cucumber/features/single-mode/recipe.feature
+++ b/test/cucumber/features/single-mode/recipe.feature
@@ -31,6 +31,7 @@ Feature: CDR-DW recipe?
And The service "minio-01" should be started with 1 replica
And The service "dashboard-visualiser-superset" should be started with 1 replica
And The service "analytics-datastore-clickhouse" should be started with 1 replica
+ And The service "openfn" should be started with 1 replica
Scenario: Send Fhir bundle and store the clinical data in the Fhir datastore, the patient info in the CR
Given I have configured the cdr
diff --git a/test/cucumber/package-lock.json b/test/cucumber/package-lock.json
new file mode 100644
index 00000000..09a2c410
--- /dev/null
+++ b/test/cucumber/package-lock.json
@@ -0,0 +1,3964 @@
+{
+ "name": "platform-test",
+ "version": "1.0.0",
+ "lockfileVersion": 2,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "platform-test",
+ "version": "1.0.0",
+ "devDependencies": {
+ "@cucumber/cucumber": "8.5.0",
+ "axios": "^1.6.8",
+ "chai": "4.3.6",
+ "chai-arrays": "2.2.0",
+ "chai-things": "0.2.0",
+ "clickhouse": "^2.6.0",
+ "dockerode": "3.3.2"
+ }
+ },
+ "node_modules/@babel/runtime": {
+ "version": "7.22.6",
+ "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.22.6.tgz",
+ "integrity": "sha512-wDb5pWm4WDdF6LFUde3Jl8WzPA+3ZbxYqkC6xAXuD3irdEHN1k0NfTRrJD8ZD378SJ61miMLCqIOXYhd8x+AJQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "regenerator-runtime": "^0.13.11"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@colors/colors": {
+ "version": "1.5.0",
+ "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz",
+ "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==",
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "engines": {
+ "node": ">=0.1.90"
+ }
+ },
+ "node_modules/@cspotcode/source-map-support": {
+ "version": "0.8.1",
+ "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz",
+ "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/trace-mapping": "0.3.9"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@cucumber/ci-environment": {
+ "version": "9.0.4",
+ "resolved": "https://registry.npmjs.org/@cucumber/ci-environment/-/ci-environment-9.0.4.tgz",
+ "integrity": "sha512-da6H/wtVerhGUP4OCWTOmbNd4+gC1FhAcLzYgn6O68HgQbMwkmV3M8AwtbQWZkfF+Ph7z0M/UQYYdNIDu5V5MA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@cucumber/cucumber": {
+ "version": "8.5.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/cucumber/-/cucumber-8.5.0.tgz",
+ "integrity": "sha512-Ac64lYhgzRxFGf3MnlysMG0zmHa18HfuOU1BA5RfXsrHeWtZx525zH4FsqtRW2AnlYHGu3pQpww5zyQKWi1WDA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@cspotcode/source-map-support": "^0.8.0",
+ "@cucumber/ci-environment": "9.0.4",
+ "@cucumber/cucumber-expressions": "16.0.0",
+ "@cucumber/gherkin": "24.0.0",
+ "@cucumber/gherkin-streams": "5.0.1",
+ "@cucumber/gherkin-utils": "8.0.0",
+ "@cucumber/html-formatter": "19.2.0",
+ "@cucumber/message-streams": "4.0.1",
+ "@cucumber/messages": "19.1.2",
+ "@cucumber/tag-expressions": "4.1.0",
+ "assertion-error-formatter": "^3.0.0",
+ "capital-case": "^1.0.4",
+ "chalk": "^4.1.2",
+ "cli-table3": "0.6.2",
+ "commander": "^9.0.0",
+ "duration": "^0.2.2",
+ "durations": "^3.4.2",
+ "figures": "^3.2.0",
+ "glob": "^7.1.6",
+ "has-ansi": "^4.0.1",
+ "indent-string": "^4.0.0",
+ "is-installed-globally": "^0.4.0",
+ "is-stream": "^2.0.0",
+ "knuth-shuffle-seeded": "^1.0.6",
+ "lodash.merge": "^4.6.2",
+ "lodash.mergewith": "^4.6.2",
+ "mz": "^2.7.0",
+ "progress": "^2.0.3",
+ "resolve-pkg": "^2.0.0",
+ "semver": "7.3.7",
+ "stack-chain": "^2.0.0",
+ "string-argv": "^0.3.1",
+ "strip-ansi": "6.0.1",
+ "supports-color": "^8.1.1",
+ "tmp": "^0.2.1",
+ "util-arity": "^1.1.0",
+ "verror": "^1.10.0",
+ "yup": "^0.32.11"
+ },
+ "bin": {
+ "cucumber-js": "bin/cucumber.js"
+ },
+ "engines": {
+ "node": "12 || 14 || >=16"
+ }
+ },
+ "node_modules/@cucumber/cucumber-expressions": {
+ "version": "16.0.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/cucumber-expressions/-/cucumber-expressions-16.0.0.tgz",
+ "integrity": "sha512-HTh+Pg7oQ5aLuCkSbD2Q6jBaE40M3R/XaLEz+UqD5d9dZRu6P38W4LTooV5bV6dZgBunlMLK8+6ug2ziYvRddw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "regexp-match-indices": "1.0.2"
+ }
+ },
+ "node_modules/@cucumber/gherkin": {
+ "version": "24.0.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/gherkin/-/gherkin-24.0.0.tgz",
+ "integrity": "sha512-b7OsnvX1B8myDAKMc+RAiUX9bzgtNdjGsiMj10O13xu2HBWIOQ19EqBJ4xLO5CFG/lGk1J/+L0lANQVowxLVBg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@cucumber/messages": "^19.0.0"
+ }
+ },
+ "node_modules/@cucumber/gherkin-streams": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/@cucumber/gherkin-streams/-/gherkin-streams-5.0.1.tgz",
+ "integrity": "sha512-/7VkIE/ASxIP/jd4Crlp4JHXqdNFxPGQokqWqsaCCiqBiu5qHoKMxcWNlp9njVL/n9yN4S08OmY3ZR8uC5x74Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "commander": "9.1.0",
+ "source-map-support": "0.5.21"
+ },
+ "bin": {
+ "gherkin-javascript": "bin/gherkin"
+ },
+ "peerDependencies": {
+ "@cucumber/gherkin": ">=22.0.0",
+ "@cucumber/message-streams": ">=4.0.0",
+ "@cucumber/messages": ">=17.1.1"
+ }
+ },
+ "node_modules/@cucumber/gherkin-streams/node_modules/commander": {
+ "version": "9.1.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.1.0.tgz",
+ "integrity": "sha512-i0/MaqBtdbnJ4XQs4Pmyb+oFQl+q0lsAmokVUH92SlSw4fkeAcG3bVon+Qt7hmtF+u3Het6o4VgrcY3qAoEB6w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^12.20.0 || >=14"
+ }
+ },
+ "node_modules/@cucumber/gherkin-utils": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/gherkin-utils/-/gherkin-utils-8.0.0.tgz",
+ "integrity": "sha512-8uIZInEe3cO1cASmy3BA0PbVFUI+xWBnZAxmICbVOPsZaMB85MtESZLafzErgfRQPsHf6uYbVagP7MIjNPM5Jw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@cucumber/messages": "^19.0.0",
+ "@teppeis/multimaps": "2.0.0",
+ "commander": "9.3.0"
+ },
+ "bin": {
+ "gherkin-utils": "bin/gherkin-utils"
+ }
+ },
+ "node_modules/@cucumber/gherkin-utils/node_modules/commander": {
+ "version": "9.3.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.3.0.tgz",
+ "integrity": "sha512-hv95iU5uXPbK83mjrJKuZyFM/LBAoCV/XhVGkS5Je6tl7sxr6A0ITMw5WoRV46/UaJ46Nllm3Xt7IaJhXTIkzw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^12.20.0 || >=14"
+ }
+ },
+ "node_modules/@cucumber/html-formatter": {
+ "version": "19.2.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/html-formatter/-/html-formatter-19.2.0.tgz",
+ "integrity": "sha512-qGms4588jmVF/G3fTbgZvxn6OQw9GaTFV007nZZ9/10M9DfrgRqjFjVxVI9TPV63xOLPicEVoqsKZtcECbdMSA==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "@cucumber/messages": ">=18"
+ }
+ },
+ "node_modules/@cucumber/message-streams": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/@cucumber/message-streams/-/message-streams-4.0.1.tgz",
+ "integrity": "sha512-Kxap9uP5jD8tHUZVjTWgzxemi/0uOsbGjd4LBOSxcJoOCRbESFwemUzilJuzNTB8pcTQUh8D5oudUyxfkJOKmA==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "@cucumber/messages": ">=17.1.1"
+ }
+ },
+ "node_modules/@cucumber/messages": {
+ "version": "19.1.2",
+ "resolved": "https://registry.npmjs.org/@cucumber/messages/-/messages-19.1.2.tgz",
+ "integrity": "sha512-vhWkNmQco+7tk/DWqpN0/R9KTNvsKsXVfZ7IsJs+dEeWmTuRztklHq8lJalwMSQBl71+2/KqGHzOO4BMTC9wIQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/uuid": "8.3.4",
+ "class-transformer": "0.5.1",
+ "reflect-metadata": "0.1.13",
+ "uuid": "8.3.2"
+ }
+ },
+ "node_modules/@cucumber/messages/node_modules/uuid": {
+ "version": "8.3.2",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
+ "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "uuid": "dist/bin/uuid"
+ }
+ },
+ "node_modules/@cucumber/tag-expressions": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/tag-expressions/-/tag-expressions-4.1.0.tgz",
+ "integrity": "sha512-chTnjxV3vryL75N90wJIMdMafXmZoO2JgNJLYpsfcALL2/IQrRiny3vM9DgD5RDCSt1LNloMtb7rGey9YWxCsA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@jridgewell/resolve-uri": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz",
+ "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@jridgewell/sourcemap-codec": {
+ "version": "1.4.15",
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz",
+ "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@jridgewell/trace-mapping": {
+ "version": "0.3.9",
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz",
+ "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/resolve-uri": "^3.0.3",
+ "@jridgewell/sourcemap-codec": "^1.4.10"
+ }
+ },
+ "node_modules/@teppeis/multimaps": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/@teppeis/multimaps/-/multimaps-2.0.0.tgz",
+ "integrity": "sha512-TL1adzq1HdxUf9WYduLcQ/DNGYiz71U31QRgbnr0Ef1cPyOUOsBojxHVWpFeOSUucB6Lrs0LxFRA14ntgtkc9w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10.17"
+ }
+ },
+ "node_modules/@types/lodash": {
+ "version": "4.14.195",
+ "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.195.tgz",
+ "integrity": "sha512-Hwx9EUgdwf2GLarOjQp5ZH8ZmblzcbTBC2wtQWNKARBSxM9ezRIAUpeDTgoQRAFB0+8CNWXVA9+MaSOzOF3nPg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/uuid": {
+ "version": "8.3.4",
+ "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.4.tgz",
+ "integrity": "sha512-c/I8ZRb51j+pYGAu5CrFMRxqZ2ke4y2grEBO5AUjgSkSk+qT2Ea+OdWElz/OiMf5MNpn2b17kuVBwZLQJXzihw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/ajv": {
+ "version": "6.12.6",
+ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+ "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "fast-deep-equal": "^3.1.1",
+ "fast-json-stable-stringify": "^2.0.0",
+ "json-schema-traverse": "^0.4.1",
+ "uri-js": "^4.2.2"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/epoberezkin"
+ }
+ },
+ "node_modules/ansi-regex": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz",
+ "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/ansi-styles": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+ "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "color-convert": "^2.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/ansi-styles?sponsor=1"
+ }
+ },
+ "node_modules/any-promise": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
+ "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/asn1": {
+ "version": "0.2.6",
+ "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz",
+ "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "safer-buffer": "~2.1.0"
+ }
+ },
+ "node_modules/assert-plus": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz",
+ "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.8"
+ }
+ },
+ "node_modules/assertion-error": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz",
+ "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/assertion-error-formatter": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/assertion-error-formatter/-/assertion-error-formatter-3.0.0.tgz",
+ "integrity": "sha512-6YyAVLrEze0kQ7CmJfUgrLHb+Y7XghmL2Ie7ijVa2Y9ynP3LV+VDiwFk62Dn0qtqbmY0BT0ss6p1xxpiF2PYbQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "diff": "^4.0.1",
+ "pad-right": "^0.2.2",
+ "repeat-string": "^1.6.1"
+ }
+ },
+ "node_modules/asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/aws-sign2": {
+ "version": "0.7.0",
+ "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz",
+ "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/aws4": {
+ "version": "1.12.0",
+ "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.12.0.tgz",
+ "integrity": "sha512-NmWvPnx0F1SfrQbYwOi7OeaNGokp9XhzNioJ/CSBs8Qa4vxug81mhJEAVZwxXuBmYB5KDRfMq/F3RR0BIU7sWg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/axios": {
+ "version": "1.6.8",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz",
+ "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "follow-redirects": "^1.15.6",
+ "form-data": "^4.0.0",
+ "proxy-from-env": "^1.1.0"
+ }
+ },
+ "node_modules/balanced-match": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/base64-js": {
+ "version": "1.5.1",
+ "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+ "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT"
+ },
+ "node_modules/bcrypt-pbkdf": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz",
+ "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "tweetnacl": "^0.14.3"
+ }
+ },
+ "node_modules/bl": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz",
+ "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "buffer": "^5.5.0",
+ "inherits": "^2.0.4",
+ "readable-stream": "^3.4.0"
+ }
+ },
+ "node_modules/brace-expansion": {
+ "version": "1.1.11",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
+ "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "balanced-match": "^1.0.0",
+ "concat-map": "0.0.1"
+ }
+ },
+ "node_modules/buffer": {
+ "version": "5.7.1",
+ "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz",
+ "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "base64-js": "^1.3.1",
+ "ieee754": "^1.1.13"
+ }
+ },
+ "node_modules/buffer-from": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
+ "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/capital-case": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/capital-case/-/capital-case-1.0.4.tgz",
+ "integrity": "sha512-ds37W8CytHgwnhGGTi88pcPyR15qoNkOpYwmMMfnWqqWgESapLqvDx6huFjQ5vqWSn2Z06173XNA7LtMOeUh1A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "no-case": "^3.0.4",
+ "tslib": "^2.0.3",
+ "upper-case-first": "^2.0.2"
+ }
+ },
+ "node_modules/caseless": {
+ "version": "0.12.0",
+ "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
+ "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==",
+ "dev": true,
+ "license": "Apache-2.0"
+ },
+ "node_modules/chai": {
+ "version": "4.3.6",
+ "resolved": "https://registry.npmjs.org/chai/-/chai-4.3.6.tgz",
+ "integrity": "sha512-bbcp3YfHCUzMOvKqsztczerVgBKSsEijCySNlHHbX3VG1nskvqjz5Rfso1gGwD6w6oOV3eI60pKuMOV5MV7p3Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "assertion-error": "^1.1.0",
+ "check-error": "^1.0.2",
+ "deep-eql": "^3.0.1",
+ "get-func-name": "^2.0.0",
+ "loupe": "^2.3.1",
+ "pathval": "^1.1.1",
+ "type-detect": "^4.0.5"
+ },
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/chai-arrays": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/chai-arrays/-/chai-arrays-2.2.0.tgz",
+ "integrity": "sha512-4awrdGI2EH8owJ9I58PXwG4N56/FiM8bsn4CVSNEgr4GKAM6Kq5JPVApUbhUBjDakbZNuRvV7quRSC38PWq/tg==",
+ "dev": true,
+ "license": "ISC",
+ "engines": {
+ "node": ">=0.10"
+ }
+ },
+ "node_modules/chai-things": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/chai-things/-/chai-things-0.2.0.tgz",
+ "integrity": "sha512-6ns0SU21xdRCoEXVKH3HGbwnsgfVMXQ+sU5V8PI9rfxaITos8lss1vUxbF1FAcJKjfqmmmLVlr/z3sLes00w+A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/chalk": {
+ "version": "4.1.2",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+ "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-styles": "^4.1.0",
+ "supports-color": "^7.1.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/chalk?sponsor=1"
+ }
+ },
+ "node_modules/chalk/node_modules/supports-color": {
+ "version": "7.2.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+ "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "has-flag": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/check-error": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz",
+ "integrity": "sha512-BrgHpW9NURQgzoNyjfq0Wu6VFO6D7IZEmJNdtgNqpzGG8RuNFHt2jQxWlAs4HMe119chBnv+34syEZtc6IhLtA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/chownr": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz",
+ "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/class-transformer": {
+ "version": "0.5.1",
+ "resolved": "https://registry.npmjs.org/class-transformer/-/class-transformer-0.5.1.tgz",
+ "integrity": "sha512-SQa1Ws6hUbfC98vKGxZH3KFY0Y1lm5Zm0SY8XX9zbK7FJCyVEac3ATW0RIpwzW+oOfmHE5PMPufDG9hCfoEOMw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/cli-table3": {
+ "version": "0.6.2",
+ "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.2.tgz",
+ "integrity": "sha512-QyavHCaIC80cMivimWu4aWHilIpiDpfm3hGmqAmXVL1UsnbLuBSMd21hTX6VY4ZSDSM73ESLeF8TOYId3rBTbw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "string-width": "^4.2.0"
+ },
+ "engines": {
+ "node": "10.* || >= 12.*"
+ },
+ "optionalDependencies": {
+ "@colors/colors": "1.5.0"
+ }
+ },
+ "node_modules/clickhouse": {
+ "version": "2.6.0",
+ "resolved": "https://registry.npmjs.org/clickhouse/-/clickhouse-2.6.0.tgz",
+ "integrity": "sha512-HC5OV99GJOup4qZsTuWWPpXlj+847Z0OeygDU2x22rNYost0V/vWapzFWYZdV/5iRbGMrhFQPOyQEzmGvoaWRQ==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "JSONStream": "1.3.4",
+ "lodash": "4.17.21",
+ "querystring": "0.2.0",
+ "request": "2.88.0",
+ "stream2asynciter": "1.0.3",
+ "through": "2.3.8",
+ "tsv": "0.2.0",
+ "uuid": "3.4.0"
+ }
+ },
+ "node_modules/color-convert": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+ "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "color-name": "~1.1.4"
+ },
+ "engines": {
+ "node": ">=7.0.0"
+ }
+ },
+ "node_modules/color-name": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "delayed-stream": "~1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/commander": {
+ "version": "9.5.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.5.0.tgz",
+ "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^12.20.0 || >=14"
+ }
+ },
+ "node_modules/concat-map": {
+ "version": "0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+ "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/core-util-is": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
+ "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/d": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz",
+ "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "es5-ext": "^0.10.50",
+ "type": "^1.0.1"
+ }
+ },
+ "node_modules/dashdash": {
+ "version": "1.14.1",
+ "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz",
+ "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "assert-plus": "^1.0.0"
+ },
+ "engines": {
+ "node": ">=0.10"
+ }
+ },
+ "node_modules/debug": {
+ "version": "4.3.4",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
+ "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ms": "2.1.2"
+ },
+ "engines": {
+ "node": ">=6.0"
+ },
+ "peerDependenciesMeta": {
+ "supports-color": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/deep-eql": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz",
+ "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "type-detect": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=0.12"
+ }
+ },
+ "node_modules/delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
+ "node_modules/diff": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz",
+ "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "engines": {
+ "node": ">=0.3.1"
+ }
+ },
+ "node_modules/docker-modem": {
+ "version": "3.0.8",
+ "resolved": "https://registry.npmjs.org/docker-modem/-/docker-modem-3.0.8.tgz",
+ "integrity": "sha512-f0ReSURdM3pcKPNS30mxOHSbaFLcknGmQjwSfmbcdOw1XWKXVhukM3NJHhr7NpY9BIyyWQb0EBo3KQvvuU5egQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "debug": "^4.1.1",
+ "readable-stream": "^3.5.0",
+ "split-ca": "^1.0.1",
+ "ssh2": "^1.11.0"
+ },
+ "engines": {
+ "node": ">= 8.0"
+ }
+ },
+ "node_modules/dockerode": {
+ "version": "3.3.2",
+ "resolved": "https://registry.npmjs.org/dockerode/-/dockerode-3.3.2.tgz",
+ "integrity": "sha512-oXN+1XVH2TeyE0Jj9Ci6Fim8ZIDxyqeJrkx9qhEOaRiA+nhLihKfd3M2L+Aqrj5C2ObPw8RVN2zPWvvk0x2dwg==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "docker-modem": "^3.0.0",
+ "tar-fs": "~2.0.1"
+ },
+ "engines": {
+ "node": ">= 8.0"
+ }
+ },
+ "node_modules/duration": {
+ "version": "0.2.2",
+ "resolved": "https://registry.npmjs.org/duration/-/duration-0.2.2.tgz",
+ "integrity": "sha512-06kgtea+bGreF5eKYgI/36A6pLXggY7oR4p1pq4SmdFBn1ReOL5D8RhG64VrqfTTKNucqqtBAwEj8aB88mcqrg==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "d": "1",
+ "es5-ext": "~0.10.46"
+ }
+ },
+ "node_modules/durations": {
+ "version": "3.4.2",
+ "resolved": "https://registry.npmjs.org/durations/-/durations-3.4.2.tgz",
+ "integrity": "sha512-V/lf7y33dGaypZZetVI1eu7BmvkbC4dItq12OElLRpKuaU5JxQstV2zHwLv8P7cNbQ+KL1WD80zMCTx5dNC4dg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/ecc-jsbn": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz",
+ "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "jsbn": "~0.1.0",
+ "safer-buffer": "^2.1.0"
+ }
+ },
+ "node_modules/emoji-regex": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
+ "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/end-of-stream": {
+ "version": "1.4.4",
+ "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
+ "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "once": "^1.4.0"
+ }
+ },
+ "node_modules/es5-ext": {
+ "version": "0.10.62",
+ "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.62.tgz",
+ "integrity": "sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==",
+ "dev": true,
+ "hasInstallScript": true,
+ "license": "ISC",
+ "dependencies": {
+ "es6-iterator": "^2.0.3",
+ "es6-symbol": "^3.1.3",
+ "next-tick": "^1.1.0"
+ },
+ "engines": {
+ "node": ">=0.10"
+ }
+ },
+ "node_modules/es6-iterator": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz",
+ "integrity": "sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "d": "1",
+ "es5-ext": "^0.10.35",
+ "es6-symbol": "^3.1.1"
+ }
+ },
+ "node_modules/es6-symbol": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz",
+ "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "d": "^1.0.1",
+ "ext": "^1.1.2"
+ }
+ },
+ "node_modules/escape-string-regexp": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
+ "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.8.0"
+ }
+ },
+ "node_modules/ext": {
+ "version": "1.7.0",
+ "resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz",
+ "integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "type": "^2.7.2"
+ }
+ },
+ "node_modules/ext/node_modules/type": {
+ "version": "2.7.2",
+ "resolved": "https://registry.npmjs.org/type/-/type-2.7.2.tgz",
+ "integrity": "sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/extend": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz",
+ "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/extsprintf": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.4.1.tgz",
+ "integrity": "sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA==",
+ "dev": true,
+ "engines": [
+ "node >=0.6.0"
+ ],
+ "license": "MIT"
+ },
+ "node_modules/fast-deep-equal": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
+ "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/fast-json-stable-stringify": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+ "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/figures": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz",
+ "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "escape-string-regexp": "^1.0.5"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/follow-redirects": {
+ "version": "1.15.6",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz",
+ "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "individual",
+ "url": "https://github.com/sponsors/RubenVerborgh"
+ }
+ ],
+ "license": "MIT",
+ "engines": {
+ "node": ">=4.0"
+ },
+ "peerDependenciesMeta": {
+ "debug": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/forever-agent": {
+ "version": "0.6.1",
+ "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz",
+ "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/form-data": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+ "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "mime-types": "^2.1.12"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/fs-constants": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz",
+ "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/fs.realpath": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
+ "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/get-func-name": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz",
+ "integrity": "sha512-Hm0ixYtaSZ/V7C8FJrtZIuBBI+iSgL+1Aq82zSu8VQNB4S3Gk8e7Qs3VwBDJAhmRZcFqkl3tQu36g/Foh5I5ig==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/getpass": {
+ "version": "0.1.7",
+ "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz",
+ "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "assert-plus": "^1.0.0"
+ }
+ },
+ "node_modules/glob": {
+ "version": "7.2.3",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
+ "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "fs.realpath": "^1.0.0",
+ "inflight": "^1.0.4",
+ "inherits": "2",
+ "minimatch": "^3.1.1",
+ "once": "^1.3.0",
+ "path-is-absolute": "^1.0.0"
+ },
+ "engines": {
+ "node": "*"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/global-dirs": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-3.0.1.tgz",
+ "integrity": "sha512-NBcGGFbBA9s1VzD41QXDG+3++t9Mn5t1FpLdhESY6oKY4gYTFpX4wO3sqGUa0Srjtbfj3szX0RnemmrVRUdULA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ini": "2.0.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/har-schema": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz",
+ "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==",
+ "dev": true,
+ "license": "ISC",
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/har-validator": {
+ "version": "5.1.5",
+ "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz",
+ "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ajv": "^6.12.3",
+ "har-schema": "^2.0.0"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/has-ansi": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-4.0.1.tgz",
+ "integrity": "sha512-Qr4RtTm30xvEdqUXbSBVWDu+PrTokJOwe/FU+VdfJPk+MXAPoeOzKpRyrDTnZIJwAkQ4oBLTU53nu0HrkF/Z2A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-regex": "^4.1.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/has-flag": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+ "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/http-signature": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz",
+ "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "assert-plus": "^1.0.0",
+ "jsprim": "^1.2.2",
+ "sshpk": "^1.7.0"
+ },
+ "engines": {
+ "node": ">=0.8",
+ "npm": ">=1.3.7"
+ }
+ },
+ "node_modules/ieee754": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
+ "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/indent-string": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz",
+ "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/inflight": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+ "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "once": "^1.3.0",
+ "wrappy": "1"
+ }
+ },
+ "node_modules/inherits": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/ini": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/ini/-/ini-2.0.0.tgz",
+ "integrity": "sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==",
+ "dev": true,
+ "license": "ISC",
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/is-fullwidth-code-point": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
+ "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/is-installed-globally": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.4.0.tgz",
+ "integrity": "sha512-iwGqO3J21aaSkC7jWnHP/difazwS7SFeIqxv6wEtLU8Y5KlzFTjyqcSIT0d8s4+dDhKytsk9PJZ2BkS5eZwQRQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "global-dirs": "^3.0.0",
+ "is-path-inside": "^3.0.2"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/is-path-inside": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
+ "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/is-stream": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz",
+ "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/is-typedarray": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
+ "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/isstream": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
+ "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/jsbn": {
+ "version": "0.1.1",
+ "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
+ "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/json-schema": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz",
+ "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==",
+ "dev": true,
+ "license": "(AFL-2.1 OR BSD-3-Clause)"
+ },
+ "node_modules/json-schema-traverse": {
+ "version": "0.4.1",
+ "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+ "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/json-stringify-safe": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz",
+ "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/jsonparse": {
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz",
+ "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==",
+ "dev": true,
+ "engines": [
+ "node >= 0.2.0"
+ ],
+ "license": "MIT"
+ },
+ "node_modules/JSONStream": {
+ "version": "1.3.4",
+ "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.4.tgz",
+ "integrity": "sha512-Y7vfi3I5oMOYIr+WxV8NZxDSwcbNgzdKYsTNInmycOq9bUYwGg9ryu57Wg5NLmCjqdFPNUmpMBo3kSJN9tCbXg==",
+ "dev": true,
+ "license": "(MIT OR Apache-2.0)",
+ "dependencies": {
+ "jsonparse": "^1.2.0",
+ "through": ">=2.2.7 <3"
+ },
+ "bin": {
+ "JSONStream": "bin.js"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/jsprim": {
+ "version": "1.4.2",
+ "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz",
+ "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "assert-plus": "1.0.0",
+ "extsprintf": "1.3.0",
+ "json-schema": "0.4.0",
+ "verror": "1.10.0"
+ },
+ "engines": {
+ "node": ">=0.6.0"
+ }
+ },
+ "node_modules/jsprim/node_modules/extsprintf": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
+ "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==",
+ "dev": true,
+ "engines": [
+ "node >=0.6.0"
+ ],
+ "license": "MIT"
+ },
+ "node_modules/jsprim/node_modules/verror": {
+ "version": "1.10.0",
+ "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz",
+ "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==",
+ "dev": true,
+ "engines": [
+ "node >=0.6.0"
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "assert-plus": "^1.0.0",
+ "core-util-is": "1.0.2",
+ "extsprintf": "^1.2.0"
+ }
+ },
+ "node_modules/jsprim/node_modules/verror/node_modules/extsprintf": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.4.1.tgz",
+ "integrity": "sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA==",
+ "dev": true,
+ "engines": [
+ "node >=0.6.0"
+ ],
+ "license": "MIT"
+ },
+ "node_modules/knuth-shuffle-seeded": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/knuth-shuffle-seeded/-/knuth-shuffle-seeded-1.0.6.tgz",
+ "integrity": "sha512-9pFH0SplrfyKyojCLxZfMcvkhf5hH0d+UwR9nTVJ/DDQJGuzcXjTwB7TP7sDfehSudlGGaOLblmEWqv04ERVWg==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "seed-random": "~2.2.0"
+ }
+ },
+ "node_modules/lodash": {
+ "version": "4.17.21",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/lodash-es": {
+ "version": "4.17.21",
+ "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz",
+ "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/lodash.merge": {
+ "version": "4.6.2",
+ "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
+ "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/lodash.mergewith": {
+ "version": "4.6.2",
+ "resolved": "https://registry.npmjs.org/lodash.mergewith/-/lodash.mergewith-4.6.2.tgz",
+ "integrity": "sha512-GK3g5RPZWTRSeLSpgP8Xhra+pnjBC56q9FZYe1d5RN3TJ35dbkGy3YqBSMbyCrlbi+CM9Z3Jk5yTL7RCsqboyQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/loupe": {
+ "version": "2.3.6",
+ "resolved": "https://registry.npmjs.org/loupe/-/loupe-2.3.6.tgz",
+ "integrity": "sha512-RaPMZKiMy8/JruncMU5Bt6na1eftNoo++R4Y+N2FrxkDVTrGvcyzFTsaGif4QTeKESheMGegbhw6iUAq+5A8zA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "get-func-name": "^2.0.0"
+ }
+ },
+ "node_modules/lower-case": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz",
+ "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "tslib": "^2.0.3"
+ }
+ },
+ "node_modules/lru-cache": {
+ "version": "6.0.0",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
+ "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "yallist": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "mime-db": "1.52.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/minimatch": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+ "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "brace-expansion": "^1.1.7"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/mkdirp-classic": {
+ "version": "0.5.3",
+ "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz",
+ "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/ms": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
+ "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/mz": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
+ "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "any-promise": "^1.0.0",
+ "object-assign": "^4.0.1",
+ "thenify-all": "^1.0.0"
+ }
+ },
+ "node_modules/nan": {
+ "version": "2.17.0",
+ "resolved": "https://registry.npmjs.org/nan/-/nan-2.17.0.tgz",
+ "integrity": "sha512-2ZTgtl0nJsO0KQCjEpxcIr5D+Yv90plTitZt9JBfQvVJDS5seMl3FOvsh3+9CoYWXf/1l5OaZzzF6nDm4cagaQ==",
+ "dev": true,
+ "license": "MIT",
+ "optional": true
+ },
+ "node_modules/nanoclone": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/nanoclone/-/nanoclone-0.2.1.tgz",
+ "integrity": "sha512-wynEP02LmIbLpcYw8uBKpcfF6dmg2vcpKqxeH5UcoKEYdExslsdUA4ugFauuaeYdTB76ez6gJW8XAZ6CgkXYxA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/next-tick": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz",
+ "integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/no-case": {
+ "version": "3.0.4",
+ "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz",
+ "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "lower-case": "^2.0.2",
+ "tslib": "^2.0.3"
+ }
+ },
+ "node_modules/oauth-sign": {
+ "version": "0.9.0",
+ "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz",
+ "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/once": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+ "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "wrappy": "1"
+ }
+ },
+ "node_modules/pad-right": {
+ "version": "0.2.2",
+ "resolved": "https://registry.npmjs.org/pad-right/-/pad-right-0.2.2.tgz",
+ "integrity": "sha512-4cy8M95ioIGolCoMmm2cMntGR1lPLEbOMzOKu8bzjuJP6JpzEMQcDHmh7hHLYGgob+nKe1YHFMaG4V59HQa89g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "repeat-string": "^1.5.2"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/path-is-absolute": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+ "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/pathval": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz",
+ "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/performance-now": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
+ "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/progress": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz",
+ "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
+ "node_modules/property-expr": {
+ "version": "2.0.5",
+ "resolved": "https://registry.npmjs.org/property-expr/-/property-expr-2.0.5.tgz",
+ "integrity": "sha512-IJUkICM5dP5znhCckHSv30Q4b5/JA5enCtkRHYaOVOAocnH/1BQEYTC5NMfT3AVl/iXKdr3aqQbQn9DxyWknwA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/proxy-from-env": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+ "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/psl": {
+ "version": "1.9.0",
+ "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz",
+ "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/pump": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
+ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "end-of-stream": "^1.1.0",
+ "once": "^1.3.1"
+ }
+ },
+ "node_modules/punycode": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz",
+ "integrity": "sha512-jmYNElW7yvO7TV33CjSmvSiE2yco3bV2czu/OzDKdMNVZQWfxCblURLhf+47syQRBntjfLdd/H0egrzIG+oaFQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/qs": {
+ "version": "6.5.3",
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.3.tgz",
+ "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "engines": {
+ "node": ">=0.6"
+ }
+ },
+ "node_modules/querystring": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz",
+ "integrity": "sha512-X/xY82scca2tau62i9mDyU9K+I+djTMUsvwf7xnUX5GLvVzgJybOJf4Y6o9Zx3oJK/LSXg5tTZBjwzqVPaPO2g==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.4.x"
+ }
+ },
+ "node_modules/readable-stream": {
+ "version": "3.6.2",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz",
+ "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "inherits": "^2.0.3",
+ "string_decoder": "^1.1.1",
+ "util-deprecate": "^1.0.1"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/reflect-metadata": {
+ "version": "0.1.13",
+ "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz",
+ "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==",
+ "dev": true,
+ "license": "Apache-2.0"
+ },
+ "node_modules/regenerator-runtime": {
+ "version": "0.13.11",
+ "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz",
+ "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/regexp-match-indices": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/regexp-match-indices/-/regexp-match-indices-1.0.2.tgz",
+ "integrity": "sha512-DwZuAkt8NF5mKwGGER1EGh2PRqyvhRhhLviH+R8y8dIuaQROlUfXjt4s9ZTXstIsSkptf06BSvwcEmmfheJJWQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "regexp-tree": "^0.1.11"
+ }
+ },
+ "node_modules/regexp-tree": {
+ "version": "0.1.27",
+ "resolved": "https://registry.npmjs.org/regexp-tree/-/regexp-tree-0.1.27.tgz",
+ "integrity": "sha512-iETxpjK6YoRWJG5o6hXLwvjYAoW+FEZn9os0PD/b6AP6xQwsa/Y7lCVgIixBbUPMfhu+i2LtdeAqVTgGlQarfA==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "regexp-tree": "bin/regexp-tree"
+ }
+ },
+ "node_modules/repeat-string": {
+ "version": "1.6.1",
+ "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz",
+ "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10"
+ }
+ },
+ "node_modules/request": {
+ "version": "2.88.0",
+ "resolved": "https://registry.npmjs.org/request/-/request-2.88.0.tgz",
+ "integrity": "sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "aws-sign2": "~0.7.0",
+ "aws4": "^1.8.0",
+ "caseless": "~0.12.0",
+ "combined-stream": "~1.0.6",
+ "extend": "~3.0.2",
+ "forever-agent": "~0.6.1",
+ "form-data": "~2.3.2",
+ "har-validator": "~5.1.0",
+ "http-signature": "~1.2.0",
+ "is-typedarray": "~1.0.0",
+ "isstream": "~0.1.2",
+ "json-stringify-safe": "~5.0.1",
+ "mime-types": "~2.1.19",
+ "oauth-sign": "~0.9.0",
+ "performance-now": "^2.1.0",
+ "qs": "~6.5.2",
+ "safe-buffer": "^5.1.2",
+ "tough-cookie": "~2.4.3",
+ "tunnel-agent": "^0.6.0",
+ "uuid": "^3.3.2"
+ },
+ "engines": {
+ "node": ">= 4"
+ }
+ },
+ "node_modules/request/node_modules/form-data": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz",
+ "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.6",
+ "mime-types": "^2.1.12"
+ },
+ "engines": {
+ "node": ">= 0.12"
+ }
+ },
+ "node_modules/resolve-from": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz",
+ "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/resolve-pkg": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/resolve-pkg/-/resolve-pkg-2.0.0.tgz",
+ "integrity": "sha512-+1lzwXehGCXSeryaISr6WujZzowloigEofRB+dj75y9RRa/obVcYgbHJd53tdYw8pvZj8GojXaaENws8Ktw/hQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "resolve-from": "^5.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/rimraf": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
+ "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "glob": "^7.1.3"
+ },
+ "bin": {
+ "rimraf": "bin.js"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/safe-buffer": {
+ "version": "5.2.1",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT"
+ },
+ "node_modules/safer-buffer": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/seed-random": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/seed-random/-/seed-random-2.2.0.tgz",
+ "integrity": "sha512-34EQV6AAHQGhoc0tn/96a9Fsi6v2xdqe/dMUwljGRaFOzR3EgRmECvD0O8vi8X+/uQ50LGHfkNu/Eue5TPKZkQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/semver": {
+ "version": "7.3.7",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz",
+ "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "lru-cache": "^6.0.0"
+ },
+ "bin": {
+ "semver": "bin/semver.js"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/source-map": {
+ "version": "0.6.1",
+ "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
+ "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/source-map-support": {
+ "version": "0.5.21",
+ "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz",
+ "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "buffer-from": "^1.0.0",
+ "source-map": "^0.6.0"
+ }
+ },
+ "node_modules/split-ca": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/split-ca/-/split-ca-1.0.1.tgz",
+ "integrity": "sha512-Q5thBSxp5t8WPTTJQS59LrGqOZqOsrhDGDVm8azCqIBjSBd7nd9o2PM+mDulQQkh8h//4U6hFZnc/mul8t5pWQ==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/ssh2": {
+ "version": "1.14.0",
+ "resolved": "https://registry.npmjs.org/ssh2/-/ssh2-1.14.0.tgz",
+ "integrity": "sha512-AqzD1UCqit8tbOKoj6ztDDi1ffJZ2rV2SwlgrVVrHPkV5vWqGJOVp5pmtj18PunkPJAuKQsnInyKV+/Nb2bUnA==",
+ "dev": true,
+ "hasInstallScript": true,
+ "dependencies": {
+ "asn1": "^0.2.6",
+ "bcrypt-pbkdf": "^1.0.2"
+ },
+ "engines": {
+ "node": ">=10.16.0"
+ },
+ "optionalDependencies": {
+ "cpu-features": "~0.0.8",
+ "nan": "^2.17.0"
+ }
+ },
+ "node_modules/sshpk": {
+ "version": "1.18.0",
+ "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.18.0.tgz",
+ "integrity": "sha512-2p2KJZTSqQ/I3+HX42EpYOa2l3f8Erv8MWKsy2I9uf4wA7yFIkXRffYdsx86y6z4vHtV8u7g+pPlr8/4ouAxsQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "asn1": "~0.2.3",
+ "assert-plus": "^1.0.0",
+ "bcrypt-pbkdf": "^1.0.0",
+ "dashdash": "^1.12.0",
+ "ecc-jsbn": "~0.1.1",
+ "getpass": "^0.1.1",
+ "jsbn": "~0.1.0",
+ "safer-buffer": "^2.0.2",
+ "tweetnacl": "~0.14.0"
+ },
+ "bin": {
+ "sshpk-conv": "bin/sshpk-conv",
+ "sshpk-sign": "bin/sshpk-sign",
+ "sshpk-verify": "bin/sshpk-verify"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/stack-chain": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/stack-chain/-/stack-chain-2.0.0.tgz",
+ "integrity": "sha512-GGrHXePi305aW7XQweYZZwiRwR7Js3MWoK/EHzzB9ROdc75nCnjSJVi21rdAGxFl+yCx2L2qdfl5y7NO4lTyqg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/stream2asynciter": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/stream2asynciter/-/stream2asynciter-1.0.3.tgz",
+ "integrity": "sha512-9/dEZW+LQjuW6ub5hmWi4n9Pn8W8qA8k7NAE1isecesA164e73xTdy1CJ3S9o9YS+O21HuiK7T+4uS7FgKDy4w==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/string_decoder": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
+ "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "safe-buffer": "~5.2.0"
+ }
+ },
+ "node_modules/string-argv": {
+ "version": "0.3.2",
+ "resolved": "https://registry.npmjs.org/string-argv/-/string-argv-0.3.2.tgz",
+ "integrity": "sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.6.19"
+ }
+ },
+ "node_modules/string-width": {
+ "version": "4.2.3",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
+ "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "emoji-regex": "^8.0.0",
+ "is-fullwidth-code-point": "^3.0.0",
+ "strip-ansi": "^6.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/strip-ansi": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+ "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-regex": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/strip-ansi/node_modules/ansi-regex": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+ "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/supports-color": {
+ "version": "8.1.1",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz",
+ "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "has-flag": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/supports-color?sponsor=1"
+ }
+ },
+ "node_modules/tar-fs": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.0.1.tgz",
+ "integrity": "sha512-6tzWDMeroL87uF/+lin46k+Q+46rAJ0SyPGz7OW7wTgblI273hsBqk2C1j0/xNadNLKDTUL9BukSjB7cwgmlPA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "chownr": "^1.1.1",
+ "mkdirp-classic": "^0.5.2",
+ "pump": "^3.0.0",
+ "tar-stream": "^2.0.0"
+ }
+ },
+ "node_modules/tar-stream": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz",
+ "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "bl": "^4.0.3",
+ "end-of-stream": "^1.4.1",
+ "fs-constants": "^1.0.0",
+ "inherits": "^2.0.3",
+ "readable-stream": "^3.1.1"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/thenify": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
+ "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "any-promise": "^1.0.0"
+ }
+ },
+ "node_modules/thenify-all": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz",
+ "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "thenify": ">= 3.1.0 < 4"
+ },
+ "engines": {
+ "node": ">=0.8"
+ }
+ },
+ "node_modules/through": {
+ "version": "2.3.8",
+ "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
+ "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/tmp": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz",
+ "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "rimraf": "^3.0.0"
+ },
+ "engines": {
+ "node": ">=8.17.0"
+ }
+ },
+ "node_modules/toposort": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/toposort/-/toposort-2.0.2.tgz",
+ "integrity": "sha512-0a5EOkAUp8D4moMi2W8ZF8jcga7BgZd91O/yabJCFY8az+XSzeGyTKs0Aoo897iV1Nj6guFq8orWDS96z91oGg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/tough-cookie": {
+ "version": "2.4.3",
+ "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.4.3.tgz",
+ "integrity": "sha512-Q5srk/4vDM54WJsJio3XNn6K2sCG+CQ8G5Wz6bZhRZoAe/+TxjWB/GlFAnYEbkYVlON9FMk/fE3h2RLpPXo4lQ==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "psl": "^1.1.24",
+ "punycode": "^1.4.1"
+ },
+ "engines": {
+ "node": ">=0.8"
+ }
+ },
+ "node_modules/tslib": {
+ "version": "2.6.0",
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.0.tgz",
+ "integrity": "sha512-7At1WUettjcSRHXCyYtTselblcHl9PJFFVKiCAy/bY97+BPZXSQ2wbq0P9s8tK2G7dFQfNnlJnPAiArVBVBsfA==",
+ "dev": true,
+ "license": "0BSD"
+ },
+ "node_modules/tsv": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/tsv/-/tsv-0.2.0.tgz",
+ "integrity": "sha512-GG6xbOP85giXXom0dS6z9uyDsxktznjpa1AuDlPrIXDqDnbhjr9Vk6Us8iz6U1nENL4CPS2jZDvIjEdaZsmc4Q==",
+ "dev": true,
+ "license": "MIT (ricardo.mit-license.org)"
+ },
+ "node_modules/tunnel-agent": {
+ "version": "0.6.0",
+ "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
+ "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "safe-buffer": "^5.0.1"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/tweetnacl": {
+ "version": "0.14.5",
+ "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz",
+ "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==",
+ "dev": true,
+ "license": "Unlicense"
+ },
+ "node_modules/type": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz",
+ "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/type-detect": {
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz",
+ "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/upper-case-first": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/upper-case-first/-/upper-case-first-2.0.2.tgz",
+ "integrity": "sha512-514ppYHBaKwfJRK/pNC6c/OxfGa0obSnAl106u97Ed0I625Nin96KAjttZF6ZL3e1XLtphxnqrOi9iWgm+u+bg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "tslib": "^2.0.3"
+ }
+ },
+ "node_modules/uri-js": {
+ "version": "4.4.1",
+ "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+ "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "punycode": "^2.1.0"
+ }
+ },
+ "node_modules/uri-js/node_modules/punycode": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
+ "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/util-arity": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/util-arity/-/util-arity-1.1.0.tgz",
+ "integrity": "sha512-kkyIsXKwemfSy8ZEoaIz06ApApnWsk5hQO0vLjZS6UkBiGiW++Jsyb8vSBoc0WKlffGoGs5yYy/j5pp8zckrFA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/util-deprecate": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/uuid": {
+ "version": "3.4.0",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz",
+ "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "uuid": "bin/uuid"
+ }
+ },
+ "node_modules/verror": {
+ "version": "1.10.1",
+ "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.1.tgz",
+ "integrity": "sha512-veufcmxri4e3XSrT0xwfUR7kguIkaxBeosDg00yDWhk49wdwkSUrvvsm7nc75e1PUyvIeZj6nS8VQRYz2/S4Xg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "assert-plus": "^1.0.0",
+ "core-util-is": "1.0.2",
+ "extsprintf": "^1.2.0"
+ },
+ "engines": {
+ "node": ">=0.6.0"
+ }
+ },
+ "node_modules/wrappy": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+ "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/yallist": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
+ "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/yup": {
+ "version": "0.32.11",
+ "resolved": "https://registry.npmjs.org/yup/-/yup-0.32.11.tgz",
+ "integrity": "sha512-Z2Fe1bn+eLstG8DRR6FTavGD+MeAwyfmouhHsIUgaADz8jvFKbO/fXc2trJKZg+5EBjh4gGm3iU/t3onKlXHIg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.15.4",
+ "@types/lodash": "^4.14.175",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "nanoclone": "^0.2.1",
+ "property-expr": "^2.0.4",
+ "toposort": "^2.0.2"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ }
+ },
+ "dependencies": {
+ "@babel/runtime": {
+ "version": "7.22.6",
+ "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.22.6.tgz",
+ "integrity": "sha512-wDb5pWm4WDdF6LFUde3Jl8WzPA+3ZbxYqkC6xAXuD3irdEHN1k0NfTRrJD8ZD378SJ61miMLCqIOXYhd8x+AJQ==",
+ "dev": true,
+ "requires": {
+ "regenerator-runtime": "^0.13.11"
+ }
+ },
+ "@colors/colors": {
+ "version": "1.5.0",
+ "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz",
+ "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==",
+ "dev": true,
+ "optional": true
+ },
+ "@cspotcode/source-map-support": {
+ "version": "0.8.1",
+ "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz",
+ "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==",
+ "dev": true,
+ "requires": {
+ "@jridgewell/trace-mapping": "0.3.9"
+ }
+ },
+ "@cucumber/ci-environment": {
+ "version": "9.0.4",
+ "resolved": "https://registry.npmjs.org/@cucumber/ci-environment/-/ci-environment-9.0.4.tgz",
+ "integrity": "sha512-da6H/wtVerhGUP4OCWTOmbNd4+gC1FhAcLzYgn6O68HgQbMwkmV3M8AwtbQWZkfF+Ph7z0M/UQYYdNIDu5V5MA==",
+ "dev": true
+ },
+ "@cucumber/cucumber": {
+ "version": "8.5.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/cucumber/-/cucumber-8.5.0.tgz",
+ "integrity": "sha512-Ac64lYhgzRxFGf3MnlysMG0zmHa18HfuOU1BA5RfXsrHeWtZx525zH4FsqtRW2AnlYHGu3pQpww5zyQKWi1WDA==",
+ "dev": true,
+ "requires": {
+ "@cspotcode/source-map-support": "^0.8.0",
+ "@cucumber/ci-environment": "9.0.4",
+ "@cucumber/cucumber-expressions": "16.0.0",
+ "@cucumber/gherkin": "24.0.0",
+ "@cucumber/gherkin-streams": "5.0.1",
+ "@cucumber/gherkin-utils": "8.0.0",
+ "@cucumber/html-formatter": "19.2.0",
+ "@cucumber/message-streams": "4.0.1",
+ "@cucumber/messages": "19.1.2",
+ "@cucumber/tag-expressions": "4.1.0",
+ "assertion-error-formatter": "^3.0.0",
+ "capital-case": "^1.0.4",
+ "chalk": "^4.1.2",
+ "cli-table3": "0.6.2",
+ "commander": "^9.0.0",
+ "duration": "^0.2.2",
+ "durations": "^3.4.2",
+ "figures": "^3.2.0",
+ "glob": "^7.1.6",
+ "has-ansi": "^4.0.1",
+ "indent-string": "^4.0.0",
+ "is-installed-globally": "^0.4.0",
+ "is-stream": "^2.0.0",
+ "knuth-shuffle-seeded": "^1.0.6",
+ "lodash.merge": "^4.6.2",
+ "lodash.mergewith": "^4.6.2",
+ "mz": "^2.7.0",
+ "progress": "^2.0.3",
+ "resolve-pkg": "^2.0.0",
+ "semver": "7.3.7",
+ "stack-chain": "^2.0.0",
+ "string-argv": "^0.3.1",
+ "strip-ansi": "6.0.1",
+ "supports-color": "^8.1.1",
+ "tmp": "^0.2.1",
+ "util-arity": "^1.1.0",
+ "verror": "^1.10.0",
+ "yup": "^0.32.11"
+ }
+ },
+ "@cucumber/cucumber-expressions": {
+ "version": "16.0.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/cucumber-expressions/-/cucumber-expressions-16.0.0.tgz",
+ "integrity": "sha512-HTh+Pg7oQ5aLuCkSbD2Q6jBaE40M3R/XaLEz+UqD5d9dZRu6P38W4LTooV5bV6dZgBunlMLK8+6ug2ziYvRddw==",
+ "dev": true,
+ "requires": {
+ "regexp-match-indices": "1.0.2"
+ }
+ },
+ "@cucumber/gherkin": {
+ "version": "24.0.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/gherkin/-/gherkin-24.0.0.tgz",
+ "integrity": "sha512-b7OsnvX1B8myDAKMc+RAiUX9bzgtNdjGsiMj10O13xu2HBWIOQ19EqBJ4xLO5CFG/lGk1J/+L0lANQVowxLVBg==",
+ "dev": true,
+ "requires": {
+ "@cucumber/messages": "^19.0.0"
+ }
+ },
+ "@cucumber/gherkin-streams": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/@cucumber/gherkin-streams/-/gherkin-streams-5.0.1.tgz",
+ "integrity": "sha512-/7VkIE/ASxIP/jd4Crlp4JHXqdNFxPGQokqWqsaCCiqBiu5qHoKMxcWNlp9njVL/n9yN4S08OmY3ZR8uC5x74Q==",
+ "dev": true,
+ "requires": {
+ "commander": "9.1.0",
+ "source-map-support": "0.5.21"
+ },
+ "dependencies": {
+ "commander": {
+ "version": "9.1.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.1.0.tgz",
+ "integrity": "sha512-i0/MaqBtdbnJ4XQs4Pmyb+oFQl+q0lsAmokVUH92SlSw4fkeAcG3bVon+Qt7hmtF+u3Het6o4VgrcY3qAoEB6w==",
+ "dev": true
+ }
+ }
+ },
+ "@cucumber/gherkin-utils": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/gherkin-utils/-/gherkin-utils-8.0.0.tgz",
+ "integrity": "sha512-8uIZInEe3cO1cASmy3BA0PbVFUI+xWBnZAxmICbVOPsZaMB85MtESZLafzErgfRQPsHf6uYbVagP7MIjNPM5Jw==",
+ "dev": true,
+ "requires": {
+ "@cucumber/messages": "^19.0.0",
+ "@teppeis/multimaps": "2.0.0",
+ "commander": "9.3.0"
+ },
+ "dependencies": {
+ "commander": {
+ "version": "9.3.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.3.0.tgz",
+ "integrity": "sha512-hv95iU5uXPbK83mjrJKuZyFM/LBAoCV/XhVGkS5Je6tl7sxr6A0ITMw5WoRV46/UaJ46Nllm3Xt7IaJhXTIkzw==",
+ "dev": true
+ }
+ }
+ },
+ "@cucumber/html-formatter": {
+ "version": "19.2.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/html-formatter/-/html-formatter-19.2.0.tgz",
+ "integrity": "sha512-qGms4588jmVF/G3fTbgZvxn6OQw9GaTFV007nZZ9/10M9DfrgRqjFjVxVI9TPV63xOLPicEVoqsKZtcECbdMSA==",
+ "dev": true,
+ "requires": {}
+ },
+ "@cucumber/message-streams": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/@cucumber/message-streams/-/message-streams-4.0.1.tgz",
+ "integrity": "sha512-Kxap9uP5jD8tHUZVjTWgzxemi/0uOsbGjd4LBOSxcJoOCRbESFwemUzilJuzNTB8pcTQUh8D5oudUyxfkJOKmA==",
+ "dev": true,
+ "requires": {}
+ },
+ "@cucumber/messages": {
+ "version": "19.1.2",
+ "resolved": "https://registry.npmjs.org/@cucumber/messages/-/messages-19.1.2.tgz",
+ "integrity": "sha512-vhWkNmQco+7tk/DWqpN0/R9KTNvsKsXVfZ7IsJs+dEeWmTuRztklHq8lJalwMSQBl71+2/KqGHzOO4BMTC9wIQ==",
+ "dev": true,
+ "requires": {
+ "@types/uuid": "8.3.4",
+ "class-transformer": "0.5.1",
+ "reflect-metadata": "0.1.13",
+ "uuid": "8.3.2"
+ },
+ "dependencies": {
+ "uuid": {
+ "version": "8.3.2",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
+ "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
+ "dev": true
+ }
+ }
+ },
+ "@cucumber/tag-expressions": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/@cucumber/tag-expressions/-/tag-expressions-4.1.0.tgz",
+ "integrity": "sha512-chTnjxV3vryL75N90wJIMdMafXmZoO2JgNJLYpsfcALL2/IQrRiny3vM9DgD5RDCSt1LNloMtb7rGey9YWxCsA==",
+ "dev": true
+ },
+ "@jridgewell/resolve-uri": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz",
+ "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==",
+ "dev": true
+ },
+ "@jridgewell/sourcemap-codec": {
+ "version": "1.4.15",
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz",
+ "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==",
+ "dev": true
+ },
+ "@jridgewell/trace-mapping": {
+ "version": "0.3.9",
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz",
+ "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==",
+ "dev": true,
+ "requires": {
+ "@jridgewell/resolve-uri": "^3.0.3",
+ "@jridgewell/sourcemap-codec": "^1.4.10"
+ }
+ },
+ "@teppeis/multimaps": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/@teppeis/multimaps/-/multimaps-2.0.0.tgz",
+ "integrity": "sha512-TL1adzq1HdxUf9WYduLcQ/DNGYiz71U31QRgbnr0Ef1cPyOUOsBojxHVWpFeOSUucB6Lrs0LxFRA14ntgtkc9w==",
+ "dev": true
+ },
+ "@types/lodash": {
+ "version": "4.14.195",
+ "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.195.tgz",
+ "integrity": "sha512-Hwx9EUgdwf2GLarOjQp5ZH8ZmblzcbTBC2wtQWNKARBSxM9ezRIAUpeDTgoQRAFB0+8CNWXVA9+MaSOzOF3nPg==",
+ "dev": true
+ },
+ "@types/uuid": {
+ "version": "8.3.4",
+ "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.4.tgz",
+ "integrity": "sha512-c/I8ZRb51j+pYGAu5CrFMRxqZ2ke4y2grEBO5AUjgSkSk+qT2Ea+OdWElz/OiMf5MNpn2b17kuVBwZLQJXzihw==",
+ "dev": true
+ },
+ "ajv": {
+ "version": "6.12.6",
+ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+ "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+ "dev": true,
+ "requires": {
+ "fast-deep-equal": "^3.1.1",
+ "fast-json-stable-stringify": "^2.0.0",
+ "json-schema-traverse": "^0.4.1",
+ "uri-js": "^4.2.2"
+ }
+ },
+ "ansi-regex": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz",
+ "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==",
+ "dev": true
+ },
+ "ansi-styles": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+ "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+ "dev": true,
+ "requires": {
+ "color-convert": "^2.0.1"
+ }
+ },
+ "any-promise": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
+ "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==",
+ "dev": true
+ },
+ "asn1": {
+ "version": "0.2.6",
+ "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz",
+ "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==",
+ "dev": true,
+ "requires": {
+ "safer-buffer": "~2.1.0"
+ }
+ },
+ "assert-plus": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz",
+ "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==",
+ "dev": true
+ },
+ "assertion-error": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz",
+ "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==",
+ "dev": true
+ },
+ "assertion-error-formatter": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/assertion-error-formatter/-/assertion-error-formatter-3.0.0.tgz",
+ "integrity": "sha512-6YyAVLrEze0kQ7CmJfUgrLHb+Y7XghmL2Ie7ijVa2Y9ynP3LV+VDiwFk62Dn0qtqbmY0BT0ss6p1xxpiF2PYbQ==",
+ "dev": true,
+ "requires": {
+ "diff": "^4.0.1",
+ "pad-right": "^0.2.2",
+ "repeat-string": "^1.6.1"
+ }
+ },
+ "asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+ "dev": true
+ },
+ "aws-sign2": {
+ "version": "0.7.0",
+ "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz",
+ "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==",
+ "dev": true
+ },
+ "aws4": {
+ "version": "1.12.0",
+ "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.12.0.tgz",
+ "integrity": "sha512-NmWvPnx0F1SfrQbYwOi7OeaNGokp9XhzNioJ/CSBs8Qa4vxug81mhJEAVZwxXuBmYB5KDRfMq/F3RR0BIU7sWg==",
+ "dev": true
+ },
+ "axios": {
+ "version": "1.6.8",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz",
+ "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==",
+ "dev": true,
+ "requires": {
+ "follow-redirects": "^1.15.6",
+ "form-data": "^4.0.0",
+ "proxy-from-env": "^1.1.0"
+ }
+ },
+ "balanced-match": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+ "dev": true
+ },
+ "base64-js": {
+ "version": "1.5.1",
+ "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+ "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+ "dev": true
+ },
+ "bcrypt-pbkdf": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz",
+ "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==",
+ "dev": true,
+ "requires": {
+ "tweetnacl": "^0.14.3"
+ }
+ },
+ "bl": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz",
+ "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==",
+ "dev": true,
+ "requires": {
+ "buffer": "^5.5.0",
+ "inherits": "^2.0.4",
+ "readable-stream": "^3.4.0"
+ }
+ },
+ "brace-expansion": {
+ "version": "1.1.11",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
+ "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+ "dev": true,
+ "requires": {
+ "balanced-match": "^1.0.0",
+ "concat-map": "0.0.1"
+ }
+ },
+ "buffer": {
+ "version": "5.7.1",
+ "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz",
+ "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
+ "dev": true,
+ "requires": {
+ "base64-js": "^1.3.1",
+ "ieee754": "^1.1.13"
+ }
+ },
+ "buffer-from": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
+ "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
+ "dev": true
+ },
+ "capital-case": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/capital-case/-/capital-case-1.0.4.tgz",
+ "integrity": "sha512-ds37W8CytHgwnhGGTi88pcPyR15qoNkOpYwmMMfnWqqWgESapLqvDx6huFjQ5vqWSn2Z06173XNA7LtMOeUh1A==",
+ "dev": true,
+ "requires": {
+ "no-case": "^3.0.4",
+ "tslib": "^2.0.3",
+ "upper-case-first": "^2.0.2"
+ }
+ },
+ "caseless": {
+ "version": "0.12.0",
+ "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
+ "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==",
+ "dev": true
+ },
+ "chai": {
+ "version": "4.3.6",
+ "resolved": "https://registry.npmjs.org/chai/-/chai-4.3.6.tgz",
+ "integrity": "sha512-bbcp3YfHCUzMOvKqsztczerVgBKSsEijCySNlHHbX3VG1nskvqjz5Rfso1gGwD6w6oOV3eI60pKuMOV5MV7p3Q==",
+ "dev": true,
+ "requires": {
+ "assertion-error": "^1.1.0",
+ "check-error": "^1.0.2",
+ "deep-eql": "^3.0.1",
+ "get-func-name": "^2.0.0",
+ "loupe": "^2.3.1",
+ "pathval": "^1.1.1",
+ "type-detect": "^4.0.5"
+ }
+ },
+ "chai-arrays": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/chai-arrays/-/chai-arrays-2.2.0.tgz",
+ "integrity": "sha512-4awrdGI2EH8owJ9I58PXwG4N56/FiM8bsn4CVSNEgr4GKAM6Kq5JPVApUbhUBjDakbZNuRvV7quRSC38PWq/tg==",
+ "dev": true
+ },
+ "chai-things": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/chai-things/-/chai-things-0.2.0.tgz",
+ "integrity": "sha512-6ns0SU21xdRCoEXVKH3HGbwnsgfVMXQ+sU5V8PI9rfxaITos8lss1vUxbF1FAcJKjfqmmmLVlr/z3sLes00w+A==",
+ "dev": true
+ },
+ "chalk": {
+ "version": "4.1.2",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+ "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+ "dev": true,
+ "requires": {
+ "ansi-styles": "^4.1.0",
+ "supports-color": "^7.1.0"
+ },
+ "dependencies": {
+ "supports-color": {
+ "version": "7.2.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+ "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+ "dev": true,
+ "requires": {
+ "has-flag": "^4.0.0"
+ }
+ }
+ }
+ },
+ "check-error": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz",
+ "integrity": "sha512-BrgHpW9NURQgzoNyjfq0Wu6VFO6D7IZEmJNdtgNqpzGG8RuNFHt2jQxWlAs4HMe119chBnv+34syEZtc6IhLtA==",
+ "dev": true
+ },
+ "chownr": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz",
+ "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==",
+ "dev": true
+ },
+ "class-transformer": {
+ "version": "0.5.1",
+ "resolved": "https://registry.npmjs.org/class-transformer/-/class-transformer-0.5.1.tgz",
+ "integrity": "sha512-SQa1Ws6hUbfC98vKGxZH3KFY0Y1lm5Zm0SY8XX9zbK7FJCyVEac3ATW0RIpwzW+oOfmHE5PMPufDG9hCfoEOMw==",
+ "dev": true
+ },
+ "cli-table3": {
+ "version": "0.6.2",
+ "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.2.tgz",
+ "integrity": "sha512-QyavHCaIC80cMivimWu4aWHilIpiDpfm3hGmqAmXVL1UsnbLuBSMd21hTX6VY4ZSDSM73ESLeF8TOYId3rBTbw==",
+ "dev": true,
+ "requires": {
+ "@colors/colors": "1.5.0",
+ "string-width": "^4.2.0"
+ }
+ },
+ "clickhouse": {
+ "version": "2.6.0",
+ "resolved": "https://registry.npmjs.org/clickhouse/-/clickhouse-2.6.0.tgz",
+ "integrity": "sha512-HC5OV99GJOup4qZsTuWWPpXlj+847Z0OeygDU2x22rNYost0V/vWapzFWYZdV/5iRbGMrhFQPOyQEzmGvoaWRQ==",
+ "dev": true,
+ "requires": {
+ "JSONStream": "1.3.4",
+ "lodash": "4.17.21",
+ "querystring": "0.2.0",
+ "request": "2.88.0",
+ "stream2asynciter": "1.0.3",
+ "through": "2.3.8",
+ "tsv": "0.2.0",
+ "uuid": "3.4.0"
+ }
+ },
+ "color-convert": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+ "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+ "dev": true,
+ "requires": {
+ "color-name": "~1.1.4"
+ }
+ },
+ "color-name": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+ "dev": true
+ },
+ "combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "dev": true,
+ "requires": {
+ "delayed-stream": "~1.0.0"
+ }
+ },
+ "commander": {
+ "version": "9.5.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.5.0.tgz",
+ "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==",
+ "dev": true
+ },
+ "concat-map": {
+ "version": "0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+ "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+ "dev": true
+ },
+ "core-util-is": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
+ "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==",
+ "dev": true
+ },
+ "d": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz",
+ "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==",
+ "dev": true,
+ "requires": {
+ "es5-ext": "^0.10.50",
+ "type": "^1.0.1"
+ }
+ },
+ "dashdash": {
+ "version": "1.14.1",
+ "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz",
+ "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==",
+ "dev": true,
+ "requires": {
+ "assert-plus": "^1.0.0"
+ }
+ },
+ "debug": {
+ "version": "4.3.4",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
+ "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
+ "dev": true,
+ "requires": {
+ "ms": "2.1.2"
+ }
+ },
+ "deep-eql": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz",
+ "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==",
+ "dev": true,
+ "requires": {
+ "type-detect": "^4.0.0"
+ }
+ },
+ "delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+ "dev": true
+ },
+ "diff": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz",
+ "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==",
+ "dev": true
+ },
+ "docker-modem": {
+ "version": "3.0.8",
+ "resolved": "https://registry.npmjs.org/docker-modem/-/docker-modem-3.0.8.tgz",
+ "integrity": "sha512-f0ReSURdM3pcKPNS30mxOHSbaFLcknGmQjwSfmbcdOw1XWKXVhukM3NJHhr7NpY9BIyyWQb0EBo3KQvvuU5egQ==",
+ "dev": true,
+ "requires": {
+ "debug": "^4.1.1",
+ "readable-stream": "^3.5.0",
+ "split-ca": "^1.0.1",
+ "ssh2": "^1.11.0"
+ }
+ },
+ "dockerode": {
+ "version": "3.3.2",
+ "resolved": "https://registry.npmjs.org/dockerode/-/dockerode-3.3.2.tgz",
+ "integrity": "sha512-oXN+1XVH2TeyE0Jj9Ci6Fim8ZIDxyqeJrkx9qhEOaRiA+nhLihKfd3M2L+Aqrj5C2ObPw8RVN2zPWvvk0x2dwg==",
+ "dev": true,
+ "requires": {
+ "docker-modem": "^3.0.0",
+ "tar-fs": "~2.0.1"
+ }
+ },
+ "duration": {
+ "version": "0.2.2",
+ "resolved": "https://registry.npmjs.org/duration/-/duration-0.2.2.tgz",
+ "integrity": "sha512-06kgtea+bGreF5eKYgI/36A6pLXggY7oR4p1pq4SmdFBn1ReOL5D8RhG64VrqfTTKNucqqtBAwEj8aB88mcqrg==",
+ "dev": true,
+ "requires": {
+ "d": "1",
+ "es5-ext": "~0.10.46"
+ }
+ },
+ "durations": {
+ "version": "3.4.2",
+ "resolved": "https://registry.npmjs.org/durations/-/durations-3.4.2.tgz",
+ "integrity": "sha512-V/lf7y33dGaypZZetVI1eu7BmvkbC4dItq12OElLRpKuaU5JxQstV2zHwLv8P7cNbQ+KL1WD80zMCTx5dNC4dg==",
+ "dev": true
+ },
+ "ecc-jsbn": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz",
+ "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==",
+ "dev": true,
+ "requires": {
+ "jsbn": "~0.1.0",
+ "safer-buffer": "^2.1.0"
+ }
+ },
+ "emoji-regex": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
+ "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
+ "dev": true
+ },
+ "end-of-stream": {
+ "version": "1.4.4",
+ "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
+ "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
+ "dev": true,
+ "requires": {
+ "once": "^1.4.0"
+ }
+ },
+ "es5-ext": {
+ "version": "0.10.62",
+ "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.62.tgz",
+ "integrity": "sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==",
+ "dev": true,
+ "requires": {
+ "es6-iterator": "^2.0.3",
+ "es6-symbol": "^3.1.3",
+ "next-tick": "^1.1.0"
+ }
+ },
+ "es6-iterator": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz",
+ "integrity": "sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==",
+ "dev": true,
+ "requires": {
+ "d": "1",
+ "es5-ext": "^0.10.35",
+ "es6-symbol": "^3.1.1"
+ }
+ },
+ "es6-symbol": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz",
+ "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==",
+ "dev": true,
+ "requires": {
+ "d": "^1.0.1",
+ "ext": "^1.1.2"
+ }
+ },
+ "escape-string-regexp": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
+ "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==",
+ "dev": true
+ },
+ "ext": {
+ "version": "1.7.0",
+ "resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz",
+ "integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==",
+ "dev": true,
+ "requires": {
+ "type": "^2.7.2"
+ },
+ "dependencies": {
+ "type": {
+ "version": "2.7.2",
+ "resolved": "https://registry.npmjs.org/type/-/type-2.7.2.tgz",
+ "integrity": "sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==",
+ "dev": true
+ }
+ }
+ },
+ "extend": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz",
+ "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==",
+ "dev": true
+ },
+ "extsprintf": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.4.1.tgz",
+ "integrity": "sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA==",
+ "dev": true
+ },
+ "fast-deep-equal": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
+ "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
+ "dev": true
+ },
+ "fast-json-stable-stringify": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+ "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+ "dev": true
+ },
+ "figures": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz",
+ "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==",
+ "dev": true,
+ "requires": {
+ "escape-string-regexp": "^1.0.5"
+ }
+ },
+ "follow-redirects": {
+ "version": "1.15.6",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz",
+ "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==",
+ "dev": true
+ },
+ "forever-agent": {
+ "version": "0.6.1",
+ "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz",
+ "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==",
+ "dev": true
+ },
+ "form-data": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+ "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+ "dev": true,
+ "requires": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "mime-types": "^2.1.12"
+ }
+ },
+ "fs-constants": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz",
+ "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==",
+ "dev": true
+ },
+ "fs.realpath": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
+ "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==",
+ "dev": true
+ },
+ "get-func-name": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz",
+ "integrity": "sha512-Hm0ixYtaSZ/V7C8FJrtZIuBBI+iSgL+1Aq82zSu8VQNB4S3Gk8e7Qs3VwBDJAhmRZcFqkl3tQu36g/Foh5I5ig==",
+ "dev": true
+ },
+ "getpass": {
+ "version": "0.1.7",
+ "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz",
+ "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==",
+ "dev": true,
+ "requires": {
+ "assert-plus": "^1.0.0"
+ }
+ },
+ "glob": {
+ "version": "7.2.3",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
+ "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==",
+ "dev": true,
+ "requires": {
+ "fs.realpath": "^1.0.0",
+ "inflight": "^1.0.4",
+ "inherits": "2",
+ "minimatch": "^3.1.1",
+ "once": "^1.3.0",
+ "path-is-absolute": "^1.0.0"
+ }
+ },
+ "global-dirs": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-3.0.1.tgz",
+ "integrity": "sha512-NBcGGFbBA9s1VzD41QXDG+3++t9Mn5t1FpLdhESY6oKY4gYTFpX4wO3sqGUa0Srjtbfj3szX0RnemmrVRUdULA==",
+ "dev": true,
+ "requires": {
+ "ini": "2.0.0"
+ }
+ },
+ "har-schema": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz",
+ "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==",
+ "dev": true
+ },
+ "har-validator": {
+ "version": "5.1.5",
+ "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz",
+ "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==",
+ "dev": true,
+ "requires": {
+ "ajv": "^6.12.3",
+ "har-schema": "^2.0.0"
+ }
+ },
+ "has-ansi": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-4.0.1.tgz",
+ "integrity": "sha512-Qr4RtTm30xvEdqUXbSBVWDu+PrTokJOwe/FU+VdfJPk+MXAPoeOzKpRyrDTnZIJwAkQ4oBLTU53nu0HrkF/Z2A==",
+ "dev": true,
+ "requires": {
+ "ansi-regex": "^4.1.0"
+ }
+ },
+ "has-flag": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+ "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+ "dev": true
+ },
+ "http-signature": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz",
+ "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==",
+ "dev": true,
+ "requires": {
+ "assert-plus": "^1.0.0",
+ "jsprim": "^1.2.2",
+ "sshpk": "^1.7.0"
+ }
+ },
+ "ieee754": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
+ "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
+ "dev": true
+ },
+ "indent-string": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz",
+ "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==",
+ "dev": true
+ },
+ "inflight": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+ "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
+ "dev": true,
+ "requires": {
+ "once": "^1.3.0",
+ "wrappy": "1"
+ }
+ },
+ "inherits": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
+ "dev": true
+ },
+ "ini": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/ini/-/ini-2.0.0.tgz",
+ "integrity": "sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==",
+ "dev": true
+ },
+ "is-fullwidth-code-point": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
+ "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
+ "dev": true
+ },
+ "is-installed-globally": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.4.0.tgz",
+ "integrity": "sha512-iwGqO3J21aaSkC7jWnHP/difazwS7SFeIqxv6wEtLU8Y5KlzFTjyqcSIT0d8s4+dDhKytsk9PJZ2BkS5eZwQRQ==",
+ "dev": true,
+ "requires": {
+ "global-dirs": "^3.0.0",
+ "is-path-inside": "^3.0.2"
+ }
+ },
+ "is-path-inside": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
+ "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
+ "dev": true
+ },
+ "is-stream": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz",
+ "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==",
+ "dev": true
+ },
+ "is-typedarray": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
+ "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==",
+ "dev": true
+ },
+ "isstream": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
+ "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==",
+ "dev": true
+ },
+ "jsbn": {
+ "version": "0.1.1",
+ "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
+ "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==",
+ "dev": true
+ },
+ "json-schema": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz",
+ "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==",
+ "dev": true
+ },
+ "json-schema-traverse": {
+ "version": "0.4.1",
+ "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+ "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+ "dev": true
+ },
+ "json-stringify-safe": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz",
+ "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==",
+ "dev": true
+ },
+ "jsonparse": {
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz",
+ "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==",
+ "dev": true
+ },
+ "JSONStream": {
+ "version": "1.3.4",
+ "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.4.tgz",
+ "integrity": "sha512-Y7vfi3I5oMOYIr+WxV8NZxDSwcbNgzdKYsTNInmycOq9bUYwGg9ryu57Wg5NLmCjqdFPNUmpMBo3kSJN9tCbXg==",
+ "dev": true,
+ "requires": {
+ "jsonparse": "^1.2.0",
+ "through": ">=2.2.7 <3"
+ }
+ },
+ "jsprim": {
+ "version": "1.4.2",
+ "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz",
+ "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==",
+ "dev": true,
+ "requires": {
+ "assert-plus": "1.0.0",
+ "extsprintf": "1.3.0",
+ "json-schema": "0.4.0",
+ "verror": "1.10.0"
+ },
+ "dependencies": {
+ "extsprintf": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
+ "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==",
+ "dev": true
+ },
+ "verror": {
+ "version": "1.10.0",
+ "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz",
+ "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==",
+ "dev": true,
+ "requires": {
+ "assert-plus": "^1.0.0",
+ "core-util-is": "1.0.2",
+ "extsprintf": "^1.2.0"
+ },
+ "dependencies": {
+ "extsprintf": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.4.1.tgz",
+ "integrity": "sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA==",
+ "dev": true
+ }
+ }
+ }
+ }
+ },
+ "knuth-shuffle-seeded": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/knuth-shuffle-seeded/-/knuth-shuffle-seeded-1.0.6.tgz",
+ "integrity": "sha512-9pFH0SplrfyKyojCLxZfMcvkhf5hH0d+UwR9nTVJ/DDQJGuzcXjTwB7TP7sDfehSudlGGaOLblmEWqv04ERVWg==",
+ "dev": true,
+ "requires": {
+ "seed-random": "~2.2.0"
+ }
+ },
+ "lodash": {
+ "version": "4.17.21",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
+ "dev": true
+ },
+ "lodash-es": {
+ "version": "4.17.21",
+ "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz",
+ "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==",
+ "dev": true
+ },
+ "lodash.merge": {
+ "version": "4.6.2",
+ "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
+ "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
+ "dev": true
+ },
+ "lodash.mergewith": {
+ "version": "4.6.2",
+ "resolved": "https://registry.npmjs.org/lodash.mergewith/-/lodash.mergewith-4.6.2.tgz",
+ "integrity": "sha512-GK3g5RPZWTRSeLSpgP8Xhra+pnjBC56q9FZYe1d5RN3TJ35dbkGy3YqBSMbyCrlbi+CM9Z3Jk5yTL7RCsqboyQ==",
+ "dev": true
+ },
+ "loupe": {
+ "version": "2.3.6",
+ "resolved": "https://registry.npmjs.org/loupe/-/loupe-2.3.6.tgz",
+ "integrity": "sha512-RaPMZKiMy8/JruncMU5Bt6na1eftNoo++R4Y+N2FrxkDVTrGvcyzFTsaGif4QTeKESheMGegbhw6iUAq+5A8zA==",
+ "dev": true,
+ "requires": {
+ "get-func-name": "^2.0.0"
+ }
+ },
+ "lower-case": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz",
+ "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==",
+ "dev": true,
+ "requires": {
+ "tslib": "^2.0.3"
+ }
+ },
+ "lru-cache": {
+ "version": "6.0.0",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
+ "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
+ "dev": true,
+ "requires": {
+ "yallist": "^4.0.0"
+ }
+ },
+ "mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "dev": true
+ },
+ "mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "dev": true,
+ "requires": {
+ "mime-db": "1.52.0"
+ }
+ },
+ "minimatch": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+ "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+ "dev": true,
+ "requires": {
+ "brace-expansion": "^1.1.7"
+ }
+ },
+ "mkdirp-classic": {
+ "version": "0.5.3",
+ "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz",
+ "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==",
+ "dev": true
+ },
+ "ms": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
+ "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
+ "dev": true
+ },
+ "mz": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
+ "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==",
+ "dev": true,
+ "requires": {
+ "any-promise": "^1.0.0",
+ "object-assign": "^4.0.1",
+ "thenify-all": "^1.0.0"
+ }
+ },
+ "nan": {
+ "version": "2.17.0",
+ "resolved": "https://registry.npmjs.org/nan/-/nan-2.17.0.tgz",
+ "integrity": "sha512-2ZTgtl0nJsO0KQCjEpxcIr5D+Yv90plTitZt9JBfQvVJDS5seMl3FOvsh3+9CoYWXf/1l5OaZzzF6nDm4cagaQ==",
+ "dev": true,
+ "optional": true
+ },
+ "nanoclone": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/nanoclone/-/nanoclone-0.2.1.tgz",
+ "integrity": "sha512-wynEP02LmIbLpcYw8uBKpcfF6dmg2vcpKqxeH5UcoKEYdExslsdUA4ugFauuaeYdTB76ez6gJW8XAZ6CgkXYxA==",
+ "dev": true
+ },
+ "next-tick": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz",
+ "integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==",
+ "dev": true
+ },
+ "no-case": {
+ "version": "3.0.4",
+ "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz",
+ "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==",
+ "dev": true,
+ "requires": {
+ "lower-case": "^2.0.2",
+ "tslib": "^2.0.3"
+ }
+ },
+ "oauth-sign": {
+ "version": "0.9.0",
+ "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz",
+ "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==",
+ "dev": true
+ },
+ "object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+ "dev": true
+ },
+ "once": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+ "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
+ "dev": true,
+ "requires": {
+ "wrappy": "1"
+ }
+ },
+ "pad-right": {
+ "version": "0.2.2",
+ "resolved": "https://registry.npmjs.org/pad-right/-/pad-right-0.2.2.tgz",
+ "integrity": "sha512-4cy8M95ioIGolCoMmm2cMntGR1lPLEbOMzOKu8bzjuJP6JpzEMQcDHmh7hHLYGgob+nKe1YHFMaG4V59HQa89g==",
+ "dev": true,
+ "requires": {
+ "repeat-string": "^1.5.2"
+ }
+ },
+ "path-is-absolute": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+ "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
+ "dev": true
+ },
+ "pathval": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz",
+ "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==",
+ "dev": true
+ },
+ "performance-now": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
+ "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==",
+ "dev": true
+ },
+ "progress": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz",
+ "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==",
+ "dev": true
+ },
+ "property-expr": {
+ "version": "2.0.5",
+ "resolved": "https://registry.npmjs.org/property-expr/-/property-expr-2.0.5.tgz",
+ "integrity": "sha512-IJUkICM5dP5znhCckHSv30Q4b5/JA5enCtkRHYaOVOAocnH/1BQEYTC5NMfT3AVl/iXKdr3aqQbQn9DxyWknwA==",
+ "dev": true
+ },
+ "proxy-from-env": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+ "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+ "dev": true
+ },
+ "psl": {
+ "version": "1.9.0",
+ "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz",
+ "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==",
+ "dev": true
+ },
+ "pump": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
+ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
+ "dev": true,
+ "requires": {
+ "end-of-stream": "^1.1.0",
+ "once": "^1.3.1"
+ }
+ },
+ "punycode": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz",
+ "integrity": "sha512-jmYNElW7yvO7TV33CjSmvSiE2yco3bV2czu/OzDKdMNVZQWfxCblURLhf+47syQRBntjfLdd/H0egrzIG+oaFQ==",
+ "dev": true
+ },
+ "qs": {
+ "version": "6.5.3",
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.3.tgz",
+ "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==",
+ "dev": true
+ },
+ "querystring": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz",
+ "integrity": "sha512-X/xY82scca2tau62i9mDyU9K+I+djTMUsvwf7xnUX5GLvVzgJybOJf4Y6o9Zx3oJK/LSXg5tTZBjwzqVPaPO2g==",
+ "dev": true
+ },
+ "readable-stream": {
+ "version": "3.6.2",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz",
+ "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==",
+ "dev": true,
+ "requires": {
+ "inherits": "^2.0.3",
+ "string_decoder": "^1.1.1",
+ "util-deprecate": "^1.0.1"
+ }
+ },
+ "reflect-metadata": {
+ "version": "0.1.13",
+ "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz",
+ "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==",
+ "dev": true
+ },
+ "regenerator-runtime": {
+ "version": "0.13.11",
+ "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz",
+ "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==",
+ "dev": true
+ },
+ "regexp-match-indices": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/regexp-match-indices/-/regexp-match-indices-1.0.2.tgz",
+ "integrity": "sha512-DwZuAkt8NF5mKwGGER1EGh2PRqyvhRhhLviH+R8y8dIuaQROlUfXjt4s9ZTXstIsSkptf06BSvwcEmmfheJJWQ==",
+ "dev": true,
+ "requires": {
+ "regexp-tree": "^0.1.11"
+ }
+ },
+ "regexp-tree": {
+ "version": "0.1.27",
+ "resolved": "https://registry.npmjs.org/regexp-tree/-/regexp-tree-0.1.27.tgz",
+ "integrity": "sha512-iETxpjK6YoRWJG5o6hXLwvjYAoW+FEZn9os0PD/b6AP6xQwsa/Y7lCVgIixBbUPMfhu+i2LtdeAqVTgGlQarfA==",
+ "dev": true
+ },
+ "repeat-string": {
+ "version": "1.6.1",
+ "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz",
+ "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==",
+ "dev": true
+ },
+ "request": {
+ "version": "2.88.0",
+ "resolved": "https://registry.npmjs.org/request/-/request-2.88.0.tgz",
+ "integrity": "sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==",
+ "dev": true,
+ "requires": {
+ "aws-sign2": "~0.7.0",
+ "aws4": "^1.8.0",
+ "caseless": "~0.12.0",
+ "combined-stream": "~1.0.6",
+ "extend": "~3.0.2",
+ "forever-agent": "~0.6.1",
+ "form-data": "~2.3.2",
+ "har-validator": "~5.1.0",
+ "http-signature": "~1.2.0",
+ "is-typedarray": "~1.0.0",
+ "isstream": "~0.1.2",
+ "json-stringify-safe": "~5.0.1",
+ "mime-types": "~2.1.19",
+ "oauth-sign": "~0.9.0",
+ "performance-now": "^2.1.0",
+ "qs": "~6.5.2",
+ "safe-buffer": "^5.1.2",
+ "tough-cookie": "~2.4.3",
+ "tunnel-agent": "^0.6.0",
+ "uuid": "^3.3.2"
+ },
+ "dependencies": {
+ "form-data": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz",
+ "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==",
+ "dev": true,
+ "requires": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.6",
+ "mime-types": "^2.1.12"
+ }
+ }
+ }
+ },
+ "resolve-from": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz",
+ "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==",
+ "dev": true
+ },
+ "resolve-pkg": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/resolve-pkg/-/resolve-pkg-2.0.0.tgz",
+ "integrity": "sha512-+1lzwXehGCXSeryaISr6WujZzowloigEofRB+dj75y9RRa/obVcYgbHJd53tdYw8pvZj8GojXaaENws8Ktw/hQ==",
+ "dev": true,
+ "requires": {
+ "resolve-from": "^5.0.0"
+ }
+ },
+ "rimraf": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
+ "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
+ "dev": true,
+ "requires": {
+ "glob": "^7.1.3"
+ }
+ },
+ "safe-buffer": {
+ "version": "5.2.1",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+ "dev": true
+ },
+ "safer-buffer": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
+ "dev": true
+ },
+ "seed-random": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/seed-random/-/seed-random-2.2.0.tgz",
+ "integrity": "sha512-34EQV6AAHQGhoc0tn/96a9Fsi6v2xdqe/dMUwljGRaFOzR3EgRmECvD0O8vi8X+/uQ50LGHfkNu/Eue5TPKZkQ==",
+ "dev": true
+ },
+ "semver": {
+ "version": "7.3.7",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz",
+ "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==",
+ "dev": true,
+ "requires": {
+ "lru-cache": "^6.0.0"
+ }
+ },
+ "source-map": {
+ "version": "0.6.1",
+ "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
+ "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
+ "dev": true
+ },
+ "source-map-support": {
+ "version": "0.5.21",
+ "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz",
+ "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==",
+ "dev": true,
+ "requires": {
+ "buffer-from": "^1.0.0",
+ "source-map": "^0.6.0"
+ }
+ },
+ "split-ca": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/split-ca/-/split-ca-1.0.1.tgz",
+ "integrity": "sha512-Q5thBSxp5t8WPTTJQS59LrGqOZqOsrhDGDVm8azCqIBjSBd7nd9o2PM+mDulQQkh8h//4U6hFZnc/mul8t5pWQ==",
+ "dev": true
+ },
+ "ssh2": {
+ "version": "1.14.0",
+ "resolved": "https://registry.npmjs.org/ssh2/-/ssh2-1.14.0.tgz",
+ "integrity": "sha512-AqzD1UCqit8tbOKoj6ztDDi1ffJZ2rV2SwlgrVVrHPkV5vWqGJOVp5pmtj18PunkPJAuKQsnInyKV+/Nb2bUnA==",
+ "dev": true,
+ "requires": {
+ "asn1": "^0.2.6",
+ "bcrypt-pbkdf": "^1.0.2",
+ "cpu-features": "~0.0.8",
+ "nan": "^2.17.0"
+ }
+ },
+ "sshpk": {
+ "version": "1.18.0",
+ "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.18.0.tgz",
+ "integrity": "sha512-2p2KJZTSqQ/I3+HX42EpYOa2l3f8Erv8MWKsy2I9uf4wA7yFIkXRffYdsx86y6z4vHtV8u7g+pPlr8/4ouAxsQ==",
+ "dev": true,
+ "requires": {
+ "asn1": "~0.2.3",
+ "assert-plus": "^1.0.0",
+ "bcrypt-pbkdf": "^1.0.0",
+ "dashdash": "^1.12.0",
+ "ecc-jsbn": "~0.1.1",
+ "getpass": "^0.1.1",
+ "jsbn": "~0.1.0",
+ "safer-buffer": "^2.0.2",
+ "tweetnacl": "~0.14.0"
+ }
+ },
+ "stack-chain": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/stack-chain/-/stack-chain-2.0.0.tgz",
+ "integrity": "sha512-GGrHXePi305aW7XQweYZZwiRwR7Js3MWoK/EHzzB9ROdc75nCnjSJVi21rdAGxFl+yCx2L2qdfl5y7NO4lTyqg==",
+ "dev": true
+ },
+ "stream2asynciter": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/stream2asynciter/-/stream2asynciter-1.0.3.tgz",
+ "integrity": "sha512-9/dEZW+LQjuW6ub5hmWi4n9Pn8W8qA8k7NAE1isecesA164e73xTdy1CJ3S9o9YS+O21HuiK7T+4uS7FgKDy4w==",
+ "dev": true
+ },
+ "string_decoder": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
+ "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
+ "dev": true,
+ "requires": {
+ "safe-buffer": "~5.2.0"
+ }
+ },
+ "string-argv": {
+ "version": "0.3.2",
+ "resolved": "https://registry.npmjs.org/string-argv/-/string-argv-0.3.2.tgz",
+ "integrity": "sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==",
+ "dev": true
+ },
+ "string-width": {
+ "version": "4.2.3",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
+ "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
+ "dev": true,
+ "requires": {
+ "emoji-regex": "^8.0.0",
+ "is-fullwidth-code-point": "^3.0.0",
+ "strip-ansi": "^6.0.1"
+ }
+ },
+ "strip-ansi": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+ "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+ "dev": true,
+ "requires": {
+ "ansi-regex": "^5.0.1"
+ },
+ "dependencies": {
+ "ansi-regex": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+ "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+ "dev": true
+ }
+ }
+ },
+ "supports-color": {
+ "version": "8.1.1",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz",
+ "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==",
+ "dev": true,
+ "requires": {
+ "has-flag": "^4.0.0"
+ }
+ },
+ "tar-fs": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.0.1.tgz",
+ "integrity": "sha512-6tzWDMeroL87uF/+lin46k+Q+46rAJ0SyPGz7OW7wTgblI273hsBqk2C1j0/xNadNLKDTUL9BukSjB7cwgmlPA==",
+ "dev": true,
+ "requires": {
+ "chownr": "^1.1.1",
+ "mkdirp-classic": "^0.5.2",
+ "pump": "^3.0.0",
+ "tar-stream": "^2.0.0"
+ }
+ },
+ "tar-stream": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz",
+ "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==",
+ "dev": true,
+ "requires": {
+ "bl": "^4.0.3",
+ "end-of-stream": "^1.4.1",
+ "fs-constants": "^1.0.0",
+ "inherits": "^2.0.3",
+ "readable-stream": "^3.1.1"
+ }
+ },
+ "thenify": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
+ "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==",
+ "dev": true,
+ "requires": {
+ "any-promise": "^1.0.0"
+ }
+ },
+ "thenify-all": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz",
+ "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==",
+ "dev": true,
+ "requires": {
+ "thenify": ">= 3.1.0 < 4"
+ }
+ },
+ "through": {
+ "version": "2.3.8",
+ "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
+ "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==",
+ "dev": true
+ },
+ "tmp": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz",
+ "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==",
+ "dev": true,
+ "requires": {
+ "rimraf": "^3.0.0"
+ }
+ },
+ "toposort": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/toposort/-/toposort-2.0.2.tgz",
+ "integrity": "sha512-0a5EOkAUp8D4moMi2W8ZF8jcga7BgZd91O/yabJCFY8az+XSzeGyTKs0Aoo897iV1Nj6guFq8orWDS96z91oGg==",
+ "dev": true
+ },
+ "tough-cookie": {
+ "version": "2.4.3",
+ "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.4.3.tgz",
+ "integrity": "sha512-Q5srk/4vDM54WJsJio3XNn6K2sCG+CQ8G5Wz6bZhRZoAe/+TxjWB/GlFAnYEbkYVlON9FMk/fE3h2RLpPXo4lQ==",
+ "dev": true,
+ "requires": {
+ "psl": "^1.1.24",
+ "punycode": "^1.4.1"
+ }
+ },
+ "tslib": {
+ "version": "2.6.0",
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.0.tgz",
+ "integrity": "sha512-7At1WUettjcSRHXCyYtTselblcHl9PJFFVKiCAy/bY97+BPZXSQ2wbq0P9s8tK2G7dFQfNnlJnPAiArVBVBsfA==",
+ "dev": true
+ },
+ "tsv": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/tsv/-/tsv-0.2.0.tgz",
+ "integrity": "sha512-GG6xbOP85giXXom0dS6z9uyDsxktznjpa1AuDlPrIXDqDnbhjr9Vk6Us8iz6U1nENL4CPS2jZDvIjEdaZsmc4Q==",
+ "dev": true
+ },
+ "tunnel-agent": {
+ "version": "0.6.0",
+ "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
+ "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==",
+ "dev": true,
+ "requires": {
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "tweetnacl": {
+ "version": "0.14.5",
+ "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz",
+ "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==",
+ "dev": true
+ },
+ "type": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz",
+ "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==",
+ "dev": true
+ },
+ "type-detect": {
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz",
+ "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==",
+ "dev": true
+ },
+ "upper-case-first": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/upper-case-first/-/upper-case-first-2.0.2.tgz",
+ "integrity": "sha512-514ppYHBaKwfJRK/pNC6c/OxfGa0obSnAl106u97Ed0I625Nin96KAjttZF6ZL3e1XLtphxnqrOi9iWgm+u+bg==",
+ "dev": true,
+ "requires": {
+ "tslib": "^2.0.3"
+ }
+ },
+ "uri-js": {
+ "version": "4.4.1",
+ "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+ "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+ "dev": true,
+ "requires": {
+ "punycode": "^2.1.0"
+ },
+ "dependencies": {
+ "punycode": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
+ "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
+ "dev": true
+ }
+ }
+ },
+ "util-arity": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/util-arity/-/util-arity-1.1.0.tgz",
+ "integrity": "sha512-kkyIsXKwemfSy8ZEoaIz06ApApnWsk5hQO0vLjZS6UkBiGiW++Jsyb8vSBoc0WKlffGoGs5yYy/j5pp8zckrFA==",
+ "dev": true
+ },
+ "util-deprecate": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
+ "dev": true
+ },
+ "uuid": {
+ "version": "3.4.0",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz",
+ "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==",
+ "dev": true
+ },
+ "verror": {
+ "version": "1.10.1",
+ "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.1.tgz",
+ "integrity": "sha512-veufcmxri4e3XSrT0xwfUR7kguIkaxBeosDg00yDWhk49wdwkSUrvvsm7nc75e1PUyvIeZj6nS8VQRYz2/S4Xg==",
+ "dev": true,
+ "requires": {
+ "assert-plus": "^1.0.0",
+ "core-util-is": "1.0.2",
+ "extsprintf": "^1.2.0"
+ }
+ },
+ "wrappy": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+ "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
+ "dev": true
+ },
+ "yallist": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
+ "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
+ "dev": true
+ },
+ "yup": {
+ "version": "0.32.11",
+ "resolved": "https://registry.npmjs.org/yup/-/yup-0.32.11.tgz",
+ "integrity": "sha512-Z2Fe1bn+eLstG8DRR6FTavGD+MeAwyfmouhHsIUgaADz8jvFKbO/fXc2trJKZg+5EBjh4gGm3iU/t3onKlXHIg==",
+ "dev": true,
+ "requires": {
+ "@babel/runtime": "^7.15.4",
+ "@types/lodash": "^4.14.175",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "nanoclone": "^0.2.1",
+ "property-expr": "^2.0.4",
+ "toposort": "^2.0.2"
+ }
+ }
+ }
+}
diff --git a/test/cucumber/package.json b/test/cucumber/package.json
index d9ab8bd0..01c61fc7 100644
--- a/test/cucumber/package.json
+++ b/test/cucumber/package.json
@@ -30,7 +30,8 @@
"test:single:mpi-mediator": "cucumber-js 'features/single-mode/mpi-mediator.feature'",
"test:single:monitoring": "cucumber-js 'features/single-mode/monitoring.feature'",
"test:cluster:monitoring": "cucumber-js 'features/cluster-mode/monitoring.feature'",
- "test:single:traefik": "cucumber-js 'features/single-mode/reverse-proxy-traefik.feature'"
+ "test:single:traefik": "cucumber-js 'features/single-mode/reverse-proxy-traefik.feature'",
+ "test:single:openfn": "cucumber-js 'features/single-mode/openfn.feature'"
},
"devDependencies": {
"@cucumber/cucumber": "8.5.0",