diff --git a/.github/.trivyignore b/.github/.trivyignore
new file mode 100644
index 000000000..e69de29bb
diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS
new file mode 100644
index 000000000..e2e02646a
--- /dev/null
+++ b/.github/CODEOWNERS
@@ -0,0 +1,2 @@
+# Code owners for the .github directory
+/.github/ @PilotDataPlatform/indoc-devops
diff --git a/.github/workflows/build-and-publish-admin-server.yml b/.github/workflows/build-and-publish-admin-server.yml
index ba4707523..7ab5ce112 100644
--- a/.github/workflows/build-and-publish-admin-server.yml
+++ b/.github/workflows/build-and-publish-admin-server.yml
@@ -14,18 +14,25 @@ jobs:
- name: Extract Branch Name
id: extract_branch
shell: bash
- run: echo "##[set-output name=branch;]$(echo ${GITHUB_REF#refs/heads/})"
+ run: echo "branch=${GITHUB_REF#refs/heads/}" >> $GITHUB_OUTPUT
get-version:
runs-on: ubuntu-20.04
outputs:
app_version: ${{steps.get-version.outputs.app_version}}
steps:
- name: Checkout code
- uses: actions/checkout@v2
+ uses: actions/checkout@v3
- name: Get Version
id: get-version
shell: bash
- run: echo "app_version=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`" >> $GITHUB_OUTPUT
+ run: |
+ BRANCH=${GITHUB_REF#refs/heads/}
+ BASE_VERSION=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`
+ if [ $BRANCH == "patched-version" ]; then
+ echo "app_version=$BASE_VERSION-`git rev-parse --short HEAD`" >> $GITHUB_OUTPUT
+ else
+ echo "app_version=$BASE_VERSION" >> $GITHUB_OUTPUT
+ fi
build-and-push-docker-image:
needs: [extract-branch-name, get-version]
@@ -34,12 +41,12 @@ jobs:
runs-on: ubuntu-20.04
steps:
- name: Checkout code
- uses: actions/checkout@v2
+ uses: actions/checkout@v3
- name: Set up Docker Buildx
id: buildx
- uses: docker/setup-buildx-action@v1
+ uses: docker/setup-buildx-action@v2
- name: Login to Github Packages
- uses: docker/login-action@v1
+ uses: docker/login-action@v2
with:
registry: pilotdataplatform.azurecr.io
username: ${{ secrets.ACR_CLIENT }}
@@ -54,12 +61,11 @@ jobs:
# generate Docker tags based on the following events/attributes
sep-tags: ','
tags: |
- type=sha,enable=true,prefix=arranger-admin-server-,suffix=,format=short
- type=raw,prefix=arranger-admin-server-,suffix=,value=${{needs.get-version.outputs.app_version}}, enable=${{needs.extract-branch-name.outputs.branch=='patched-version'}}
+ type=raw,prefix=arranger-admin-server-,suffix=,value=${{needs.get-version.outputs.app_version}}
# - name: Image digest
# run: echo ${{ steps.meta.outputs.tags }}
- name: Build image and push to GitHub Container Registry
- uses: docker/build-push-action@v2
+ uses: docker/build-push-action@v4
with:
# relative path to the place where source code with Dockerfile is located
context: .
@@ -71,7 +77,7 @@ jobs:
target: arranger-admin-server
Trigger:
- needs: [build-and-push-docker-image, extract-branch-name]
+ needs: [build-and-push-docker-image, extract-branch-name, get-version]
#if: ${{ (github.event.workflow_run.conclusion == 'success' && needs.extract-branch-name.outputs.branch == 'patched-version') || (needs.extract-branch-name.outputs.branch == 'main') }}
if: ${{ (needs.extract-branch-name.outputs.branch == 'patched-version') || (needs.extract-branch-name.outputs.branch == 'main') }}
name: Trigger jenkins job
@@ -99,7 +105,7 @@ jobs:
target_release=arranger-admin-server
env=${ENV}
echo "The CI external URL is ${CI_EXTERNAL_URL}"
- curl --silent -i -X POST -m 60 -L -o output.txt --user jenkins:${CI_API_TOKEN} ${CI_EXTERNAL_URL}/job/Infra/job/UpdateAppVersion/buildWithParameters --data TF_TARGET_ENV=$env --data TARGET_RELEASE=$target_release --data NEW_APP_VERSION="${{ env.SHA }}"
+ curl --silent -i -X POST -m 60 -L -o output.txt --user jenkins:${CI_API_TOKEN} ${CI_EXTERNAL_URL}/job/Infra/job/UpdateAppVersion/buildWithParameters --data TF_TARGET_ENV=$env --data TARGET_RELEASE=$target_release --data NEW_APP_VERSION="${{needs.get-version.outputs.app_version}}"
location=$(cat output.txt | grep location | cut -d " " -f2 | sed 's/\r//')
link=${location}api/json && echo $link && echo -n "location_link=$link" >> $GITHUB_ENV
- name: getting the json
diff --git a/.github/workflows/build-and-publish-admin-ui.yml b/.github/workflows/build-and-publish-admin-ui.yml
index cc07ee1a7..96c00be76 100644
--- a/.github/workflows/build-and-publish-admin-ui.yml
+++ b/.github/workflows/build-and-publish-admin-ui.yml
@@ -14,18 +14,25 @@ jobs:
- name: Extract Branch Name
id: extract_branch
shell: bash
- run: echo "##[set-output name=branch;]$(echo ${GITHUB_REF#refs/heads/})"
+ run: echo "branch=${GITHUB_REF#refs/heads/}" >> $GITHUB_OUTPUT
get-version:
runs-on: ubuntu-20.04
outputs:
app_version: ${{steps.get-version.outputs.app_version}}
steps:
- name: Checkout code
- uses: actions/checkout@v2
+ uses: actions/checkout@v3
- name: Get Version
id: get-version
shell: bash
- run: echo "app_version=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`" >> $GITHUB_OUTPUT
+ run: |
+ BRANCH=${GITHUB_REF#refs/heads/}
+ BASE_VERSION=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`
+ if [ $BRANCH == "patched-version" ]; then
+ echo "app_version=$BASE_VERSION-`git rev-parse --short HEAD`" >> $GITHUB_OUTPUT
+ else
+ echo "app_version=$BASE_VERSION" >> $GITHUB_OUTPUT
+ fi
build-and-push-docker-image:
needs: [extract-branch-name, get-version]
@@ -34,12 +41,12 @@ jobs:
runs-on: ubuntu-20.04
steps:
- name: Checkout code
- uses: actions/checkout@v2
+ uses: actions/checkout@v3
- name: Set up Docker Buildx
id: buildx
- uses: docker/setup-buildx-action@v1
+ uses: docker/setup-buildx-action@v2
- name: Login to Github Packages
- uses: docker/login-action@v1
+ uses: docker/login-action@v2
with:
registry: pilotdataplatform.azurecr.io
username: ${{ secrets.ACR_CLIENT }}
@@ -54,12 +61,11 @@ jobs:
# generate Docker tags based on the following events/attributes
sep-tags: ','
tags: |
- type=sha,enable=true,prefix=arranger-admin-ui-,suffix=,format=short
- type=raw,prefix=arranger-admin-ui-,suffix=,value=${{needs.get-version.outputs.app_version}}, enable=${{needs.extract-branch-name.outputs.branch=='patched-version'}}
+ type=raw,prefix=arranger-admin-ui-,suffix=,value=${{needs.get-version.outputs.app_version}}
# - name: Image digest
# run: echo ${{ steps.meta.outputs.tags }}
- name: Build image and push to GitHub Container Registry
- uses: docker/build-push-action@v2
+ uses: docker/build-push-action@v4
with:
# relative path to the place where source code with Dockerfile is located
context: .
@@ -71,7 +77,7 @@ jobs:
target: arranger-admin-ui
Trigger:
- needs: [build-and-push-docker-image, extract-branch-name]
+ needs: [build-and-push-docker-image, extract-branch-name, get-version]
#if: ${{ (github.event.workflow_run.conclusion == 'success' && needs.extract-branch-name.outputs.branch == 'patched-version') || (needs.extract-branch-name.outputs.branch == 'main') }}
if: ${{ (needs.extract-branch-name.outputs.branch == 'patched-version') || (needs.extract-branch-name.outputs.branch == 'main') }}
name: Trigger jenkins job
@@ -99,7 +105,7 @@ jobs:
target_release=arranger-admin-ui
env=${ENV}
echo "The CI external URL is ${CI_EXTERNAL_URL}"
- curl --silent -i -X POST -m 60 -L -o output.txt --user jenkins:${CI_API_TOKEN} ${CI_EXTERNAL_URL}/job/Infra/job/UpdateAppVersion/buildWithParameters --data TF_TARGET_ENV=$env --data TARGET_RELEASE=$target_release --data NEW_APP_VERSION="${{ env.SHA }}"
+ curl --silent -i -X POST -m 60 -L -o output.txt --user jenkins:${CI_API_TOKEN} ${CI_EXTERNAL_URL}/job/Infra/job/UpdateAppVersion/buildWithParameters --data TF_TARGET_ENV=$env --data TARGET_RELEASE=$target_release --data NEW_APP_VERSION="${{needs.get-version.outputs.app_version}}"
location=$(cat output.txt | grep location | cut -d " " -f2 | sed 's/\r//')
link=${location}api/json && echo $link && echo -n "location_link=$link" >> $GITHUB_ENV
- name: getting the json
diff --git a/.github/workflows/build-and-publish-server-filter.yml b/.github/workflows/build-and-publish-server-filter.yml
index 1602b72b6..b0dec9212 100644
--- a/.github/workflows/build-and-publish-server-filter.yml
+++ b/.github/workflows/build-and-publish-server-filter.yml
@@ -14,18 +14,25 @@ jobs:
- name: Extract Branch Name
id: extract_branch
shell: bash
- run: echo "##[set-output name=branch;]$(echo ${GITHUB_REF#refs/heads/})"
+ run: echo "branch=${GITHUB_REF#refs/heads/}" >> $GITHUB_OUTPUT
get-version:
runs-on: ubuntu-20.04
outputs:
app_version: ${{steps.get-version.outputs.app_version}}
steps:
- name: Checkout code
- uses: actions/checkout@v2
+ uses: actions/checkout@v3
- name: Get Version
id: get-version
shell: bash
- run: echo "app_version=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`" >> $GITHUB_OUTPUT
+ run: |
+ BRANCH=${GITHUB_REF#refs/heads/}
+ BASE_VERSION=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`
+ if [ $BRANCH == "patched-version" ]; then
+ echo "app_version=$BASE_VERSION-`git rev-parse --short HEAD`" >> $GITHUB_OUTPUT
+ else
+ echo "app_version=$BASE_VERSION" >> $GITHUB_OUTPUT
+ fi
build-and-push-docker-image:
needs: [extract-branch-name, get-version]
@@ -34,12 +41,12 @@ jobs:
runs-on: ubuntu-20.04
steps:
- name: Checkout code
- uses: actions/checkout@v2
+ uses: actions/checkout@v3
- name: Set up Docker Buildx
id: buildx
- uses: docker/setup-buildx-action@v1
+ uses: docker/setup-buildx-action@v2
- name: Login to Github Packages
- uses: docker/login-action@v1
+ uses: docker/login-action@v2
with:
registry: pilotdataplatform.azurecr.io
username: ${{ secrets.ACR_CLIENT }}
@@ -54,12 +61,11 @@ jobs:
# generate Docker tags based on the following events/attributes
sep-tags: ','
tags: |
- type=sha,enable=true,prefix=arranger-server-filter-,suffix=,format=short
- type=raw,prefix=arranger-server-filter-,suffix=,value=${{needs.get-version.outputs.app_version}}, enable=${{needs.extract-branch-name.outputs.branch=='patched-version'}}
+ type=raw,prefix=arranger-server-filter-,suffix=,value=${{needs.get-version.outputs.app_version}}
# - name: Image digest
# run: echo ${{ steps.meta.outputs.tags }}
- name: Build image and push to GitHub Container Registry
- uses: docker/build-push-action@v2
+ uses: docker/build-push-action@v4
with:
# relative path to the place where source code with Dockerfile is located
context: .
@@ -71,7 +77,7 @@ jobs:
target: arranger-server
Trigger:
- needs: [build-and-push-docker-image, extract-branch-name]
+ needs: [build-and-push-docker-image, extract-branch-name, get-version]
#if: ${{ (github.event.workflow_run.conclusion == 'success' && needs.extract-branch-name.outputs.branch == 'patched-version') || (needs.extract-branch-name.outputs.branch == 'main') }}
if: ${{ (needs.extract-branch-name.outputs.branch == 'patched-version') || (needs.extract-branch-name.outputs.branch == 'main') }}
name: Trigger jenkins job
@@ -99,7 +105,7 @@ jobs:
target_release=arranger-server-filter
env=${ENV}
echo "The CI external URL is ${CI_EXTERNAL_URL}"
- curl --silent -i -X POST -m 60 -L -o output.txt --user jenkins:${CI_API_TOKEN} ${CI_EXTERNAL_URL}/job/Infra/job/UpdateAppVersion/buildWithParameters --data TF_TARGET_ENV=$env --data TARGET_RELEASE=$target_release --data NEW_APP_VERSION="${{ env.SHA }}"
+ curl --silent -i -X POST -m 60 -L -o output.txt --user jenkins:${CI_API_TOKEN} ${CI_EXTERNAL_URL}/job/Infra/job/UpdateAppVersion/buildWithParameters --data TF_TARGET_ENV=$env --data TARGET_RELEASE=$target_release --data NEW_APP_VERSION="${{needs.get-version.outputs.app_version}}"
location=$(cat output.txt | grep location | cut -d " " -f2 | sed 's/\r//')
link=${location}api/json && echo $link && echo -n "location_link=$link" >> $GITHUB_ENV
- name: getting the json
diff --git a/.github/workflows/pilot-pipeline-admin-server.yml b/.github/workflows/pilot-pipeline-admin-server.yml
new file mode 100644
index 000000000..29fbb7c2f
--- /dev/null
+++ b/.github/workflows/pilot-pipeline-admin-server.yml
@@ -0,0 +1,165 @@
+name: CI pipeline admin server
+
+on:
+ push:
+ branches:
+ - patched-version
+ paths:
+ - 'modules/admin/**'
+ pull_request:
+ branches:
+ - patched-version
+ paths:
+ - 'modules/admin/**'
+
+jobs:
+
+ extract-branch-name:
+ runs-on: ubuntu-20.04
+ outputs:
+ branch: ${{steps.extract_branch.outputs.branch}}
+ steps:
+ - name: Extract Branch Name
+ id: extract_branch
+ shell: bash
+ run: echo "branch=${GITHUB_REF#refs/heads/}" >> $GITHUB_OUTPUT
+ get-version:
+ runs-on: ubuntu-20.04
+ outputs:
+ app_version: ${{steps.get-version.outputs.app_version}}
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+ - name: Get Version
+ id: get-version
+ shell: bash
+ run: |
+ BRANCH=${GITHUB_REF#refs/heads/}
+ BASE_VERSION=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`
+ echo "app_version=$BASE_VERSION" >> $GITHUB_OUTPUT
+
+ build-and-push-docker-image:
+ needs: [extract-branch-name, get-version]
+ if: ${{ needs.extract-branch-name.outputs.branch == 'patched-version'}}
+ name: Build admin server Docker image and push to repositories
+ runs-on: ubuntu-20.04
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+ - name: Set up Docker Buildx
+ id: buildx
+ uses: docker/setup-buildx-action@v2
+ - name: Login to Github Packages
+ uses: docker/login-action@v2
+ with:
+ registry: pilotdataplatform.azurecr.io
+ username: ${{ secrets.ACR_CLIENT }}
+ password: ${{ secrets.ACR_SECRET }}
+ - name: Docker metadata
+ id: meta
+ uses: docker/metadata-action@v4
+ with:
+ # list of Docker images to use as base name for tags
+ images: |
+ pilotdataplatform.azurecr.io/arranger/arranger-admin-server
+ # generate Docker tags based on the following events/attributes
+ sep-tags: ','
+ tags: |
+ type=raw,prefix=arranger-admin-server-,suffix=,value=${{needs.get-version.outputs.app_version}}
+# - name: Image digest
+# run: echo ${{ steps.meta.outputs.tags }}
+ - name: Image digest
+ run: echo ${{ steps.meta.outputs.tags }}
+ - name: Check if Docker image tags exist
+ shell: bash
+ run: |
+ image_tag=$(echo "${{ steps.meta.outputs.tags }}")
+ if docker manifest inspect $image_tag >/dev/null; then
+ echo "Docker image with tag already exists. Please update the version."
+ exit 1
+ else
+ echo "Image tags do not exist, proceeding..."
+ fi
+ - name: Check Trivy ignore exceptions
+ # Check if the Trivy ignorefile has exceptions and the exceptions
+ # date is defined and less than 90 days.
+ #
+ # The ignorefile has the following format:
+ # CVE-YYYY-XXXXX exp:YYYY-MM-DD
+ #
+ # Error out if the format is not respected.
+ # Given a CVE-YYYY-XXXXX defined in the ignorefile,
+ # if no exception date is defined, we will error out.
+ # if the exception date is more than 90 days, we will error out too.
+ #
+ # Successfully parsed lines will be ignored,
+ # if the exception date exist and is less than 90 days, the step will exit with 0.
+ uses: PilotDataPlatform/shared-ci-tools/.github/actions/trivyignore-checking@main
+ - name: Build image and push to GitHub Container Registry
+ uses: docker/build-push-action@v4
+ with:
+ # relative path to the place where source code with Dockerfile is located
+ context: .
+ # Note: tags has to be all lower-case
+ tags: ${{ steps.meta.outputs.tags }}
+ # build on feature branches, push only on main branch
+ push: ${{ github.event_name != 'pull_request' }}
+ # Sets the target stage to build
+ target: arranger-admin-server
+ load: true
+
+ - name: Run Trivy vulnerability scanner
+ uses: aquasecurity/trivy-action@0.20.0
+ with:
+ image-ref: '${{ steps.meta.outputs.tags }}'
+ format: 'table'
+ severity: 'CRITICAL'
+ exit-code: '1'
+ hide-progress: true
+ trivyignores: .github/.trivyignore
+ output: scan-results.txt
+ env:
+ TRIVY_IGNORE_STATUS: 'will_not_fix'
+
+ - name: Publish Trivy Scan Results to Summary
+ if: always()
+ run: |
+ if [[ -s scan-results.txt ]]; then
+ {
+ echo "### Trivy Scan Results"
+ echo "Click to expand
"
+ echo ""
+ echo '```arranger-admin-server'
+ cat scan-results.txt
+ echo '```'
+ echo " "
+ } >> $GITHUB_STEP_SUMMARY
+ fi
+
+ trigger_pilot_dev_deployment:
+ needs: [build-and-push-docker-image, get-version]
+ runs-on: ubuntu-20.04
+ if: github.event_name != 'pull_request'
+ steps:
+ - name: Checkout helmfile repo
+ uses: actions/checkout@v2
+ with:
+ repository: PilotDataPlatform/pilot-helmfile
+ ref: 'main' #always checkout main branch
+ ssh-key: ${{ secrets.PILOT_HELMFILE_REPO_DEPLOYMENT_KEY }}
+
+ - name: Update service version
+ run: |
+ git config user.name "GitHub Actions"
+ git config user.email "github-actions@users.noreply.indocresearch.org"
+ BASE_FILE='./helmfile.d/values/shared/lab/shared-services-values.yaml'
+
+ pattern="arranger_admin_server"
+
+ current_version=$(grep -m 1 $pattern $BASE_FILE)
+ echo "current version is $current_version"
+
+ sed -i "/$pattern/,/charts/ s/$current_version/ $pattern: ${{needs.get-version.outputs.app_version}}/" $BASE_FILE
+ git add $BASE_FILE
+ git commit -m "Deploy arranger-admin-server ${{needs.get-version.outputs.app_version}} [app_name:arranger-admin-server, app_version:${{needs.get-version.outputs.app_version}}]"
+ git push origin main
diff --git a/.github/workflows/pilot-pipeline-admin-ui.yml b/.github/workflows/pilot-pipeline-admin-ui.yml
new file mode 100644
index 000000000..8e47d2f01
--- /dev/null
+++ b/.github/workflows/pilot-pipeline-admin-ui.yml
@@ -0,0 +1,165 @@
+name: CI pipeline admin ui
+
+on:
+ push:
+ branches:
+ - patched-version
+ paths:
+ - 'modules/admin-ui/**'
+ pull_request:
+ branches:
+ - patched-version
+ paths:
+ - 'modules/admin-ui/**'
+
+jobs:
+
+ extract-branch-name:
+ runs-on: ubuntu-20.04
+ outputs:
+ branch: ${{steps.extract_branch.outputs.branch}}
+ steps:
+ - name: Extract Branch Name
+ id: extract_branch
+ shell: bash
+ run: echo "branch=${GITHUB_REF#refs/heads/}" >> $GITHUB_OUTPUT
+ get-version:
+ runs-on: ubuntu-20.04
+ outputs:
+ app_version: ${{steps.get-version.outputs.app_version}}
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+ - name: Get Version
+ id: get-version
+ shell: bash
+ run: |
+ BRANCH=${GITHUB_REF#refs/heads/}
+ BASE_VERSION=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`
+ echo "app_version=$BASE_VERSION" >> $GITHUB_OUTPUT
+
+ build-and-push-docker-image:
+ needs: [extract-branch-name, get-version]
+ if: ${{ needs.extract-branch-name.outputs.branch == 'patched-version'}}
+ name: Build admin ui Docker image and push to repositories
+ runs-on: ubuntu-20.04
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+ - name: Set up Docker Buildx
+ id: buildx
+ uses: docker/setup-buildx-action@v2
+ - name: Login to Github Packages
+ uses: docker/login-action@v2
+ with:
+ registry: pilotdataplatform.azurecr.io
+ username: ${{ secrets.ACR_CLIENT }}
+ password: ${{ secrets.ACR_SECRET }}
+ - name: Docker metadata
+ id: meta
+ uses: docker/metadata-action@v4
+ with:
+ # list of Docker images to use as base name for tags
+ images: |
+ pilotdataplatform.azurecr.io/arranger/arranger-admin-ui
+ # generate Docker tags based on the following events/attributes
+ sep-tags: ','
+ tags: |
+ type=raw,prefix=arranger-admin-ui-,suffix=,value=${{needs.get-version.outputs.app_version}}
+# - name: Image digest
+# run: echo ${{ steps.meta.outputs.tags }}
+ - name: Image digest
+ run: echo ${{ steps.meta.outputs.tags }}
+ - name: Check if Docker image tags exist
+ shell: bash
+ run: |
+ image_tag=$(echo "${{ steps.meta.outputs.tags }}")
+ if docker manifest inspect $image_tag >/dev/null; then
+ echo "Docker image with tag already exists. Please update the version."
+ exit 1
+ else
+ echo "Image tags do not exist, proceeding..."
+ fi
+ - name: Check Trivy ignore exceptions
+ # Check if the Trivy ignorefile has exceptions and the exceptions
+ # date is defined and less than 90 days.
+ #
+ # The ignorefile has the following format:
+ # CVE-YYYY-XXXXX exp:YYYY-MM-DD
+ #
+ # Error out if the format is not respected.
+ # Given a CVE-YYYY-XXXXX defined in the ignorefile,
+ # if no exception date is defined, we will error out.
+ # if the exception date is more than 90 days, we will error out too.
+ #
+ # Successfully parsed lines will be ignored,
+ # if the exception date exist and is less than 90 days, the step will exit with 0.
+ uses: PilotDataPlatform/shared-ci-tools/.github/actions/trivyignore-checking@main
+ - name: Build image and push to GitHub Container Registry
+ uses: docker/build-push-action@v4
+ with:
+ # relative path to the place where source code with Dockerfile is located
+ context: .
+ # Note: tags has to be all lower-case
+ tags: ${{ steps.meta.outputs.tags }}
+ # build on feature branches, push only on main branch
+ push: ${{ github.event_name != 'pull_request' }}
+ # Sets the target stage to build
+ target: arranger-admin-ui
+ load: true
+
+ - name: Run Trivy vulnerability scanner
+ uses: aquasecurity/trivy-action@0.20.0
+ with:
+ image-ref: '${{ steps.meta.outputs.tags }}'
+ format: 'table'
+ severity: 'CRITICAL'
+ exit-code: '1'
+ hide-progress: true
+ trivyignores: .github/.trivyignore
+ output: scan-results.txt
+ env:
+ TRIVY_IGNORE_STATUS: 'will_not_fix'
+
+ - name: Publish Trivy Scan Results to Summary
+ if: always()
+ run: |
+ if [[ -s scan-results.txt ]]; then
+ {
+ echo "### Trivy Scan Results"
+ echo "Click to expand
"
+ echo ""
+ echo '```arranger-admin-ui'
+ cat scan-results.txt
+ echo '```'
+ echo " "
+ } >> $GITHUB_STEP_SUMMARY
+ fi
+
+ trigger_pilot_dev_deployment:
+ needs: [build-and-push-docker-image, get-version]
+ runs-on: ubuntu-20.04
+ if: github.event_name != 'pull_request'
+ steps:
+ - name: Checkout helmfile repo
+ uses: actions/checkout@v2
+ with:
+ repository: PilotDataPlatform/pilot-helmfile
+ ref: 'main' #always checkout main branch
+ ssh-key: ${{ secrets.PILOT_HELMFILE_REPO_DEPLOYMENT_KEY }}
+
+ - name: Update service version
+ run: |
+ git config user.name "GitHub Actions"
+ git config user.email "github-actions@users.noreply.indocresearch.org"
+ BASE_FILE='./helmfile.d/values/shared/lab/shared-services-values.yaml'
+
+ pattern="arranger_admin_ui"
+
+ current_version=$(grep -m 1 $pattern $BASE_FILE)
+ echo "current version is $current_version"
+
+ sed -i "/$pattern/,/charts/ s/$current_version/ $pattern: ${{needs.get-version.outputs.app_version}}/" $BASE_FILE
+ git add $BASE_FILE
+ git commit -m "Deploy arranger-admin-ui ${{needs.get-version.outputs.app_version}} [app_name:arranger-admin-ui, app_version:${{needs.get-version.outputs.app_version}}]"
+ git push origin main
diff --git a/.github/workflows/pilot-pipeline-server-filter.yml b/.github/workflows/pilot-pipeline-server-filter.yml
new file mode 100644
index 000000000..269f1670e
--- /dev/null
+++ b/.github/workflows/pilot-pipeline-server-filter.yml
@@ -0,0 +1,165 @@
+name: CI pipeline server filter
+
+on:
+ push:
+ branches:
+ - patched-version
+ paths:
+ - 'modules/server-filter/**'
+ pull_request:
+ branches:
+ - patched-version
+ paths:
+ - 'modules/server-filter/**'
+
+jobs:
+
+ extract-branch-name:
+ runs-on: ubuntu-20.04
+ outputs:
+ branch: ${{steps.extract_branch.outputs.branch}}
+ steps:
+ - name: Extract Branch Name
+ id: extract_branch
+ shell: bash
+ run: echo "branch=${GITHUB_REF#refs/heads/}" >> $GITHUB_OUTPUT
+ get-version:
+ runs-on: ubuntu-20.04
+ outputs:
+ app_version: ${{steps.get-version.outputs.app_version}}
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+ - name: Get Version
+ id: get-version
+ shell: bash
+ run: |
+ BRANCH=${GITHUB_REF#refs/heads/}
+ BASE_VERSION=`sed -n 's/^ *"version":.*"\([^"]*\)".*/\1/p' package.json`
+ echo "app_version=$BASE_VERSION" >> $GITHUB_OUTPUT
+
+ build-and-push-docker-image:
+ needs: [extract-branch-name, get-version]
+ if: ${{ needs.extract-branch-name.outputs.branch == 'patched-version'}}
+ name: Build server filter Docker image and push to repositories
+ runs-on: ubuntu-20.04
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+ - name: Set up Docker Buildx
+ id: buildx
+ uses: docker/setup-buildx-action@v2
+ - name: Login to Github Packages
+ uses: docker/login-action@v2
+ with:
+ registry: pilotdataplatform.azurecr.io
+ username: ${{ secrets.ACR_CLIENT }}
+ password: ${{ secrets.ACR_SECRET }}
+ - name: Docker metadata
+ id: meta
+ uses: docker/metadata-action@v4
+ with:
+ # list of Docker images to use as base name for tags
+ images: |
+ pilotdataplatform.azurecr.io/arranger/arranger-server
+ # generate Docker tags based on the following events/attributes
+ sep-tags: ','
+ tags: |
+ type=raw,prefix=arranger-server-filter-,suffix=,value=${{needs.get-version.outputs.app_version}}
+# - name: Image digest
+# run: echo ${{ steps.meta.outputs.tags }}
+ - name: Image digest
+ run: echo ${{ steps.meta.outputs.tags }}
+ - name: Check if Docker image tags exist
+ shell: bash
+ run: |
+ image_tag=$(echo "${{ steps.meta.outputs.tags }}")
+ if docker manifest inspect $image_tag >/dev/null; then
+ echo "Docker image with tag already exists. Please update the version."
+ exit 1
+ else
+ echo "Image tags do not exist, proceeding..."
+ fi
+ - name: Check Trivy ignore exceptions
+ # Check if the Trivy ignorefile has exceptions and the exceptions
+ # date is defined and less than 90 days.
+ #
+ # The ignorefile has the following format:
+ # CVE-YYYY-XXXXX exp:YYYY-MM-DD
+ #
+ # Error out if the format is not respected.
+ # Given a CVE-YYYY-XXXXX defined in the ignorefile,
+ # if no exception date is defined, we will error out.
+ # if the exception date is more than 90 days, we will error out too.
+ #
+ # Successfully parsed lines will be ignored,
+ # if the exception date exist and is less than 90 days, the step will exit with 0.
+ uses: PilotDataPlatform/shared-ci-tools/.github/actions/trivyignore-checking@main
+ - name: Build image and push to GitHub Container Registry
+ uses: docker/build-push-action@v4
+ with:
+ # relative path to the place where source code with Dockerfile is located
+ context: .
+ # Note: tags has to be all lower-case
+ tags: ${{ steps.meta.outputs.tags }}
+ # build on feature branches, push only on main branch
+ push: ${{ github.event_name != 'pull_request' }}
+ # Sets the target stage to build
+ target: arranger-server
+ load: true
+
+ - name: Run Trivy vulnerability scanner
+ uses: aquasecurity/trivy-action@0.20.0
+ with:
+ image-ref: '${{ steps.meta.outputs.tags }}'
+ format: 'table'
+ severity: 'CRITICAL'
+ exit-code: '1'
+ hide-progress: true
+ trivyignores: .github/.trivyignore
+ output: scan-results.txt
+ env:
+ TRIVY_IGNORE_STATUS: 'will_not_fix'
+
+ - name: Publish Trivy Scan Results to Summary
+ if: always()
+ run: |
+ if [[ -s scan-results.txt ]]; then
+ {
+ echo "### Trivy Scan Results"
+ echo "Click to expand
"
+ echo ""
+ echo '```arranger-server-filter'
+ cat scan-results.txt
+ echo '```'
+ echo " "
+ } >> $GITHUB_STEP_SUMMARY
+ fi
+
+ trigger_pilot_dev_deployment:
+ needs: [build-and-push-docker-image, get-version]
+ runs-on: ubuntu-20.04
+ if: github.event_name != 'pull_request'
+ steps:
+ - name: Checkout helmfile repo
+ uses: actions/checkout@v2
+ with:
+ repository: PilotDataPlatform/pilot-helmfile
+ ref: 'main' #always checkout main branch
+ ssh-key: ${{ secrets.PILOT_HELMFILE_REPO_DEPLOYMENT_KEY }}
+
+ - name: Update service version
+ run: |
+ git config user.name "GitHub Actions"
+ git config user.email "github-actions@users.noreply.indocresearch.org"
+ BASE_FILE='./helmfile.d/values/development/dev/pilot-versions-values.yaml'
+
+ pattern="arranger_server_filter_service"
+
+ current_version=$(grep -m 1 $pattern $BASE_FILE)
+ echo "current version is $current_version"
+
+ sed -i "/$pattern/,/service/ s/$current_version/ $pattern: ${{needs.get-version.outputs.app_version}}/" $BASE_FILE
+ git add $BASE_FILE
+ git commit -m "Deploy arranger-server-filter ${{needs.get-version.outputs.app_version}} [app_name:arranger-server-filter, app_version:${{needs.get-version.outputs.app_version}}]"
+ git push origin main
diff --git a/Dockerfile b/Dockerfile
index 5b1921bce..7c2d58d7b 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -1,11 +1,17 @@
FROM node:14.21-alpine as builder
+RUN addgroup --system arranger && \
+ adduser -S arranger -G arranger
+
WORKDIR /app
COPY . .
+RUN chown -R arranger:arranger /app
FROM builder as arranger-server
WORKDIR modules/server-filter
+RUN chown -R arranger:arranger /app/modules/server-filter
+USER arranger
RUN npm install
ENTRYPOINT ["node","server.js"]
@@ -13,11 +19,17 @@ ENTRYPOINT ["node","server.js"]
FROM builder as arranger-admin-server
WORKDIR modules/admin
+RUN chown -R arranger:arranger /app/modules/admin
+USER arranger
RUN npm install
ENTRYPOINT ["node","admin-server.js"]
FROM builder as arranger-admin-ui
+USER arranger
RUN npm ci
RUN npm run bootstrap
WORKDIR modules/admin-ui
+USER root
+RUN chown -R arranger:arranger /app/modules/admin-ui
+USER arranger
ENTRYPOINT ["npm", "run", "start"]
diff --git a/modules/admin/config.js b/modules/admin/config.js
index f360a8b94..75d1bf9e3 100644
--- a/modules/admin/config.js
+++ b/modules/admin/config.js
@@ -1,2 +1,2 @@
-export const ADMIN_SERVER_PORT = process.env.ADMIN_SERVER_PORT || 5050;
+export const ADMIN_SERVER_PORT = process.env.ADMIN_SERVER_PORT || 5051;
export const ELASTICSEARCH = process.env.ELASTICSEARCH || 'http://127.0.0.1:9200';
diff --git a/modules/components/src/Aggs/AggsState.js b/modules/components/src/Aggs/AggsState.js
index 2443fede7..5553f22e5 100644
--- a/modules/components/src/Aggs/AggsState.js
+++ b/modules/components/src/Aggs/AggsState.js
@@ -59,7 +59,6 @@ export default class extends Component {
let { data } = await api({
endpoint: `/${this.props.projectId}/graphql/aggsStateQuery`,
body: {
- project_code: 'indoctestproject',
query: `query aggsStateQuery
{
${graphqlField} {
diff --git a/modules/components/src/Arranger/GetProjects.js b/modules/components/src/Arranger/GetProjects.js
index fb1abfbbd..b7489cf7b 100644
--- a/modules/components/src/Arranger/GetProjects.js
+++ b/modules/components/src/Arranger/GetProjects.js
@@ -1,11 +1,11 @@
import React from 'react';
-import defaultApi from '../utils/api';
+import defaultAdminApi from '../utils/adminAPI';
class GetProjects extends React.Component {
state = { projects: [] };
async componentDidMount() {
- const { api = defaultApi } = this.props;
+ const { api = defaultAdminApi } = this.props;
let { data } = await api({
endpoint: '/admin/graphql',
body: {
diff --git a/modules/components/src/DataTable/ColumnsState.js b/modules/components/src/DataTable/ColumnsState.js
index f37d20311..86ccf4a72 100644
--- a/modules/components/src/DataTable/ColumnsState.js
+++ b/modules/components/src/DataTable/ColumnsState.js
@@ -63,7 +63,6 @@ export default class extends Component {
let { data } = await api({
endpoint: `/${this.props.projectId}/graphql/columnsStateQuery`,
body: {
- project_code: 'indoctestproject',
query: `query columnsStateQuery
{
${graphqlField} {
@@ -84,7 +83,6 @@ export default class extends Component {
} = await api({
endpoint: `/${this.props.projectId}/graphql`,
body: {
- project_code: 'indoctestproject',
query: `
query{
${this.props.graphqlField} {
diff --git a/modules/components/src/Query.js b/modules/components/src/Query.js
index e60f14566..79e061317 100644
--- a/modules/components/src/Query.js
+++ b/modules/components/src/Query.js
@@ -32,7 +32,7 @@ class Query extends Component {
let { data, errors } = await api({
...options,
endpoint: path.join(projectId, 'graphql', name || ''),
- body: { query, variables, project_code: 'indoctestproject' },
+ body: { query, variables },
});
this.setState({
data,
diff --git a/modules/components/src/utils/adminAPI.js b/modules/components/src/utils/adminAPI.js
new file mode 100644
index 000000000..12d8865f9
--- /dev/null
+++ b/modules/components/src/utils/adminAPI.js
@@ -0,0 +1,17 @@
+import { ARRANGER_ADMIN_API } from './config';
+import urlJoin from 'url-join';
+
+let alwaysSendHeaders = { 'Content-Type': 'application/json' };
+
+let Token = {
+ Authorization: `Bearer ${process.env.STORYBOOK_TOKEN}`,
+};
+
+export const defaultAdminApi = ({ endpoint = '', body, headers, method }) =>
+ fetch(urlJoin(ARRANGER_ADMIN_API, endpoint), {
+ method: method || 'POST',
+ headers: { ...alwaysSendHeaders, ...headers, ...Token },
+ body: JSON.stringify(body),
+ }).then((r) => r.json());
+
+export default defaultAdminApi;
diff --git a/modules/components/src/utils/api.js b/modules/components/src/utils/api.js
index 8a4fa03c7..db535ce05 100644
--- a/modules/components/src/utils/api.js
+++ b/modules/components/src/utils/api.js
@@ -21,7 +21,6 @@ export const fetchExtendedMapping = ({ graphqlField, projectId, api = defaultApi
api({
endpoint: `/${projectId}/graphql`,
body: {
- project_code: 'indoctestproject',
query: `
{
${graphqlField}{
diff --git a/modules/components/src/utils/config.js b/modules/components/src/utils/config.js
index a26dbf077..99597fa0f 100644
--- a/modules/components/src/utils/config.js
+++ b/modules/components/src/utils/config.js
@@ -16,6 +16,7 @@ export function deleteValue(key) {
}
export const ARRANGER_API = getValue('ARRANGER_API', 'http://localhost:5050');
+export const ARRANGER_ADMIN_API = getValue('ARRANGER_ADMIN_API', 'http://localhost:5051');
export const ES_HOST = getValue('ES_HOST', 'http://localhost:9200');
export const PROJECT_ID = getValue('PROJECT_ID', null);
export const ACTIVE_INDEX = getValue('ACTIVE_INDEX', null);
diff --git a/modules/mapping-utils/src/utils/columnsToGraphql.js b/modules/mapping-utils/src/utils/columnsToGraphql.js
index bb67681b7..ff9b31d27 100644
--- a/modules/mapping-utils/src/utils/columnsToGraphql.js
+++ b/modules/mapping-utils/src/utils/columnsToGraphql.js
@@ -27,7 +27,6 @@ export default function columnsToGraphql({ config = {}, sqon, queryName, sort, o
.join('\n');
return {
- project_code: 'indoctestproject',
fields,
query: `
query($sort: [Sort], $first: Int, $offset: Int, $score: String, $sqon: JSON) {
diff --git a/modules/nf-server-filter/.env.schema b/modules/nf-server-filter/.env.schema
new file mode 100644
index 000000000..2088b27fa
--- /dev/null
+++ b/modules/nf-server-filter/.env.schema
@@ -0,0 +1,12 @@
+# ARRANGER SERVER PORT
+SERVER_PORT=
+
+# ARRANGER PROJECT ID SETUP BY ADMIN UI
+ARRANGER_PROJECT_ID=
+
+# ELASTICSEARCH HOST:PORT
+ELASTICSEARCH=
+
+# DOWNLOAD FILE STREAM BUFFER SIZE (export)
+DOWNLOAD_STREAM_BUFFER_SIZE=
+
diff --git a/modules/nf-server-filter/README.md b/modules/nf-server-filter/README.md
new file mode 100644
index 000000000..336621a33
--- /dev/null
+++ b/modules/nf-server-filter/README.md
@@ -0,0 +1,23 @@
+# Nextflow Arranger Server
+
+### Overview
+
+This arranger server acts as a proxy to obtain nextflow workflow updates.
+
+### Prerequisites
+
+NodeJs version: `v14.21.0`; NPM version: `6.14.17`. Recommended to use node version manager (NVM).
+
+### Installation
+
+1. `cd nf-server-filter`
+2. `npm install`
+
+### Start up
+
+1. Set `.env` to point to desired server port, arranger project ID, and elasticsearch instance.
+2. Run `node server.js`
+
+### Important notes
+
+1. Export-to-file logic exists under `/export`.
diff --git a/modules/nf-server-filter/authFilter.js b/modules/nf-server-filter/authFilter.js
new file mode 100644
index 000000000..afe3ea23d
--- /dev/null
+++ b/modules/nf-server-filter/authFilter.js
@@ -0,0 +1,19 @@
+import { BaseSQON } from './sqon/base_sqon.js';
+
+/**
+ * Function will create custom SQON filter
+ * @returns a JSON encoded SQON filter to apply to graphql query, including aggregations
+ */
+
+export async function arrangerFilter() {
+ return BaseSQON.generate();
+}
+
+/**
+ * Function will process incoming request to obtain SQON filter
+ * @returns a JSON encoded SQON filter to apply to graphql query, including aggregations
+ */
+
+export async function processRequest() {
+ return await arrangerFilter();
+}
diff --git a/modules/nf-server-filter/config.js b/modules/nf-server-filter/config.js
new file mode 100644
index 000000000..c998cd175
--- /dev/null
+++ b/modules/nf-server-filter/config.js
@@ -0,0 +1,4 @@
+export const SERVER_PORT = process.env.SERVER_PORT || 5050;
+export const ARRANGER_PROJECT_ID = process.env.ARRANGER_PROJECT_ID || 'nextflow-runs';
+export const ELASTICSEARCH = process.env.ELASTICSEARCH || 'http://127.0.0.1:9200';
+export const DOWNLOAD_STREAM_BUFFER_SIZE = process.env.DOWNLOAD_STREAM_BUFFER_SIZE || 2000;
diff --git a/modules/nf-server-filter/export/export-file.js b/modules/nf-server-filter/export/export-file.js
new file mode 100644
index 000000000..346e3c017
--- /dev/null
+++ b/modules/nf-server-filter/export/export-file.js
@@ -0,0 +1,160 @@
+import express from 'express';
+import zlib from 'zlib';
+import tar from 'tar-stream';
+import pkg from 'lodash';
+const { defaults } = pkg;
+
+import { getAllData } from './utils/getAllData.js';
+import { dataToExportFormat } from './utils/dataToExportFormat.js';
+
+const convertDataToExportFormat =
+ ({ req, project_info, params, headers, ctx, fileType }) =>
+ async (args) =>
+ (
+ await getAllData({
+ req,
+ project_info,
+ params,
+ headers,
+ ctx,
+ ...args,
+ })
+ ).pipe(dataToExportFormat({ ...args, fileType }));
+
+const getFileStream = async ({
+ req,
+ project_info,
+ params,
+ headers,
+ chunkSize,
+ ctx,
+ file,
+ fileType,
+ mock,
+}) => {
+ const exportArgs = defaults(file, { chunkSize, fileType, mock });
+
+ return convertDataToExportFormat({ req, project_info, params, headers, ctx, fileType })({
+ ...exportArgs,
+ mock,
+ });
+};
+
+const multipleFiles = async ({
+ req,
+ project_info,
+ params,
+ headers,
+ chunkSize,
+ ctx,
+ files,
+ mock,
+}) => {
+ const pack = tar.pack();
+
+ Promise.all(
+ files.map((file, i) => {
+ return new Promise(async (resolve, reject) => {
+ // pack needs the size of the stream. We don't know that until we get all the data.
+ // This collects all the data before adding it.
+ let data = '';
+ const fileStream = await getFileStream({
+ req,
+ project_info,
+ params,
+ headers,
+ chunkSize,
+ ctx,
+ file,
+ fileType: file.fileType,
+ mock,
+ });
+
+ fileStream.on('data', (chunk) => (data += chunk));
+ fileStream.on('end', () => {
+ pack.entry({ name: file.fileName || `file-${i + 1}.tsv` }, data, function (err) {
+ if (err) {
+ reject(err);
+ } else {
+ resolve(null);
+ }
+ });
+ });
+ });
+ }),
+ ).then(() => pack.finalize());
+
+ /** NOTE from zlib's maintainers:
+ * (This library) is only intended for small (< 128 KB) data
+ * that you already have buffered. It is not meant for input/output streams.
+ * -- Found at: https://www.npmjs.com/package/zlib
+ *
+ * TODO: may have to find one that manages larger buffer sizes.
+ * Must do testing for this
+ */
+ return pack.pipe(zlib.createGzip());
+};
+
+export const dataStream = async ({ req, project_info, headers, ctx, params }) => {
+ const { chunkSize, files, fileName = 'file.tar.gz', fileType = 'tsv', mock } = params;
+
+ if (files?.length > 0) {
+ return files.length === 1
+ ? {
+ contentType: 'text/plain',
+ output: await getFileStream({
+ req,
+ project_info,
+ params,
+ headers,
+ chunkSize,
+ ctx,
+ file: files[0],
+ fileType: files[0].fileType || fileType,
+ mock,
+ }),
+ responseFileName: files[0].fileName || fileName,
+ }
+ : {
+ contentType: 'application/gzip',
+ output: multipleFiles({ req, project_info, headers, chunkSize, ctx, files, mock }),
+ responseFileName: fileName.replace(/(\.tar(\.gz)?)?$/, '.tar.gz'), // make sure file ends with '.tar.gz'
+ };
+ }
+
+ console.warn('no files defined to download');
+ throw new Error('files array was missing or empty');
+};
+
+export function downloader(project_info) {
+ const router = express.Router();
+
+ router.use(express.json());
+
+ router.post('/', async function (req, res) {
+ try {
+ const data = req.body;
+ const requestHeaders = {
+ ...req.headers,
+ Authorization: req.headers.authorization,
+ };
+ const { output, responseFileName, contentType } = await dataStream({
+ req,
+ project_info,
+ headers: requestHeaders,
+ ctx: req.context,
+ params: data,
+ });
+
+ res.set('Content-Type', contentType);
+ res.set('Content-disposition', `attachment; filename=${responseFileName}`);
+ output.pipe(res).on('finish', () => {});
+ } catch (err) {
+ console.error(err);
+
+ res.status(400).send(err?.message || err?.details || 'An unknown error occurred.');
+ }
+ });
+
+ return router;
+}
diff --git a/modules/nf-server-filter/export/utils/arrangerFilterDownload.js b/modules/nf-server-filter/export/utils/arrangerFilterDownload.js
new file mode 100644
index 000000000..abe3b1c33
--- /dev/null
+++ b/modules/nf-server-filter/export/utils/arrangerFilterDownload.js
@@ -0,0 +1,14 @@
+/**
+ * Function will return SQON to be passed to download stream
+ * @param req request that contains sqon generated by arranger filter.
+ */
+export function arrangerFilterDownload(req) {
+ try {
+ // add user-selected identifiers
+
+ // return custom sqon
+ return req.sqon;
+ } catch (error) {
+ console.log(`Error returning download SQON: ${error}`);
+ }
+}
diff --git a/modules/nf-server-filter/export/utils/dataToExportFormat.js b/modules/nf-server-filter/export/utils/dataToExportFormat.js
new file mode 100644
index 000000000..c3b08086d
--- /dev/null
+++ b/modules/nf-server-filter/export/utils/dataToExportFormat.js
@@ -0,0 +1,323 @@
+import { format, isValid, parseISO } from 'date-fns';
+import jsonPath from 'jsonpath';
+import pkg from 'lodash';
+const { get, flatten, isNil } = pkg;
+import through2 from 'through2';
+
+const STANDARD_DATE = 'yyyy-MM-dd';
+
+const dateHandler = (value, { dateFormat = STANDARD_DATE }) => {
+ switch (true) {
+ case isNil(value):
+ return '';
+
+ case isValid(new Date(value)):
+ return format(new Date(value), dateFormat);
+
+ case isValid(parseISO(value)):
+ return format(parseISO(value), dateFormat);
+
+ case !isNaN(parseInt(value, 10)):
+ return format(parseInt(value, 10), dateFormat);
+
+ default: {
+ console.error('unhandled "date"', value, dateFormat);
+ return value;
+ }
+ }
+};
+
+const getAllValue = (data) => {
+ if (typeof data === 'object') {
+ return Object.values(data || {})
+ .map(getAllValue)
+ .reduce((a, b) => a.concat(b), []);
+ } else {
+ return data;
+ }
+};
+
+const getValue = (row, column) => {
+ const valueFromExtended = (value) => {
+ switch (true) {
+ case column?.extendedDisplayValues?.constructor === Object &&
+ Object.keys(column.extendedDisplayValues).length > 0:
+ return column.extendedDisplayValues[value];
+
+ case column.isArray && Array.isArray(value):
+ return value
+ .map((each) => valueFromExtended(each))
+ .join(';')
+ .replace(';;', ';');
+
+ case [column.displayType, column.type].includes('date'):
+ return dateHandler(value, { dateFormat: column.displayFormat });
+
+ default:
+ return value;
+ }
+ };
+
+ if (column.jsonPath) {
+ return jsonPath
+ .query(row, column.jsonPath.split('.hits.edges[*].node.').join('[*].'))
+ .map(getAllValue)
+ .reduce((a, b) => a.concat(b), [])
+ .map(valueFromExtended)
+ .join(', ');
+ } else if (column.accessor) {
+ return valueFromExtended(get(row, column.accessor));
+ } else {
+ return '';
+ }
+};
+
+const getRows = (args) => {
+ const { row, data = row, paths, pathIndex = 0, columns, entities = [] } = args;
+ if (pathIndex >= paths.length - 1) {
+ return [
+ columns.map((column) => {
+ const entity = entities
+ .slice()
+ .reverse()
+ .find((entity) => column.field.indexOf(entity.field) === 0);
+
+ if (entity) {
+ return getValue(entity.data, {
+ ...column,
+ jsonPath: column.field.replace(`${entity.path.join('.')}.`, ''),
+ });
+ } else {
+ return getValue(row, column);
+ }
+ }),
+ ];
+ } else {
+ const currentPath = paths[pathIndex];
+ return flatten(
+ (get(data, currentPath) || []).map((node) => {
+ return getRows({
+ ...args,
+ data: node,
+ pathIndex: pathIndex + 1,
+ entities: [
+ ...entities,
+ {
+ path: paths.slice(0, pathIndex + 1),
+ field: paths.slice(0, pathIndex + 1).join(''),
+ // TODO: don't assume hits.edges.node.
+ // .replace(/(\.hits.edges(node)?)/g, ''),
+ data: node,
+ },
+ ],
+ });
+ }),
+ );
+ }
+};
+
+export const columnsToHeader = ({ columns, fileType = 'tsv' }) => {
+ return fileType === 'tsv'
+ ? `${columns.map(({ Header, displayName }) => (displayName ? displayName : Header)).join('\t')}`
+ : fileType === 'json'
+ ? columns.reduce((output, { Header, displayName, accessor }) => {
+ output[[accessor]] = displayName ? displayName : Header;
+ return output;
+ }, {})
+ : '';
+};
+
+export const dataToTSV = ({ isFirst, pipe, columns, ...args }) => {
+ if (isFirst) {
+ const headerRow = columnsToHeader({ columns, fileType: 'tsv' });
+ pushToStream(headerRow, pipe);
+ }
+
+ transformData({
+ isFirst,
+ pipe,
+ columns,
+ ...args,
+ dataTransformer: transformDataToTSV,
+ });
+};
+
+/**
+ * This should ideally stream data as a JSON list using JSONStream
+ * but as of now; in favor of simplicity; it streams each row as separate JSON object
+ * and it is left up to consuming application to make a well formatted
+ * JSON list from individual JSON objects
+ * See https://github.com/nci-hcmi-catalog/portal/tree/master/api/src/dataExport.js for an example consumer
+ * @param {*} param0
+ */
+export const dataToJSON = ({ isFirst, pipe, columns, ...args }) => {
+ if (isFirst) {
+ const headerRow = columnsToHeader({ columns, fileType: 'json' });
+ pushToStream(JSON.stringify(headerRow), pipe);
+ }
+
+ transformData({
+ isFirst,
+ pipe,
+ columns,
+ ...args,
+ dataTransformer: transformDataToJSON,
+ });
+};
+
+export function dataToExportFormat({
+ index,
+ columns,
+ emptyValue, // kept for backwards compatibility. To be deprecated
+ uniqueBy,
+ valueWhenEmpty = emptyValue || '--',
+ fileType = 'tsv',
+}) {
+ let isFirst = true;
+ let chunkCounts = 0;
+
+ return through2.obj(function ({ hits, total }, enc, callback) {
+ const outputStream = this;
+ dataToStream({
+ index,
+ columns,
+ uniqueBy,
+ valueWhenEmpty,
+ hits,
+ total,
+ pipe: outputStream,
+ isFirst,
+ fileType,
+ });
+
+ if (isFirst) {
+ isFirst = false;
+ }
+
+ callback();
+ chunkCounts++;
+ });
+}
+
+const transformData = ({
+ data: { hits, total },
+ data,
+ index,
+ uniqueBy,
+ columns,
+ valueWhenEmpty,
+ dataTransformer,
+ pipe,
+}) => {
+ hits
+ .map((row) => dataTransformer({ row, uniqueBy, columns, valueWhenEmpty }))
+ .forEach((transformedRow) => {
+ pushToStream(transformedRow, pipe);
+ });
+};
+
+const transformDataToTSV = ({ row, uniqueBy, columns, valueWhenEmpty }) => {
+ return getRows({
+ row: row._source,
+ paths: (uniqueBy || '').split('.hits.edges[].node.').filter(Boolean),
+ columns: columns,
+ valueWhenEmpty,
+ }).map((r) => rowToTSV({ row: r, valueWhenEmpty }));
+};
+
+const transformDataToJSON = ({ row, uniqueBy, columns, valueWhenEmpty }) => {
+ return JSON.stringify(
+ rowToJSON({
+ row: row._source,
+ paths: (uniqueBy || '').split('.hits.edges[].node.').filter(Boolean),
+ columns: columns,
+ valueWhenEmpty,
+ }),
+ );
+};
+
+const dataToStream = ({
+ index,
+ columns,
+ uniqueBy,
+ valueWhenEmpty = '--',
+ hits,
+ total,
+ pipe,
+ isFirst,
+ fileType = 'tsv',
+}) => {
+ const args = {
+ data: { hits, total },
+ index,
+ uniqueBy,
+ columns,
+ valueWhenEmpty,
+ pipe,
+ isFirst,
+ };
+
+ // transform and stream data
+ if (fileType === 'tsv') {
+ dataToTSV(args);
+ } else if (fileType === 'json') {
+ dataToJSON(args);
+ } else {
+ throw new Error('Unsupported file type specified for export.');
+ }
+};
+
+const rowToTSV = ({ row, valueWhenEmpty }) => row.map((r) => r || valueWhenEmpty).join('\t');
+
+/*
+example args:
+{ row: [250/1767]
+ { name: 'HCM-dddd-0000-C00',
+ type: '2-D: Conditionally reprogrammed cells',
+ growth_rate: 5,
+ files: [],
+ clinical_diagnosis: { clinical_tumor_diagnosis: 'Colorectal cancer' },
+ variants: [ [Object], [Object], [Object] ]
+ },
+ paths: [],
+ columns:
+ [ { field: 'name',
+ accessor: 'name',
+ show: true,
+ type: 'entity',
+ sortable: true,
+ canChangeShow: true,
+ query: null,
+ jsonPath: null,
+ Header: 'Name',
+ extendedType: 'keyword',
+ extendedDisplayValues: {},
+ hasCustomType: true,
+ minWidth: 140 },
+ { field: 'split_ratio',
+ accessor: 'split_ratio',
+ show: true,
+ type: 'string',
+ sortable: true,
+ canChangeShow: true,
+ query: null,
+ jsonPath: null,
+ Header: 'Split Ratio',
+ extendedType: 'keyword',
+ extendedDisplayValues: {},
+ hasCustomType: false } ],
+ valueWhenEmpty: '--' }
+*/
+const rowToJSON = (args) => {
+ const { row, data = row, paths, pathIndex = 0, columns, valueWhenEmpty, entities = [] } = args;
+ return (columns || [])
+ .filter((col) => col.show)
+ .reduce((output, col) => {
+ output[[col.accessor]] = row[col.accessor] || valueWhenEmpty;
+ return output;
+ }, {});
+};
+
+const pushToStream = (line, stream) => {
+ stream.push(`${line}\n`);
+};
diff --git a/modules/nf-server-filter/export/utils/getAllData.js b/modules/nf-server-filter/export/utils/getAllData.js
new file mode 100644
index 000000000..2e5aa0973
--- /dev/null
+++ b/modules/nf-server-filter/export/utils/getAllData.js
@@ -0,0 +1,107 @@
+import { PassThrough } from 'stream';
+import { buildQuery, esToSafeJsInt } from '@arranger/middleware';
+import { DOWNLOAD_STREAM_BUFFER_SIZE } from '../../config.js';
+import { graphql } from 'graphql';
+import { arrangerFilterDownload } from './arrangerFilterDownload.js';
+
+export function runQuery({ project_id, esClient, query, schema, variables }) {
+ return graphql({
+ schema,
+ contextValue: {
+ schema,
+ es: esClient,
+ projectId: project_id,
+ },
+ source: query,
+ variableValues: variables,
+ });
+}
+
+export async function getAllData({
+ req,
+ project_info,
+ params,
+ headers,
+ index,
+ columns = [],
+ sort = [],
+ sqon,
+ chunkSize = DOWNLOAD_STREAM_BUFFER_SIZE,
+ ...rest
+}) {
+ const stream = new PassThrough({ objectMode: true });
+ const toHits = ({
+ body: {
+ hits: { hits },
+ },
+ }) => hits;
+ const esSort = sort.map(({ field, order }) => ({ [field]: order })).concat({ _id: 'asc' });
+
+ const { esIndex, esType, extended } = await project_info.es
+ .search({
+ index: `arranger-projects-${project_info.projectId}`,
+ })
+ .then(toHits)
+ .then((hits) => hits.map(({ _source }) => _source))
+ .then(
+ (hits) =>
+ hits
+ .map(({ index: esIndex, name, esType, config: { extended } }) => ({
+ esIndex,
+ name,
+ esType,
+ extended,
+ }))
+ .filter(({ name }) => name === index)[0],
+ );
+
+ const nestedFields = extended.filter(({ type }) => type === 'nested').map(({ field }) => field);
+
+ const es_schema = project_info.arranger_schema['schema'];
+ sqon = arrangerFilterDownload(req, params, sqon);
+ const query = buildQuery({ nestedFields, filters: sqon });
+
+ runQuery({
+ project_id: project_info.projectId,
+ esClient: project_info.es,
+ query: `
+ query ($sqon: JSON) {
+ ${index} {
+ hits(filters: $sqon) {
+ total
+ }
+ }
+ }
+ `,
+ schema: es_schema,
+ variables: { sqon },
+ })
+ .then(({ data }) => {
+ const total = data[index].hits.total;
+ const steps = Array(Math.ceil(total / chunkSize)).fill();
+ // async reduce because each cycle is dependent on result of the previous
+ return steps.reduce(async (previous) => {
+ const previousHits = await previous;
+ const hits = await project_info.es
+ .search({
+ index: esIndex,
+ size: chunkSize,
+ body: {
+ sort: esSort,
+ ...(previousHits
+ ? {
+ search_after: previousHits[previousHits.length - 1].sort.map(esToSafeJsInt),
+ }
+ : {}),
+ ...(Object.entries(query).length ? { query } : {}),
+ },
+ })
+ .then(toHits);
+ stream.write({ hits, total });
+ return hits;
+ }, Promise.resolve());
+ })
+ .then(() => stream.end())
+ .catch(console.error);
+ return stream;
+}
diff --git a/modules/nf-server-filter/package-lock.json b/modules/nf-server-filter/package-lock.json
new file mode 100644
index 000000000..df1bbc1de
--- /dev/null
+++ b/modules/nf-server-filter/package-lock.json
@@ -0,0 +1,2753 @@
+{
+ "name": "server-filter",
+ "version": "1.0.0",
+ "lockfileVersion": 1,
+ "requires": true,
+ "dependencies": {
+ "@apollo/protobufjs": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@apollo/protobufjs/-/protobufjs-1.2.2.tgz",
+ "integrity": "sha512-vF+zxhPiLtkwxONs6YanSt1EpwpGilThpneExUN5K3tCymuxNnVq2yojTvnpRjv2QfsEIt/n7ozPIIzBLwGIDQ==",
+ "requires": {
+ "@protobufjs/aspromise": "^1.1.2",
+ "@protobufjs/base64": "^1.1.2",
+ "@protobufjs/codegen": "^2.0.4",
+ "@protobufjs/eventemitter": "^1.1.0",
+ "@protobufjs/fetch": "^1.1.0",
+ "@protobufjs/float": "^1.0.2",
+ "@protobufjs/inquire": "^1.1.0",
+ "@protobufjs/path": "^1.1.2",
+ "@protobufjs/pool": "^1.1.0",
+ "@protobufjs/utf8": "^1.1.0",
+ "@types/long": "^4.0.0",
+ "@types/node": "^10.1.0",
+ "long": "^4.0.0"
+ },
+ "dependencies": {
+ "@types/node": {
+ "version": "10.17.60",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.60.tgz",
+ "integrity": "sha512-F0KIgDJfy2nA3zMLmWGKxcH2ZVEtCZXHHdOQs2gSaQ27+lNeEfGxzkIw90aXswATX7AZ33tahPbzy6KAfUreVw=="
+ }
+ }
+ },
+ "@apollographql/apollo-tools": {
+ "version": "0.5.4",
+ "resolved": "https://registry.npmjs.org/@apollographql/apollo-tools/-/apollo-tools-0.5.4.tgz",
+ "integrity": "sha512-shM3q7rUbNyXVVRkQJQseXv6bnYM3BUma/eZhwXR4xsuM+bqWnJKvW7SAfRjP7LuSCocrexa5AXhjjawNHrIlw=="
+ },
+ "@apollographql/graphql-playground-html": {
+ "version": "1.6.27",
+ "resolved": "https://registry.npmjs.org/@apollographql/graphql-playground-html/-/graphql-playground-html-1.6.27.tgz",
+ "integrity": "sha512-tea2LweZvn6y6xFV11K0KC8ETjmm52mQrW+ezgB2O/aTQf8JGyFmMcRPFgUaQZeHbWdm8iisDC6EjOKsXu0nfw==",
+ "requires": {
+ "xss": "^1.0.8"
+ }
+ },
+ "@apollographql/graphql-upload-8-fork": {
+ "version": "8.1.4",
+ "resolved": "https://registry.npmjs.org/@apollographql/graphql-upload-8-fork/-/graphql-upload-8-fork-8.1.4.tgz",
+ "integrity": "sha512-lHAj/PUegYu02zza9Pg0bQQYH5I0ah1nyIzu2YIqOv41P0vu3GCBISAmQCfFHThK7N3dy7dLFPhoKcXlXRLPoQ==",
+ "requires": {
+ "@types/express": "*",
+ "@types/fs-capacitor": "^2.0.0",
+ "@types/koa": "*",
+ "busboy": "^0.3.1",
+ "fs-capacitor": "^2.0.4",
+ "http-errors": "^1.7.3",
+ "object-path": "^0.11.4"
+ },
+ "dependencies": {
+ "depd": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
+ "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ=="
+ },
+ "http-errors": {
+ "version": "1.8.1",
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.1.tgz",
+ "integrity": "sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g==",
+ "requires": {
+ "depd": "~1.1.2",
+ "inherits": "2.0.4",
+ "setprototypeof": "1.2.0",
+ "statuses": ">= 1.5.0 < 2",
+ "toidentifier": "1.0.1"
+ }
+ },
+ "statuses": {
+ "version": "1.5.0",
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
+ "integrity": "sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA=="
+ }
+ }
+ },
+ "@arranger/admin": {
+ "version": "2.19.4",
+ "resolved": "https://registry.npmjs.org/@arranger/admin/-/admin-2.19.4.tgz",
+ "integrity": "sha512-ATVOjiEfNkNuamYxZjlqonDksHInWYAwszeOKlKepp6Hf1cd5pB7uxtBPmC7fwVxdnUFCJG1Z+ggNX0VZaWE7g==",
+ "requires": {
+ "@arranger/mapping-utils": "^2.19.4",
+ "@arranger/schema": "^2.19.4",
+ "@elastic/elasticsearch": "7.5.0",
+ "apollo-link-http": "^1.5.5",
+ "apollo-server": "^2.14.2",
+ "apollo-server-express": "^2.14.2",
+ "convert-units": "^2.3.4",
+ "date-fns": "^1.29.0",
+ "express": "^4.16.3",
+ "graphql": "^14.5.3",
+ "graphql-tools": "^4.0.0",
+ "graphql-type-json": "^0.2.1",
+ "jwt-decode": "^2.2.0",
+ "lodash": "^4.17.20",
+ "node-fetch": "^2.2.0",
+ "qew": "^0.9.13",
+ "ramda": "^0.26.1",
+ "tslib": "^1.10.0"
+ },
+ "dependencies": {
+ "@elastic/elasticsearch": {
+ "version": "7.5.0",
+ "resolved": "https://registry.npmjs.org/@elastic/elasticsearch/-/elasticsearch-7.5.0.tgz",
+ "integrity": "sha512-ahzu451ppXepQMb3Zr8eFfWn8QR7yCcUWQjU1dS4X63Ctin0GNwwSPertt4WwDkm6MnZ25o932wAEgyMFLwzdA==",
+ "requires": {
+ "debug": "^4.1.1",
+ "decompress-response": "^4.2.0",
+ "into-stream": "^5.1.0",
+ "ms": "^2.1.1",
+ "once": "^1.4.0",
+ "pump": "^3.0.0"
+ }
+ },
+ "date-fns": {
+ "version": "1.30.1",
+ "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-1.30.1.tgz",
+ "integrity": "sha512-hBSVCvSmWC+QypYObzwGOd9wqdDpOt+0wl0KbU+R+uuZBS1jN8VsD1ss3irQDknRj5NvxiTF6oj/nDRnN/UQNw=="
+ },
+ "debug": {
+ "version": "4.3.4",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
+ "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
+ "requires": {
+ "ms": "2.1.2"
+ },
+ "dependencies": {
+ "ms": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
+ "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
+ }
+ }
+ },
+ "jwt-decode": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-2.2.0.tgz",
+ "integrity": "sha512-86GgN2vzfUu7m9Wcj63iUkuDzFNYFVmjeDm2GzWpUk+opB0pEpMsw6ePCMrhYkumz2C1ihqtZzOMAg7FiXcNoQ=="
+ },
+ "ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
+ }
+ }
+ },
+ "@arranger/mapping-utils": {
+ "version": "2.19.4",
+ "resolved": "https://registry.npmjs.org/@arranger/mapping-utils/-/mapping-utils-2.19.4.tgz",
+ "integrity": "sha512-DnJeH9HfB0aKUjFYRF6k+9p5cTj9aJEP6ZvFL1Gebn2G/KYKPGBG1PO8E3Vm7zDBhWJtgs/bKvonZxEJoZRKXg==",
+ "requires": {
+ "@arranger/middleware": "^2.19.4",
+ "@babel/polyfill": "^7.12.1",
+ "graphql-fields": "^2.0.3",
+ "kind-of": "^6.0.3",
+ "lodash": "^4.17.20",
+ "minimist": "^1.2.5",
+ "node-fetch": "^2.2.0",
+ "uuid": "^3.2.1"
+ }
+ },
+ "@arranger/middleware": {
+ "version": "2.19.4",
+ "resolved": "https://registry.npmjs.org/@arranger/middleware/-/middleware-2.19.4.tgz",
+ "integrity": "sha512-Vd5dWnTpC7uIT3YG9RkaP+pR4u5voQyxr3tfJVgG9lJay6zhaqAYRQvs3b17GB+W2Yj6ObKm6fsDY5Gtv+f4Cg==",
+ "requires": {
+ "body-parser": "^1.18.2",
+ "cors": "^2.8.4",
+ "date-fns": "^1.29.0",
+ "express": "^4.16.3",
+ "kind-of": "^6.0.3",
+ "lodash": "^4.17.20",
+ "minimist": "^1.2.5",
+ "morgan": "^1.9.0",
+ "utf8": "^3.0.0",
+ "winston": "^2.4.0"
+ },
+ "dependencies": {
+ "date-fns": {
+ "version": "1.30.1",
+ "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-1.30.1.tgz",
+ "integrity": "sha512-hBSVCvSmWC+QypYObzwGOd9wqdDpOt+0wl0KbU+R+uuZBS1jN8VsD1ss3irQDknRj5NvxiTF6oj/nDRnN/UQNw=="
+ }
+ }
+ },
+ "@arranger/schema": {
+ "version": "2.19.4",
+ "resolved": "https://registry.npmjs.org/@arranger/schema/-/schema-2.19.4.tgz",
+ "integrity": "sha512-9EwCm+Mg0uhnqLJs3khHmxJl/LwQey/Sf4VKqVqqAdLEqba5G94yeKpUUfWPwJHOsJnJ/pwyIcjW56GTvT3Rvg==",
+ "requires": {
+ "@arranger/mapping-utils": "^2.19.4",
+ "@arranger/middleware": "^2.19.4",
+ "@babel/polyfill": "^7.12.1",
+ "graphql-middleware": "1.3.1",
+ "graphql-scalars": "^0.1.5",
+ "graphql-tools": "^4.0.0",
+ "graphql-type-json": "^0.2.1",
+ "kind-of": "^6.0.3",
+ "lodash": "^4.17.20",
+ "minimist": "^1.2.5",
+ "paralleljs": "^0.2.1",
+ "uuid": "^3.2.1"
+ }
+ },
+ "@arranger/server": {
+ "version": "2.19.3",
+ "resolved": "https://registry.npmjs.org/@arranger/server/-/server-2.19.3.tgz",
+ "integrity": "sha512-BbeC/w7ULifmQ9y6Fa4iiWhIp9ww0jVFuFdXx1auZ1o1SnUK4jf0hc2M9ix2g6VdrW+qwiWUJGKzdSAeeyYXSA==",
+ "requires": {
+ "@arranger/admin": "^2.19.3",
+ "@arranger/mapping-utils": "^2.19.3",
+ "@arranger/middleware": "^2.19.3",
+ "@arranger/schema": "^2.19.3",
+ "@babel/polyfill": "^7.12.1",
+ "@elastic/elasticsearch": "^7.5.0",
+ "apollo-server-core": "^2.14.2",
+ "apollo-server-express": "^2.14.2",
+ "body-parser": "^1.18.2",
+ "chalk": "^2.3.1",
+ "cors": "^2.8.4",
+ "date-fns": "^2.29.2",
+ "dotenv": "^5.0.0",
+ "express": "^4.16.3",
+ "graphql": "^14.5.3",
+ "graphql-playground-middleware-express": "^1.7.12",
+ "jsonpath": "^1.0.0",
+ "lodash": "^4.17.20",
+ "node-fetch": "^2.2.0",
+ "tar-stream": "^1.5.5",
+ "through2": "^2.0.3",
+ "url-join": "^4.0.0",
+ "uuid": "^3.2.1"
+ }
+ },
+ "@babel/polyfill": {
+ "version": "7.12.1",
+ "resolved": "https://registry.npmjs.org/@babel/polyfill/-/polyfill-7.12.1.tgz",
+ "integrity": "sha512-X0pi0V6gxLi6lFZpGmeNa4zxtwEmCs42isWLNjZZDE0Y8yVfgu0T2OAHlzBbdYlqbW/YXVvoBHpATEM+goCj8g==",
+ "requires": {
+ "core-js": "^2.6.5",
+ "regenerator-runtime": "^0.13.4"
+ }
+ },
+ "@elastic/elasticsearch": {
+ "version": "7.17.0",
+ "resolved": "https://registry.npmjs.org/@elastic/elasticsearch/-/elasticsearch-7.17.0.tgz",
+ "integrity": "sha512-5QLPCjd0uLmLj1lSuKSThjNpq39f6NmlTy9ROLFwG5gjyTgpwSqufDeYG/Fm43Xs05uF7WcscoO7eguI3HuuYA==",
+ "requires": {
+ "debug": "^4.3.1",
+ "hpagent": "^0.1.1",
+ "ms": "^2.1.3",
+ "secure-json-parse": "^2.4.0"
+ },
+ "dependencies": {
+ "debug": {
+ "version": "4.3.4",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
+ "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
+ "requires": {
+ "ms": "2.1.2"
+ },
+ "dependencies": {
+ "ms": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
+ "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
+ }
+ }
+ },
+ "ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
+ }
+ }
+ },
+ "@josephg/resolvable": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/@josephg/resolvable/-/resolvable-1.0.1.tgz",
+ "integrity": "sha512-CtzORUwWTTOTqfVtHaKRJ0I1kNQd1bpn3sUh8I3nJDVY+5/M/Oe1DnEWzPQvqq/xPIIkzzzIP7mfCoAjFRvDhg=="
+ },
+ "@protobufjs/aspromise": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/aspromise/-/aspromise-1.1.2.tgz",
+ "integrity": "sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ=="
+ },
+ "@protobufjs/base64": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/base64/-/base64-1.1.2.tgz",
+ "integrity": "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg=="
+ },
+ "@protobufjs/codegen": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/@protobufjs/codegen/-/codegen-2.0.4.tgz",
+ "integrity": "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg=="
+ },
+ "@protobufjs/eventemitter": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz",
+ "integrity": "sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q=="
+ },
+ "@protobufjs/fetch": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/fetch/-/fetch-1.1.0.tgz",
+ "integrity": "sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ==",
+ "requires": {
+ "@protobufjs/aspromise": "^1.1.1",
+ "@protobufjs/inquire": "^1.1.0"
+ }
+ },
+ "@protobufjs/float": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/float/-/float-1.0.2.tgz",
+ "integrity": "sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ=="
+ },
+ "@protobufjs/inquire": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/inquire/-/inquire-1.1.0.tgz",
+ "integrity": "sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q=="
+ },
+ "@protobufjs/path": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/path/-/path-1.1.2.tgz",
+ "integrity": "sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA=="
+ },
+ "@protobufjs/pool": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/pool/-/pool-1.1.0.tgz",
+ "integrity": "sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw=="
+ },
+ "@protobufjs/utf8": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/utf8/-/utf8-1.1.0.tgz",
+ "integrity": "sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw=="
+ },
+ "@types/accepts": {
+ "version": "1.3.5",
+ "resolved": "https://registry.npmjs.org/@types/accepts/-/accepts-1.3.5.tgz",
+ "integrity": "sha512-jOdnI/3qTpHABjM5cx1Hc0sKsPoYCp+DP/GJRGtDlPd7fiV9oXGGIcjW/ZOxLIvjGz8MA+uMZI9metHlgqbgwQ==",
+ "requires": {
+ "@types/node": "*"
+ }
+ },
+ "@types/body-parser": {
+ "version": "1.19.2",
+ "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.2.tgz",
+ "integrity": "sha512-ALYone6pm6QmwZoAgeyNksccT9Q4AWZQ6PvfwR37GT6r6FWUPguq6sUmNGSMV2Wr761oQoBxwGGa6DR5o1DC9g==",
+ "requires": {
+ "@types/connect": "*",
+ "@types/node": "*"
+ }
+ },
+ "@types/connect": {
+ "version": "3.4.35",
+ "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.35.tgz",
+ "integrity": "sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==",
+ "requires": {
+ "@types/node": "*"
+ }
+ },
+ "@types/content-disposition": {
+ "version": "0.5.5",
+ "resolved": "https://registry.npmjs.org/@types/content-disposition/-/content-disposition-0.5.5.tgz",
+ "integrity": "sha512-v6LCdKfK6BwcqMo+wYW05rLS12S0ZO0Fl4w1h4aaZMD7bqT3gVUns6FvLJKGZHQmYn3SX55JWGpziwJRwVgutA=="
+ },
+ "@types/cookies": {
+ "version": "0.7.7",
+ "resolved": "https://registry.npmjs.org/@types/cookies/-/cookies-0.7.7.tgz",
+ "integrity": "sha512-h7BcvPUogWbKCzBR2lY4oqaZbO3jXZksexYJVFvkrFeLgbZjQkU4x8pRq6eg2MHXQhY0McQdqmmsxRWlVAHooA==",
+ "requires": {
+ "@types/connect": "*",
+ "@types/express": "*",
+ "@types/keygrip": "*",
+ "@types/node": "*"
+ }
+ },
+ "@types/cors": {
+ "version": "2.8.10",
+ "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.10.tgz",
+ "integrity": "sha512-C7srjHiVG3Ey1nR6d511dtDkCEjxuN9W1HWAEjGq8kpcwmNM6JJkpC0xvabM7BXTG2wDq8Eu33iH9aQKa7IvLQ=="
+ },
+ "@types/express": {
+ "version": "4.17.17",
+ "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.17.tgz",
+ "integrity": "sha512-Q4FmmuLGBG58btUnfS1c1r/NQdlp3DMfGDGig8WhfpA2YRUtEkxAjkZb0yvplJGYdF1fsQ81iMDcH24sSCNC/Q==",
+ "requires": {
+ "@types/body-parser": "*",
+ "@types/express-serve-static-core": "^4.17.33",
+ "@types/qs": "*",
+ "@types/serve-static": "*"
+ },
+ "dependencies": {
+ "@types/express-serve-static-core": {
+ "version": "4.17.33",
+ "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.33.tgz",
+ "integrity": "sha512-TPBqmR/HRYI3eC2E5hmiivIzv+bidAfXofM+sbonAGvyDhySGw9/PQZFt2BLOrjUUR++4eJVpx6KnLQK1Fk9tA==",
+ "requires": {
+ "@types/node": "*",
+ "@types/qs": "*",
+ "@types/range-parser": "*"
+ }
+ }
+ }
+ },
+ "@types/express-serve-static-core": {
+ "version": "4.17.24",
+ "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.24.tgz",
+ "integrity": "sha512-3UJuW+Qxhzwjq3xhwXm2onQcFHn76frIYVbTu+kn24LFxI+dEhdfISDFovPB8VpEgW8oQCTpRuCe+0zJxB7NEA==",
+ "requires": {
+ "@types/node": "*",
+ "@types/qs": "*",
+ "@types/range-parser": "*"
+ }
+ },
+ "@types/fs-capacitor": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/@types/fs-capacitor/-/fs-capacitor-2.0.0.tgz",
+ "integrity": "sha512-FKVPOCFbhCvZxpVAMhdBdTfVfXUpsh15wFHgqOKxh9N9vzWZVuWCSijZ5T4U34XYNnuj2oduh6xcs1i+LPI+BQ==",
+ "requires": {
+ "@types/node": "*"
+ }
+ },
+ "@types/http-assert": {
+ "version": "1.5.3",
+ "resolved": "https://registry.npmjs.org/@types/http-assert/-/http-assert-1.5.3.tgz",
+ "integrity": "sha512-FyAOrDuQmBi8/or3ns4rwPno7/9tJTijVW6aQQjK02+kOQ8zmoNg2XJtAuQhvQcy1ASJq38wirX5//9J1EqoUA=="
+ },
+ "@types/http-errors": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.1.tgz",
+ "integrity": "sha512-/K3ds8TRAfBvi5vfjuz8y6+GiAYBZ0x4tXv1Av6CWBWn0IlADc+ZX9pMq7oU0fNQPnBwIZl3rmeLp6SBApbxSQ=="
+ },
+ "@types/keygrip": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/@types/keygrip/-/keygrip-1.0.2.tgz",
+ "integrity": "sha512-GJhpTepz2udxGexqos8wgaBx4I/zWIDPh/KOGEwAqtuGDkOUJu5eFvwmdBX4AmB8Odsr+9pHCQqiAqDL/yKMKw=="
+ },
+ "@types/koa": {
+ "version": "2.13.5",
+ "resolved": "https://registry.npmjs.org/@types/koa/-/koa-2.13.5.tgz",
+ "integrity": "sha512-HSUOdzKz3by4fnqagwthW/1w/yJspTgppyyalPVbgZf8jQWvdIXcVW5h2DGtw4zYntOaeRGx49r1hxoPWrD4aA==",
+ "requires": {
+ "@types/accepts": "*",
+ "@types/content-disposition": "*",
+ "@types/cookies": "*",
+ "@types/http-assert": "*",
+ "@types/http-errors": "*",
+ "@types/keygrip": "*",
+ "@types/koa-compose": "*",
+ "@types/node": "*"
+ }
+ },
+ "@types/koa-compose": {
+ "version": "3.2.5",
+ "resolved": "https://registry.npmjs.org/@types/koa-compose/-/koa-compose-3.2.5.tgz",
+ "integrity": "sha512-B8nG/OoE1ORZqCkBVsup/AKcvjdgoHnfi4pZMn5UwAPCbhk/96xyv284eBYW8JlQbQ7zDmnpFr68I/40mFoIBQ==",
+ "requires": {
+ "@types/koa": "*"
+ }
+ },
+ "@types/long": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/@types/long/-/long-4.0.2.tgz",
+ "integrity": "sha512-MqTGEo5bj5t157U6fA/BiDynNkn0YknVdh48CMPkTSpFTVmvao5UQmm7uEF6xBEo7qIMAlY/JSleYaE6VOdpaA=="
+ },
+ "@types/mime": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/@types/mime/-/mime-3.0.1.tgz",
+ "integrity": "sha512-Y4XFY5VJAuw0FgAqPNd6NNoV44jbq9Bz2L7Rh/J6jLTiHBSBJa9fxqQIvkIld4GsoDOcCbvzOUAbLPsSKKg+uA=="
+ },
+ "@types/node": {
+ "version": "18.13.0",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-18.13.0.tgz",
+ "integrity": "sha512-gC3TazRzGoOnoKAhUx+Q0t8S9Tzs74z7m0ipwGpSqQrleP14hKxP4/JUeEQcD3W1/aIpnWl8pHowI7WokuZpXg=="
+ },
+ "@types/qs": {
+ "version": "6.9.7",
+ "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz",
+ "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw=="
+ },
+ "@types/range-parser": {
+ "version": "1.2.4",
+ "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.4.tgz",
+ "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw=="
+ },
+ "@types/serve-static": {
+ "version": "1.15.0",
+ "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.0.tgz",
+ "integrity": "sha512-z5xyF6uh8CbjAu9760KDKsH2FcDxZ2tFCsA4HIMWE6IkiYMXfVoa+4f9KX+FN0ZLsaMw1WNG2ETLA6N+/YA+cg==",
+ "requires": {
+ "@types/mime": "*",
+ "@types/node": "*"
+ }
+ },
+ "@types/ws": {
+ "version": "7.4.7",
+ "resolved": "https://registry.npmjs.org/@types/ws/-/ws-7.4.7.tgz",
+ "integrity": "sha512-JQbbmxZTZehdc2iszGKs5oC3NFnjeay7mtAWrdt7qNtAVK0g19muApzAy4bm9byz79xa2ZnO/BOBC2R8RC5Lww==",
+ "requires": {
+ "@types/node": "*"
+ }
+ },
+ "@wry/equality": {
+ "version": "0.1.11",
+ "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.1.11.tgz",
+ "integrity": "sha512-mwEVBDUVODlsQQ5dfuLUS5/Tf7jqUKyhKYHmVi4fPB6bDMOfWvUPJmKgS1Z7Za/sOI3vzWt4+O7yCiL/70MogA==",
+ "requires": {
+ "tslib": "^1.9.3"
+ }
+ },
+ "accepts": {
+ "version": "1.3.8",
+ "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
+ "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
+ "requires": {
+ "mime-types": "~2.1.34",
+ "negotiator": "0.6.3"
+ }
+ },
+ "ansi-styles": {
+ "version": "3.2.1",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+ "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+ "requires": {
+ "color-convert": "^1.9.0"
+ }
+ },
+ "apollo-cache-control": {
+ "version": "0.15.0",
+ "resolved": "https://registry.npmjs.org/apollo-cache-control/-/apollo-cache-control-0.15.0.tgz",
+ "integrity": "sha512-U2uYvHZsWmR6s6CD5zlq3PepfbUAM8953CeVM2Y2QYMtJ8i4CYplEPbIWb3zTIXSPbIPeWGddM56pChI6Iz3zA==",
+ "requires": {
+ "apollo-server-env": "^3.2.0",
+ "apollo-server-plugin-base": "^0.14.0"
+ }
+ },
+ "apollo-datasource": {
+ "version": "0.10.0",
+ "resolved": "https://registry.npmjs.org/apollo-datasource/-/apollo-datasource-0.10.0.tgz",
+ "integrity": "sha512-wrLhuoM2MtA0KA0+3qyioe0H2FjAxjTvuFOlNCk6WberA887m0MQlWULZImCWTkKuN+zEAMerHfxN+F+W8+lBA==",
+ "requires": {
+ "apollo-server-caching": "^0.7.0",
+ "apollo-server-env": "^3.2.0"
+ }
+ },
+ "apollo-graphql": {
+ "version": "0.9.7",
+ "resolved": "https://registry.npmjs.org/apollo-graphql/-/apollo-graphql-0.9.7.tgz",
+ "integrity": "sha512-bezL9ItUWUGHTm1bI/XzIgiiZbhXpsC7uxk4UxFPmcVJwJsDc3ayZ99oXxAaK+3Rbg/IoqrHckA6CwmkCsbaSA==",
+ "requires": {
+ "core-js-pure": "^3.10.2",
+ "lodash.sortby": "^4.7.0",
+ "sha.js": "^2.4.11"
+ }
+ },
+ "apollo-link": {
+ "version": "1.2.14",
+ "resolved": "https://registry.npmjs.org/apollo-link/-/apollo-link-1.2.14.tgz",
+ "integrity": "sha512-p67CMEFP7kOG1JZ0ZkYZwRDa369w5PIjtMjvrQd/HnIV8FRsHRqLqK+oAZQnFa1DDdZtOtHTi+aMIW6EatC2jg==",
+ "requires": {
+ "apollo-utilities": "^1.3.0",
+ "ts-invariant": "^0.4.0",
+ "tslib": "^1.9.3",
+ "zen-observable-ts": "^0.8.21"
+ }
+ },
+ "apollo-link-http": {
+ "version": "1.5.17",
+ "resolved": "https://registry.npmjs.org/apollo-link-http/-/apollo-link-http-1.5.17.tgz",
+ "integrity": "sha512-uWcqAotbwDEU/9+Dm9e1/clO7hTB2kQ/94JYcGouBVLjoKmTeJTUPQKcJGpPwUjZcSqgYicbFqQSoJIW0yrFvg==",
+ "requires": {
+ "apollo-link": "^1.2.14",
+ "apollo-link-http-common": "^0.2.16",
+ "tslib": "^1.9.3"
+ }
+ },
+ "apollo-link-http-common": {
+ "version": "0.2.16",
+ "resolved": "https://registry.npmjs.org/apollo-link-http-common/-/apollo-link-http-common-0.2.16.tgz",
+ "integrity": "sha512-2tIhOIrnaF4UbQHf7kjeQA/EmSorB7+HyJIIrUjJOKBgnXwuexi8aMecRlqTIDWcyVXCeqLhUnztMa6bOH/jTg==",
+ "requires": {
+ "apollo-link": "^1.2.14",
+ "ts-invariant": "^0.4.0",
+ "tslib": "^1.9.3"
+ }
+ },
+ "apollo-reporting-protobuf": {
+ "version": "0.8.0",
+ "resolved": "https://registry.npmjs.org/apollo-reporting-protobuf/-/apollo-reporting-protobuf-0.8.0.tgz",
+ "integrity": "sha512-B3XmnkH6Y458iV6OsA7AhfwvTgeZnFq9nPVjbxmLKnvfkEl8hYADtz724uPa0WeBiD7DSFcnLtqg9yGmCkBohg==",
+ "requires": {
+ "@apollo/protobufjs": "1.2.2"
+ }
+ },
+ "apollo-server": {
+ "version": "2.26.1",
+ "resolved": "https://registry.npmjs.org/apollo-server/-/apollo-server-2.26.1.tgz",
+ "integrity": "sha512-aHRfp6qTCZgpD9YQAlb99js/leSthDp3/gNSlfR68R69kSxVoKMeabwEbdFr1zJKi45u9riausqh75rRaA/7AQ==",
+ "requires": {
+ "apollo-server-core": "^2.26.1",
+ "apollo-server-express": "^2.26.1",
+ "express": "^4.0.0",
+ "graphql-subscriptions": "^1.0.0",
+ "graphql-tools": "^4.0.8",
+ "stoppable": "^1.1.0"
+ }
+ },
+ "apollo-server-caching": {
+ "version": "0.7.0",
+ "resolved": "https://registry.npmjs.org/apollo-server-caching/-/apollo-server-caching-0.7.0.tgz",
+ "integrity": "sha512-MsVCuf/2FxuTFVhGLK13B+TZH9tBd2qkyoXKKILIiGcZ5CDUEBO14vIV63aNkMkS1xxvK2U4wBcuuNj/VH2Mkw==",
+ "requires": {
+ "lru-cache": "^6.0.0"
+ }
+ },
+ "apollo-server-core": {
+ "version": "2.26.1",
+ "resolved": "https://registry.npmjs.org/apollo-server-core/-/apollo-server-core-2.26.1.tgz",
+ "integrity": "sha512-YnO1YXhHOnCY7Q2SZ0uUtPq6SLCw+t2uI19l59mzWuCyZYdHrtSy3zUEU6pM3tR9vvUuRGkYIfMRlo/Q8a1U5g==",
+ "requires": {
+ "@apollographql/apollo-tools": "^0.5.0",
+ "@apollographql/graphql-playground-html": "1.6.27",
+ "@apollographql/graphql-upload-8-fork": "^8.1.4",
+ "@josephg/resolvable": "^1.0.0",
+ "@types/ws": "^7.0.0",
+ "apollo-cache-control": "^0.15.0",
+ "apollo-datasource": "^0.10.0",
+ "apollo-graphql": "^0.9.0",
+ "apollo-reporting-protobuf": "^0.8.0",
+ "apollo-server-caching": "^0.7.0",
+ "apollo-server-env": "^3.2.0",
+ "apollo-server-errors": "^2.5.0",
+ "apollo-server-plugin-base": "^0.14.0",
+ "apollo-server-types": "^0.10.0",
+ "apollo-tracing": "^0.16.0",
+ "async-retry": "^1.2.1",
+ "fast-json-stable-stringify": "^2.0.0",
+ "graphql-extensions": "^0.16.0",
+ "graphql-tag": "^2.11.0",
+ "graphql-tools": "^4.0.8",
+ "loglevel": "^1.6.7",
+ "lru-cache": "^6.0.0",
+ "sha.js": "^2.4.11",
+ "subscriptions-transport-ws": "^0.9.19",
+ "uuid": "^8.0.0"
+ },
+ "dependencies": {
+ "uuid": {
+ "version": "8.3.2",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
+ "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg=="
+ }
+ }
+ },
+ "apollo-server-env": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/apollo-server-env/-/apollo-server-env-3.2.0.tgz",
+ "integrity": "sha512-V+kO5e6vUo2JwqV1/Ng71ZE3J6x1hCOC+nID2/++bCYl0/fPY9iLChbBNSgN/uoFcjhgmBchOv+m4o0Nie/TFQ==",
+ "requires": {
+ "node-fetch": "^2.6.1",
+ "util.promisify": "^1.0.0"
+ }
+ },
+ "apollo-server-errors": {
+ "version": "2.5.0",
+ "resolved": "https://registry.npmjs.org/apollo-server-errors/-/apollo-server-errors-2.5.0.tgz",
+ "integrity": "sha512-lO5oTjgiC3vlVg2RKr3RiXIIQ5pGXBFxYGGUkKDhTud3jMIhs+gel8L8zsEjKaKxkjHhCQAA/bcEfYiKkGQIvA=="
+ },
+ "apollo-server-express": {
+ "version": "2.26.1",
+ "resolved": "https://registry.npmjs.org/apollo-server-express/-/apollo-server-express-2.26.1.tgz",
+ "integrity": "sha512-eATTtlGhZFuo4KNRgaQ25jflUchI18oMd0vZyx0uIQ/CM0FPttO1noQ0fPAO6U0oSrxS8J9fCh8naJFDTUsZ0w==",
+ "requires": {
+ "@apollographql/graphql-playground-html": "1.6.27",
+ "@types/accepts": "^1.3.5",
+ "@types/body-parser": "1.19.0",
+ "@types/cors": "2.8.10",
+ "@types/express": "^4.17.12",
+ "@types/express-serve-static-core": "^4.17.21",
+ "accepts": "^1.3.5",
+ "apollo-server-core": "^2.26.1",
+ "apollo-server-types": "^0.10.0",
+ "body-parser": "^1.18.3",
+ "cors": "^2.8.5",
+ "express": "^4.17.1",
+ "graphql-subscriptions": "^1.0.0",
+ "graphql-tools": "^4.0.8",
+ "parseurl": "^1.3.2",
+ "subscriptions-transport-ws": "^0.9.19",
+ "type-is": "^1.6.16"
+ },
+ "dependencies": {
+ "@types/body-parser": {
+ "version": "1.19.0",
+ "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.0.tgz",
+ "integrity": "sha512-W98JrE0j2K78swW4ukqMleo8R7h/pFETjM2DQ90MF6XK2i4LO4W3gQ71Lt4w3bfm2EvVSyWHplECvB5sK22yFQ==",
+ "requires": {
+ "@types/connect": "*",
+ "@types/node": "*"
+ }
+ }
+ }
+ },
+ "apollo-server-plugin-base": {
+ "version": "0.14.0",
+ "resolved": "https://registry.npmjs.org/apollo-server-plugin-base/-/apollo-server-plugin-base-0.14.0.tgz",
+ "integrity": "sha512-nTNSFuBhZURGjtWptdVqwemYUOdsvABj/GSKzeNvepiEubiv4N0rt4Gvy1inHDiMbo98wQTdF/7XohNcB9A77g==",
+ "requires": {
+ "apollo-server-types": "^0.10.0"
+ }
+ },
+ "apollo-server-types": {
+ "version": "0.10.0",
+ "resolved": "https://registry.npmjs.org/apollo-server-types/-/apollo-server-types-0.10.0.tgz",
+ "integrity": "sha512-LsB3epw1X3Co/HGiKHCGtzWG35J59gG8Ypx0p22+wgdM9AVDm1ylsNGZy+osNIVJc1lUJf3nF5kZ90vA866K/w==",
+ "requires": {
+ "apollo-reporting-protobuf": "^0.8.0",
+ "apollo-server-caching": "^0.7.0",
+ "apollo-server-env": "^3.2.0"
+ }
+ },
+ "apollo-tracing": {
+ "version": "0.16.0",
+ "resolved": "https://registry.npmjs.org/apollo-tracing/-/apollo-tracing-0.16.0.tgz",
+ "integrity": "sha512-Oy8kTggB+fJ/hHXwHyMpuTl5KW7u1XetKFDErZVOobUKc2zjc/NgWiC/s7SGYZCgfLodBjvwfa6rMcvLkz7c0w==",
+ "requires": {
+ "apollo-server-env": "^3.2.0",
+ "apollo-server-plugin-base": "^0.14.0"
+ }
+ },
+ "apollo-utilities": {
+ "version": "1.3.4",
+ "resolved": "https://registry.npmjs.org/apollo-utilities/-/apollo-utilities-1.3.4.tgz",
+ "integrity": "sha512-pk2hiWrCXMAy2fRPwEyhvka+mqwzeP60Jr1tRYi5xru+3ko94HI9o6lK0CT33/w4RDlxWchmdhDCrvdr+pHCig==",
+ "requires": {
+ "@wry/equality": "^0.1.2",
+ "fast-json-stable-stringify": "^2.0.0",
+ "ts-invariant": "^0.4.0",
+ "tslib": "^1.10.0"
+ }
+ },
+ "array-flatten": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
+ "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg=="
+ },
+ "array.prototype.reduce": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/array.prototype.reduce/-/array.prototype.reduce-1.0.5.tgz",
+ "integrity": "sha512-kDdugMl7id9COE8R7MHF5jWk7Dqt/fs4Pv+JXoICnYwqpjjjbUurz6w5fT5IG6brLdJhv6/VoHB0H7oyIBXd+Q==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.4",
+ "es-abstract": "^1.20.4",
+ "es-array-method-boxes-properly": "^1.0.0",
+ "is-string": "^1.0.7"
+ }
+ },
+ "async": {
+ "version": "2.6.4",
+ "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz",
+ "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==",
+ "requires": {
+ "lodash": "^4.17.14"
+ }
+ },
+ "async-retry": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/async-retry/-/async-retry-1.3.3.tgz",
+ "integrity": "sha512-wfr/jstw9xNi/0teMHrRW7dsz3Lt5ARhYNZ2ewpadnhaIp5mbALhOAP+EAdsC7t4Z6wqsDVv9+W6gm1Dk9mEyw==",
+ "requires": {
+ "retry": "0.13.1"
+ }
+ },
+ "asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="
+ },
+ "available-typed-arrays": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz",
+ "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw=="
+ },
+ "axios": {
+ "version": "1.3.4",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.3.4.tgz",
+ "integrity": "sha512-toYm+Bsyl6VC5wSkfkbbNB6ROv7KY93PEBBL6xyDczaIHasAiv4wPqQ/c4RjoQzipxRD2W5g21cOqQulZ7rHwQ==",
+ "requires": {
+ "follow-redirects": "^1.15.0",
+ "form-data": "^4.0.0",
+ "proxy-from-env": "^1.1.0"
+ }
+ },
+ "backo2": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz",
+ "integrity": "sha512-zj6Z6M7Eq+PBZ7PQxl5NT665MvJdAkzp0f60nAJ+sLaSCBPMwVak5ZegFbgVCzFcCJTKFoMizvM5Ld7+JrRJHA=="
+ },
+ "basic-auth": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz",
+ "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==",
+ "requires": {
+ "safe-buffer": "5.1.2"
+ },
+ "dependencies": {
+ "safe-buffer": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+ "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
+ }
+ }
+ },
+ "bl": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmjs.org/bl/-/bl-1.2.3.tgz",
+ "integrity": "sha512-pvcNpa0UU69UT341rO6AYy4FVAIkUHuZXRIWbq+zHnsVcRzDDjIAhGuuYoi0d//cwIwtt4pkpKycWEfjdV+vww==",
+ "requires": {
+ "readable-stream": "^2.3.5",
+ "safe-buffer": "^5.1.1"
+ }
+ },
+ "body-parser": {
+ "version": "1.20.1",
+ "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.1.tgz",
+ "integrity": "sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==",
+ "requires": {
+ "bytes": "3.1.2",
+ "content-type": "~1.0.4",
+ "debug": "2.6.9",
+ "depd": "2.0.0",
+ "destroy": "1.2.0",
+ "http-errors": "2.0.0",
+ "iconv-lite": "0.4.24",
+ "on-finished": "2.4.1",
+ "qs": "6.11.0",
+ "raw-body": "2.5.1",
+ "type-is": "~1.6.18",
+ "unpipe": "1.0.0"
+ }
+ },
+ "buffer-alloc": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/buffer-alloc/-/buffer-alloc-1.2.0.tgz",
+ "integrity": "sha512-CFsHQgjtW1UChdXgbyJGtnm+O/uLQeZdtbDo8mfUgYXCHSM1wgrVxXm6bSyrUuErEb+4sYVGCzASBRot7zyrow==",
+ "requires": {
+ "buffer-alloc-unsafe": "^1.1.0",
+ "buffer-fill": "^1.0.0"
+ }
+ },
+ "buffer-alloc-unsafe": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/buffer-alloc-unsafe/-/buffer-alloc-unsafe-1.1.0.tgz",
+ "integrity": "sha512-TEM2iMIEQdJ2yjPJoSIsldnleVaAk1oW3DBVUykyOLsEsFmEc9kn+SFFPz+gl54KQNxlDnAwCXosOS9Okx2xAg=="
+ },
+ "buffer-fill": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/buffer-fill/-/buffer-fill-1.0.0.tgz",
+ "integrity": "sha512-T7zexNBwiiaCOGDg9xNX9PBmjrubblRkENuptryuI64URkXDFum9il/JGL8Lm8wYfAXpredVXXZz7eMHilimiQ=="
+ },
+ "busboy": {
+ "version": "0.3.1",
+ "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.3.1.tgz",
+ "integrity": "sha512-y7tTxhGKXcyBxRKAni+awqx8uqaJKrSFSNFSeRG5CsWNdmy2BIK+6VGWEW7TZnIO/533mtMEA4rOevQV815YJw==",
+ "requires": {
+ "dicer": "0.3.0"
+ }
+ },
+ "bytes": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
+ "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg=="
+ },
+ "call-bind": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz",
+ "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==",
+ "requires": {
+ "function-bind": "^1.1.1",
+ "get-intrinsic": "^1.0.2"
+ }
+ },
+ "chalk": {
+ "version": "2.4.2",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+ "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+ "requires": {
+ "ansi-styles": "^3.2.1",
+ "escape-string-regexp": "^1.0.5",
+ "supports-color": "^5.3.0"
+ }
+ },
+ "color-convert": {
+ "version": "1.9.3",
+ "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+ "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+ "requires": {
+ "color-name": "1.1.3"
+ }
+ },
+ "color-name": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+ "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw=="
+ },
+ "colors": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/colors/-/colors-1.0.3.tgz",
+ "integrity": "sha512-pFGrxThWcWQ2MsAz6RtgeWe4NK2kUE1WfsrvvlctdII745EW9I0yflqhe7++M5LEc7bV2c/9/5zc8sFcpL0Drw=="
+ },
+ "combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "requires": {
+ "delayed-stream": "~1.0.0"
+ }
+ },
+ "commander": {
+ "version": "2.20.3",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz",
+ "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ=="
+ },
+ "content-disposition": {
+ "version": "0.5.4",
+ "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
+ "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
+ "requires": {
+ "safe-buffer": "5.2.1"
+ }
+ },
+ "content-type": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
+ "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
+ },
+ "convert-units": {
+ "version": "2.3.4",
+ "resolved": "https://registry.npmjs.org/convert-units/-/convert-units-2.3.4.tgz",
+ "integrity": "sha512-ERHfdA0UhHJp1IpwE6PnFJx8LqG7B1ZjJ20UvVCmopEnVCfER68Tbe3kvN63dLbYXDA2xFWRE6zd4Wsf0w7POg==",
+ "requires": {
+ "lodash.foreach": "2.3.x",
+ "lodash.keys": "2.3.x"
+ }
+ },
+ "cookie": {
+ "version": "0.5.0",
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz",
+ "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw=="
+ },
+ "cookie-signature": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
+ "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ=="
+ },
+ "core-js": {
+ "version": "2.6.12",
+ "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.12.tgz",
+ "integrity": "sha512-Kb2wC0fvsWfQrgk8HU5lW6U/Lcs8+9aaYcy4ZFc6DDlo4nZ7n70dEgE5rtR0oG6ufKDUnrwfWL1mXR5ljDatrQ=="
+ },
+ "core-js-pure": {
+ "version": "3.27.2",
+ "resolved": "https://registry.npmjs.org/core-js-pure/-/core-js-pure-3.27.2.tgz",
+ "integrity": "sha512-Cf2jqAbXgWH3VVzjyaaFkY1EBazxugUepGymDoeteyYr9ByX51kD2jdHZlsEF/xnJMyN3Prua7mQuzwMg6Zc9A=="
+ },
+ "core-util-is": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz",
+ "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ=="
+ },
+ "cors": {
+ "version": "2.8.5",
+ "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
+ "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
+ "requires": {
+ "object-assign": "^4",
+ "vary": "^1"
+ }
+ },
+ "cssfilter": {
+ "version": "0.0.10",
+ "resolved": "https://registry.npmjs.org/cssfilter/-/cssfilter-0.0.10.tgz",
+ "integrity": "sha512-FAaLDaplstoRsDR8XGYH51znUN0UY7nMc6Z9/fvE8EXGwvJE9hu7W2vHwx1+bd6gCYnln9nLbzxFTrcO9YQDZw=="
+ },
+ "cycle": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/cycle/-/cycle-1.0.3.tgz",
+ "integrity": "sha512-TVF6svNzeQCOpjCqsy0/CSy8VgObG3wXusJ73xW2GbG5rGx7lC8zxDSURicsXI2UsGdi2L0QNRCi745/wUDvsA=="
+ },
+ "d": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz",
+ "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==",
+ "requires": {
+ "es5-ext": "^0.10.50",
+ "type": "^1.0.1"
+ }
+ },
+ "date-fns": {
+ "version": "2.29.3",
+ "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.29.3.tgz",
+ "integrity": "sha512-dDCnyH2WnnKusqvZZ6+jA1O51Ibt8ZMRNkDZdyAyK4YfbDwa/cEmuztzG5pk6hqlp9aSBPYcjOlktquahGwGeA=="
+ },
+ "debug": {
+ "version": "2.6.9",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+ "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+ "requires": {
+ "ms": "2.0.0"
+ }
+ },
+ "decompress-response": {
+ "version": "4.2.1",
+ "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-4.2.1.tgz",
+ "integrity": "sha512-jOSne2qbyE+/r8G1VU+G/82LBs2Fs4LAsTiLSHOCOMZQl2OKZ6i8i4IyHemTe+/yIXOtTcRQMzPcgyhoFlqPkw==",
+ "requires": {
+ "mimic-response": "^2.0.0"
+ }
+ },
+ "deep-is": {
+ "version": "0.1.4",
+ "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
+ "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ=="
+ },
+ "define-properties": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.4.tgz",
+ "integrity": "sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA==",
+ "requires": {
+ "has-property-descriptors": "^1.0.0",
+ "object-keys": "^1.1.1"
+ }
+ },
+ "delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ=="
+ },
+ "depd": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
+ "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw=="
+ },
+ "deprecated-decorator": {
+ "version": "0.1.6",
+ "resolved": "https://registry.npmjs.org/deprecated-decorator/-/deprecated-decorator-0.1.6.tgz",
+ "integrity": "sha512-MHidOOnCHGlZDKsI21+mbIIhf4Fff+hhCTB7gtVg4uoIqjcrTZc5v6M+GS2zVI0sV7PqK415rb8XaOSQsQkHOw=="
+ },
+ "destroy": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
+ "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg=="
+ },
+ "dicer": {
+ "version": "0.3.0",
+ "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.3.0.tgz",
+ "integrity": "sha512-MdceRRWqltEG2dZqO769g27N/3PXfcKl04VhYnBlo2YhH7zPi88VebsjTKclaOyiuMaGU72hTfw3VkUitGcVCA==",
+ "requires": {
+ "streamsearch": "0.1.2"
+ }
+ },
+ "dotenv": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-5.0.1.tgz",
+ "integrity": "sha512-4As8uPrjfwb7VXC+WnLCbXK7y+Ueb2B3zgNCePYfhxS1PYeaO1YTeplffTEcbfLhvFNGLAz90VvJs9yomG7bow=="
+ },
+ "ee-first": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
+ "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="
+ },
+ "encodeurl": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
+ "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w=="
+ },
+ "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==",
+ "requires": {
+ "once": "^1.4.0"
+ }
+ },
+ "es-abstract": {
+ "version": "1.21.1",
+ "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.21.1.tgz",
+ "integrity": "sha512-QudMsPOz86xYz/1dG1OuGBKOELjCh99IIWHLzy5znUB6j8xG2yMA7bfTV86VSqKF+Y/H08vQPR+9jyXpuC6hfg==",
+ "requires": {
+ "available-typed-arrays": "^1.0.5",
+ "call-bind": "^1.0.2",
+ "es-set-tostringtag": "^2.0.1",
+ "es-to-primitive": "^1.2.1",
+ "function-bind": "^1.1.1",
+ "function.prototype.name": "^1.1.5",
+ "get-intrinsic": "^1.1.3",
+ "get-symbol-description": "^1.0.0",
+ "globalthis": "^1.0.3",
+ "gopd": "^1.0.1",
+ "has": "^1.0.3",
+ "has-property-descriptors": "^1.0.0",
+ "has-proto": "^1.0.1",
+ "has-symbols": "^1.0.3",
+ "internal-slot": "^1.0.4",
+ "is-array-buffer": "^3.0.1",
+ "is-callable": "^1.2.7",
+ "is-negative-zero": "^2.0.2",
+ "is-regex": "^1.1.4",
+ "is-shared-array-buffer": "^1.0.2",
+ "is-string": "^1.0.7",
+ "is-typed-array": "^1.1.10",
+ "is-weakref": "^1.0.2",
+ "object-inspect": "^1.12.2",
+ "object-keys": "^1.1.1",
+ "object.assign": "^4.1.4",
+ "regexp.prototype.flags": "^1.4.3",
+ "safe-regex-test": "^1.0.0",
+ "string.prototype.trimend": "^1.0.6",
+ "string.prototype.trimstart": "^1.0.6",
+ "typed-array-length": "^1.0.4",
+ "unbox-primitive": "^1.0.2",
+ "which-typed-array": "^1.1.9"
+ }
+ },
+ "es-array-method-boxes-properly": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/es-array-method-boxes-properly/-/es-array-method-boxes-properly-1.0.0.tgz",
+ "integrity": "sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA=="
+ },
+ "es-set-tostringtag": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz",
+ "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==",
+ "requires": {
+ "get-intrinsic": "^1.1.3",
+ "has": "^1.0.3",
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "es-to-primitive": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz",
+ "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==",
+ "requires": {
+ "is-callable": "^1.1.4",
+ "is-date-object": "^1.0.1",
+ "is-symbol": "^1.0.2"
+ }
+ },
+ "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==",
+ "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==",
+ "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==",
+ "requires": {
+ "d": "^1.0.1",
+ "ext": "^1.1.2"
+ }
+ },
+ "es6-weak-map": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.3.tgz",
+ "integrity": "sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==",
+ "requires": {
+ "d": "1",
+ "es5-ext": "^0.10.46",
+ "es6-iterator": "^2.0.3",
+ "es6-symbol": "^3.1.1"
+ }
+ },
+ "escape-html": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
+ "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="
+ },
+ "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=="
+ },
+ "escodegen": {
+ "version": "1.14.3",
+ "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.3.tgz",
+ "integrity": "sha512-qFcX0XJkdg+PB3xjZZG/wKSuT1PnQWx57+TVSjIMmILd2yC/6ByYElPwJnslDsuWuSAp4AwJGumarAAmJch5Kw==",
+ "requires": {
+ "esprima": "^4.0.1",
+ "estraverse": "^4.2.0",
+ "esutils": "^2.0.2",
+ "optionator": "^0.8.1",
+ "source-map": "~0.6.1"
+ },
+ "dependencies": {
+ "esprima": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
+ "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A=="
+ }
+ }
+ },
+ "esprima": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.2.2.tgz",
+ "integrity": "sha512-+JpPZam9w5DuJ3Q67SqsMGtiHKENSMRVoxvArfJZK01/BfLEObtZ6orJa/MtoGNR/rfMgp5837T41PAmTwAv/A=="
+ },
+ "estraverse": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz",
+ "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw=="
+ },
+ "esutils": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
+ "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g=="
+ },
+ "etag": {
+ "version": "1.8.1",
+ "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
+ "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg=="
+ },
+ "event-emitter": {
+ "version": "0.3.5",
+ "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz",
+ "integrity": "sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==",
+ "requires": {
+ "d": "1",
+ "es5-ext": "~0.10.14"
+ }
+ },
+ "eventemitter3": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-3.1.2.tgz",
+ "integrity": "sha512-tvtQIeLVHjDkJYnzf2dgVMxfuSGJeM/7UCG17TT4EumTfNtF+0nebF/4zWOIkCreAbtNqhGEboB6BWrwqNaw4Q=="
+ },
+ "express": {
+ "version": "4.18.2",
+ "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz",
+ "integrity": "sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==",
+ "requires": {
+ "accepts": "~1.3.8",
+ "array-flatten": "1.1.1",
+ "body-parser": "1.20.1",
+ "content-disposition": "0.5.4",
+ "content-type": "~1.0.4",
+ "cookie": "0.5.0",
+ "cookie-signature": "1.0.6",
+ "debug": "2.6.9",
+ "depd": "2.0.0",
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "etag": "~1.8.1",
+ "finalhandler": "1.2.0",
+ "fresh": "0.5.2",
+ "http-errors": "2.0.0",
+ "merge-descriptors": "1.0.1",
+ "methods": "~1.1.2",
+ "on-finished": "2.4.1",
+ "parseurl": "~1.3.3",
+ "path-to-regexp": "0.1.7",
+ "proxy-addr": "~2.0.7",
+ "qs": "6.11.0",
+ "range-parser": "~1.2.1",
+ "safe-buffer": "5.2.1",
+ "send": "0.18.0",
+ "serve-static": "1.15.0",
+ "setprototypeof": "1.2.0",
+ "statuses": "2.0.1",
+ "type-is": "~1.6.18",
+ "utils-merge": "1.0.1",
+ "vary": "~1.1.2"
+ }
+ },
+ "ext": {
+ "version": "1.7.0",
+ "resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz",
+ "integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==",
+ "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=="
+ }
+ }
+ },
+ "eyes": {
+ "version": "0.1.8",
+ "resolved": "https://registry.npmjs.org/eyes/-/eyes-0.1.8.tgz",
+ "integrity": "sha512-GipyPsXO1anza0AOZdy69Im7hGFCNB7Y/NGjDlZGJ3GJJLtwNSb2vrzYrTYJRrRloVx7pl+bhUaTB8yiccPvFQ=="
+ },
+ "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=="
+ },
+ "fast-levenshtein": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
+ "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw=="
+ },
+ "finalhandler": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz",
+ "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==",
+ "requires": {
+ "debug": "2.6.9",
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "on-finished": "2.4.1",
+ "parseurl": "~1.3.3",
+ "statuses": "2.0.1",
+ "unpipe": "~1.0.0"
+ }
+ },
+ "follow-redirects": {
+ "version": "1.15.2",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
+ "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA=="
+ },
+ "for-each": {
+ "version": "0.3.3",
+ "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz",
+ "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==",
+ "requires": {
+ "is-callable": "^1.1.3"
+ }
+ },
+ "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==",
+ "requires": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "mime-types": "^2.1.12"
+ }
+ },
+ "forwarded": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
+ "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow=="
+ },
+ "fresh": {
+ "version": "0.5.2",
+ "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
+ "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q=="
+ },
+ "from2": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz",
+ "integrity": "sha512-OMcX/4IC/uqEPVgGeyfN22LJk6AZrMkRZHxcHBMBvHScDGgwTm2GT2Wkgtocyd3JfZffjj2kYUDXXII0Fk9W0g==",
+ "requires": {
+ "inherits": "^2.0.1",
+ "readable-stream": "^2.0.0"
+ }
+ },
+ "fs-capacitor": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/fs-capacitor/-/fs-capacitor-2.0.4.tgz",
+ "integrity": "sha512-8S4f4WsCryNw2mJJchi46YgB6CR5Ze+4L1h8ewl9tEpL4SJ3ZO+c/bS4BWhB8bK+O3TMqhuZarTitd0S0eh2pA=="
+ },
+ "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=="
+ },
+ "function-bind": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
+ "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A=="
+ },
+ "function.prototype.name": {
+ "version": "1.1.5",
+ "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz",
+ "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.3",
+ "es-abstract": "^1.19.0",
+ "functions-have-names": "^1.2.2"
+ }
+ },
+ "functions-have-names": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz",
+ "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ=="
+ },
+ "get-intrinsic": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.0.tgz",
+ "integrity": "sha512-L049y6nFOuom5wGyRc3/gdTLO94dySVKRACj1RmJZBQXlbTMhtNIgkWkUHq+jYmZvKf14EW1EoJnnjbmoHij0Q==",
+ "requires": {
+ "function-bind": "^1.1.1",
+ "has": "^1.0.3",
+ "has-symbols": "^1.0.3"
+ }
+ },
+ "get-symbol-description": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz",
+ "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "get-intrinsic": "^1.1.1"
+ }
+ },
+ "globalthis": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.3.tgz",
+ "integrity": "sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==",
+ "requires": {
+ "define-properties": "^1.1.3"
+ }
+ },
+ "gopd": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz",
+ "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==",
+ "requires": {
+ "get-intrinsic": "^1.1.3"
+ }
+ },
+ "graphql": {
+ "version": "14.7.0",
+ "resolved": "https://registry.npmjs.org/graphql/-/graphql-14.7.0.tgz",
+ "integrity": "sha512-l0xWZpoPKpppFzMfvVyFmp9vLN7w/ZZJPefUicMCepfJeQ8sMcztloGYY9DfjVPo6tIUDzU5Hw3MUbIjj9AVVA==",
+ "requires": {
+ "iterall": "^1.2.2"
+ }
+ },
+ "graphql-extensions": {
+ "version": "0.16.0",
+ "resolved": "https://registry.npmjs.org/graphql-extensions/-/graphql-extensions-0.16.0.tgz",
+ "integrity": "sha512-rZQc/USoEIw437BGRUwoHoLPR1LA791Ltj6axONqgKIyyx2sqIO3YT9kTbB/eIUdJBrCozp4KuUeZ09xKeQDxg==",
+ "requires": {
+ "@apollographql/apollo-tools": "^0.5.0",
+ "apollo-server-env": "^3.2.0",
+ "apollo-server-types": "^0.10.0"
+ }
+ },
+ "graphql-fields": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/graphql-fields/-/graphql-fields-2.0.3.tgz",
+ "integrity": "sha512-x3VE5lUcR4XCOxPIqaO4CE+bTK8u6gVouOdpQX9+EKHr+scqtK5Pp/l8nIGqIpN1TUlkKE6jDCCycm/WtLRAwA=="
+ },
+ "graphql-middleware": {
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/graphql-middleware/-/graphql-middleware-1.3.1.tgz",
+ "integrity": "sha512-KwcS+PY74NJ87XD9vvPrfXdgbkGDw8ii10dg4mUCKvPzNEgcp/revP7l4bmfzrFaa3ZioJcJXioo3luH8/7rSQ==",
+ "requires": {
+ "graphql-tools": "^3.0.2"
+ },
+ "dependencies": {
+ "graphql-tools": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/graphql-tools/-/graphql-tools-3.1.1.tgz",
+ "integrity": "sha512-yHvPkweUB0+Q/GWH5wIG60bpt8CTwBklCSzQdEHmRUgAdEQKxw+9B7zB3dG7wB3Ym7M7lfrS4Ej+jtDZfA2UXg==",
+ "requires": {
+ "apollo-link": "^1.2.2",
+ "apollo-utilities": "^1.0.1",
+ "deprecated-decorator": "^0.1.6",
+ "iterall": "^1.1.3",
+ "uuid": "^3.1.0"
+ }
+ }
+ }
+ },
+ "graphql-playground-html": {
+ "version": "1.6.30",
+ "resolved": "https://registry.npmjs.org/graphql-playground-html/-/graphql-playground-html-1.6.30.tgz",
+ "integrity": "sha512-tpCujhsJMva4aqE8ULnF7/l3xw4sNRZcSHu+R00VV+W0mfp+Q20Plvcrp+5UXD+2yS6oyCXncA+zoQJQqhGCEw==",
+ "requires": {
+ "xss": "^1.0.6"
+ }
+ },
+ "graphql-playground-middleware-express": {
+ "version": "1.7.23",
+ "resolved": "https://registry.npmjs.org/graphql-playground-middleware-express/-/graphql-playground-middleware-express-1.7.23.tgz",
+ "integrity": "sha512-M/zbTyC1rkgiQjFSgmzAv6umMHOphYLNWZp6Ye5QrD77WfGOOoSqDsVmGUczc2pDkEPEzzGB/bvBO5rdzaTRgw==",
+ "requires": {
+ "graphql-playground-html": "^1.6.30"
+ }
+ },
+ "graphql-scalars": {
+ "version": "0.1.5",
+ "resolved": "https://registry.npmjs.org/graphql-scalars/-/graphql-scalars-0.1.5.tgz",
+ "integrity": "sha512-+1nNlZO5Gvrqih7WxyDjaRoFEY0bDyQmAsIDvjjuU1vEP+Q+FNeg9AbDtSrDkI5b+RK4duapqKAXIgK7sbhiug=="
+ },
+ "graphql-subscriptions": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/graphql-subscriptions/-/graphql-subscriptions-1.2.1.tgz",
+ "integrity": "sha512-95yD/tKi24q8xYa7Q9rhQN16AYj5wPbrb8tmHGM3WRc9EBmWrG/0kkMl+tQG8wcEuE9ibR4zyOM31p5Sdr2v4g==",
+ "requires": {
+ "iterall": "^1.3.0"
+ }
+ },
+ "graphql-tag": {
+ "version": "2.12.6",
+ "resolved": "https://registry.npmjs.org/graphql-tag/-/graphql-tag-2.12.6.tgz",
+ "integrity": "sha512-FdSNcu2QQcWnM2VNvSCCDCVS5PpPqpzgFT8+GXzqJuoDd0CBncxCY278u4mhRO7tMgo2JjgJA5aZ+nWSQ/Z+xg==",
+ "requires": {
+ "tslib": "^2.1.0"
+ },
+ "dependencies": {
+ "tslib": {
+ "version": "2.5.0",
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.0.tgz",
+ "integrity": "sha512-336iVw3rtn2BUK7ORdIAHTyxHGRIHVReokCR3XjbckJMK7ms8FysBfhLR8IXnAgy7T0PTPNBWKiH514FOW/WSg=="
+ }
+ }
+ },
+ "graphql-tools": {
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/graphql-tools/-/graphql-tools-4.0.8.tgz",
+ "integrity": "sha512-MW+ioleBrwhRjalKjYaLQbr+920pHBgy9vM/n47sswtns8+96sRn5M/G+J1eu7IMeKWiN/9p6tmwCHU7552VJg==",
+ "requires": {
+ "apollo-link": "^1.2.14",
+ "apollo-utilities": "^1.0.1",
+ "deprecated-decorator": "^0.1.6",
+ "iterall": "^1.1.3",
+ "uuid": "^3.1.0"
+ }
+ },
+ "graphql-type-json": {
+ "version": "0.2.4",
+ "resolved": "https://registry.npmjs.org/graphql-type-json/-/graphql-type-json-0.2.4.tgz",
+ "integrity": "sha512-/tq02ayMQjrG4oDFDRLLrPk0KvJXue0nVXoItBe7uAdbNXjQUu+HYCBdAmPLQoseVzUKKMzrhq2P/sfI76ON6w=="
+ },
+ "has": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
+ "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
+ "requires": {
+ "function-bind": "^1.1.1"
+ }
+ },
+ "has-bigints": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz",
+ "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ=="
+ },
+ "has-flag": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+ "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw=="
+ },
+ "has-property-descriptors": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz",
+ "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==",
+ "requires": {
+ "get-intrinsic": "^1.1.1"
+ }
+ },
+ "has-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz",
+ "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg=="
+ },
+ "has-symbols": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz",
+ "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A=="
+ },
+ "has-tostringtag": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz",
+ "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==",
+ "requires": {
+ "has-symbols": "^1.0.2"
+ }
+ },
+ "hpagent": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/hpagent/-/hpagent-0.1.2.tgz",
+ "integrity": "sha512-ePqFXHtSQWAFXYmj+JtOTHr84iNrII4/QRlAAPPE+zqnKy4xJo7Ie1Y4kC7AdB+LxLxSTTzBMASsEcy0q8YyvQ=="
+ },
+ "http-errors": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
+ "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
+ "requires": {
+ "depd": "2.0.0",
+ "inherits": "2.0.4",
+ "setprototypeof": "1.2.0",
+ "statuses": "2.0.1",
+ "toidentifier": "1.0.1"
+ }
+ },
+ "iconv-lite": {
+ "version": "0.4.24",
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
+ "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
+ "requires": {
+ "safer-buffer": ">= 2.1.2 < 3"
+ }
+ },
+ "inherits": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
+ },
+ "internal-slot": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.4.tgz",
+ "integrity": "sha512-tA8URYccNzMo94s5MQZgH8NB/XTa6HsOo0MLfXTKKEnHVVdegzaQoFZ7Jp44bdvLvY2waT5dc+j5ICEswhi7UQ==",
+ "requires": {
+ "get-intrinsic": "^1.1.3",
+ "has": "^1.0.3",
+ "side-channel": "^1.0.4"
+ }
+ },
+ "into-stream": {
+ "version": "5.1.1",
+ "resolved": "https://registry.npmjs.org/into-stream/-/into-stream-5.1.1.tgz",
+ "integrity": "sha512-krrAJ7McQxGGmvaYbB7Q1mcA+cRwg9Ij2RfWIeVesNBgVDZmzY/Fa4IpZUT3bmdRzMzdf/mzltCG2Dq99IZGBA==",
+ "requires": {
+ "from2": "^2.3.0",
+ "p-is-promise": "^3.0.0"
+ }
+ },
+ "ipaddr.js": {
+ "version": "1.9.1",
+ "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
+ "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
+ },
+ "is-array-buffer": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.1.tgz",
+ "integrity": "sha512-ASfLknmY8Xa2XtB4wmbz13Wu202baeA18cJBCeCy0wXUHZF0IPyVEXqKEcd+t2fNSLLL1vC6k7lxZEojNbISXQ==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "get-intrinsic": "^1.1.3",
+ "is-typed-array": "^1.1.10"
+ }
+ },
+ "is-bigint": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz",
+ "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==",
+ "requires": {
+ "has-bigints": "^1.0.1"
+ }
+ },
+ "is-boolean-object": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz",
+ "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "is-callable": {
+ "version": "1.2.7",
+ "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz",
+ "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA=="
+ },
+ "is-date-object": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz",
+ "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==",
+ "requires": {
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "is-negative-zero": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz",
+ "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA=="
+ },
+ "is-number-object": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz",
+ "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==",
+ "requires": {
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "is-promise": {
+ "version": "2.2.2",
+ "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.2.2.tgz",
+ "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ=="
+ },
+ "is-regex": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz",
+ "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "is-shared-array-buffer": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz",
+ "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==",
+ "requires": {
+ "call-bind": "^1.0.2"
+ }
+ },
+ "is-string": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz",
+ "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==",
+ "requires": {
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "is-symbol": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz",
+ "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==",
+ "requires": {
+ "has-symbols": "^1.0.2"
+ }
+ },
+ "is-typed-array": {
+ "version": "1.1.10",
+ "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.10.tgz",
+ "integrity": "sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==",
+ "requires": {
+ "available-typed-arrays": "^1.0.5",
+ "call-bind": "^1.0.2",
+ "for-each": "^0.3.3",
+ "gopd": "^1.0.1",
+ "has-tostringtag": "^1.0.0"
+ }
+ },
+ "is-weakref": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz",
+ "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==",
+ "requires": {
+ "call-bind": "^1.0.2"
+ }
+ },
+ "isarray": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
+ "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ=="
+ },
+ "isstream": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
+ "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g=="
+ },
+ "iterall": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/iterall/-/iterall-1.3.0.tgz",
+ "integrity": "sha512-QZ9qOMdF+QLHxy1QIpUHUU1D5pS2CG2P69LF6L6CPjPYA/XMOmKV3PZpawHoAjHNyB0swdVTRxdYT4tbBbxqwg=="
+ },
+ "jsonpath": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/jsonpath/-/jsonpath-1.1.1.tgz",
+ "integrity": "sha512-l6Cg7jRpixfbgoWgkrl77dgEj8RPvND0wMH6TwQmi9Qs4TFfS9u5cUFnbeKTwj5ga5Y3BTGGNI28k117LJ009w==",
+ "requires": {
+ "esprima": "1.2.2",
+ "static-eval": "2.0.2",
+ "underscore": "1.12.1"
+ }
+ },
+ "jwt-decode": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-3.1.2.tgz",
+ "integrity": "sha512-UfpWE/VZn0iP50d8cz9NrZLM9lSWhcJ+0Gt/nm4by88UL+J1SiKN8/5dkjMmbEzwL2CAe+67GsegCbIKtbp75A=="
+ },
+ "kind-of": {
+ "version": "6.0.3",
+ "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",
+ "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw=="
+ },
+ "levn": {
+ "version": "0.3.0",
+ "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz",
+ "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==",
+ "requires": {
+ "prelude-ls": "~1.1.2",
+ "type-check": "~0.3.2"
+ }
+ },
+ "lodash": {
+ "version": "4.17.21",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
+ },
+ "lodash._basebind": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._basebind/-/lodash._basebind-2.3.0.tgz",
+ "integrity": "sha512-SHqM7YCuJ+BeGTs7lqpWnmdHEeF4MWxS3dksJctHFNxR81FXPOzA4bS5Vs5CpcGTkBpM8FCl+YEbQEblRw8ABg==",
+ "requires": {
+ "lodash._basecreate": "~2.3.0",
+ "lodash._setbinddata": "~2.3.0",
+ "lodash.isobject": "~2.3.0"
+ }
+ },
+ "lodash._basecreate": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._basecreate/-/lodash._basecreate-2.3.0.tgz",
+ "integrity": "sha512-vwZaWldZwS2y9b99D8i9+WtgiZXbHKsBsMrpxJEqTsNW20NhJo5W8PBQkeQO9CmxuqEYn8UkMnfEM2MMT4cVrw==",
+ "requires": {
+ "lodash._renative": "~2.3.0",
+ "lodash.isobject": "~2.3.0",
+ "lodash.noop": "~2.3.0"
+ }
+ },
+ "lodash._basecreatecallback": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._basecreatecallback/-/lodash._basecreatecallback-2.3.0.tgz",
+ "integrity": "sha512-Ev+pDzzfVfgbiucpXijconLGRBar7/+KNCf05kSnk4CmdDVhAy1RdbU9efCJ/o9GXI08JdUGwZ+5QJ3QX3kj0g==",
+ "requires": {
+ "lodash._setbinddata": "~2.3.0",
+ "lodash.bind": "~2.3.0",
+ "lodash.identity": "~2.3.0",
+ "lodash.support": "~2.3.0"
+ }
+ },
+ "lodash._basecreatewrapper": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._basecreatewrapper/-/lodash._basecreatewrapper-2.3.0.tgz",
+ "integrity": "sha512-YLycQ7k8AB9Wc1EOvLNxuRWcqipDkMXq2GCgnLWQR6qtgTb3gY3LELzEpnFshrEO4LOLs+R2EpcY+uCOZaLQ8Q==",
+ "requires": {
+ "lodash._basecreate": "~2.3.0",
+ "lodash._setbinddata": "~2.3.0",
+ "lodash._slice": "~2.3.0",
+ "lodash.isobject": "~2.3.0"
+ }
+ },
+ "lodash._createwrapper": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._createwrapper/-/lodash._createwrapper-2.3.0.tgz",
+ "integrity": "sha512-XjaI/rzg9W+WO4WJDQ+PRlHD5sAMJ1RhJLuT65cBxLCb1kIYs4U20jqvTDGAWyVT3c34GYiLd9AreHYuB/8yJA==",
+ "requires": {
+ "lodash._basebind": "~2.3.0",
+ "lodash._basecreatewrapper": "~2.3.0",
+ "lodash.isfunction": "~2.3.0"
+ }
+ },
+ "lodash._objecttypes": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._objecttypes/-/lodash._objecttypes-2.3.0.tgz",
+ "integrity": "sha512-jbA6QyHt9cw3BzvbWzIcnU3Z12jSneT6xBgz3Y782CJsN1tV5aTBKrFo2B4AkeHBNaxSrbPYZZpi1Lwj3xjdtg=="
+ },
+ "lodash._renative": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._renative/-/lodash._renative-2.3.0.tgz",
+ "integrity": "sha512-v44MRirqYqZGK/h5UKoVqXWF2L+LUiLTU+Ogu5rHRVWJUA1uWIlHaMpG8f/OA8j++BzPMQij9+erXHtgFcbuwg=="
+ },
+ "lodash._setbinddata": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._setbinddata/-/lodash._setbinddata-2.3.0.tgz",
+ "integrity": "sha512-xMFfbF7dL+sFtrdE49uHFmfpBAEwlFtfgMp86nQRlAF6aizYL+3MTbnYMKJSkP1W501PhsgiBED5kBbZd8kR2g==",
+ "requires": {
+ "lodash._renative": "~2.3.0",
+ "lodash.noop": "~2.3.0"
+ }
+ },
+ "lodash._shimkeys": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._shimkeys/-/lodash._shimkeys-2.3.0.tgz",
+ "integrity": "sha512-9Iuyi7TiWMGa/9+2rqEE+Zwye4b/U2w7Saw6UX1h6Xs88mEER+uz9FZcEBPKMVKsad9Pw5GNAcIBRnW2jNpneQ==",
+ "requires": {
+ "lodash._objecttypes": "~2.3.0"
+ }
+ },
+ "lodash._slice": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash._slice/-/lodash._slice-2.3.0.tgz",
+ "integrity": "sha512-7C61GhzRUv36gTafr+RIb+AomCAYsSATEoK4OP0VkNBcwvsM022Z22AVgqjjzikeNO1U29LzsJZDvLbiNPUYvA=="
+ },
+ "lodash.bind": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.bind/-/lodash.bind-2.3.0.tgz",
+ "integrity": "sha512-goakyOo+FMN8lttMPnZ0UNlr5RlzX4IrUXyTJPT2A0tGCMXySupond9wzvDqTvVmYTcQjIKGrj8naJDS2xWAlQ==",
+ "requires": {
+ "lodash._createwrapper": "~2.3.0",
+ "lodash._renative": "~2.3.0",
+ "lodash._slice": "~2.3.0"
+ }
+ },
+ "lodash.foreach": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.foreach/-/lodash.foreach-2.3.0.tgz",
+ "integrity": "sha512-yLnyptVRJd0//AbGp480grgQG9iaDIV5uOgSbpurRy1dYybPbjNTLQ3FyLEQ84buVLPG7jyaiyvpzgfOutRB3Q==",
+ "requires": {
+ "lodash._basecreatecallback": "~2.3.0",
+ "lodash.forown": "~2.3.0"
+ }
+ },
+ "lodash.forown": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.forown/-/lodash.forown-2.3.0.tgz",
+ "integrity": "sha512-dUnCsuQTtq3Y7bxPNoEEqjJjPL2ftLtcz2PTeRKvhbpdM514AvnqCjewHGsm/W+dwspIwa14KoWEZeizJ7smxA==",
+ "requires": {
+ "lodash._basecreatecallback": "~2.3.0",
+ "lodash._objecttypes": "~2.3.0",
+ "lodash.keys": "~2.3.0"
+ }
+ },
+ "lodash.identity": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.identity/-/lodash.identity-2.3.0.tgz",
+ "integrity": "sha512-NYJ2r2cwy3tkx/saqbIZEX6oQUzjWTnGRu7d/zmBjMCZos3eHBxCpbvWFWSetv8jFVrptsp6EbWjzNgBKhUoOA=="
+ },
+ "lodash.isfunction": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.isfunction/-/lodash.isfunction-2.3.0.tgz",
+ "integrity": "sha512-X5lteBYlCrVO7Qc00fxP8W90fzRp6Ax9XcHANmU3OsZHdSyIVZ9ZlX5QTTpRq8aGY+9I5Rmd0UTzTIIyWPugEQ=="
+ },
+ "lodash.isobject": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.isobject/-/lodash.isobject-2.3.0.tgz",
+ "integrity": "sha512-jo1pfV61C4TE8BfEzqaHj6EIKiSkFANJrB6yscwuCJMSRw5tbqjk4Gv7nJzk4Z6nFKobZjGZ8Qd41vmnwgeQqQ==",
+ "requires": {
+ "lodash._objecttypes": "~2.3.0"
+ }
+ },
+ "lodash.keys": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.keys/-/lodash.keys-2.3.0.tgz",
+ "integrity": "sha512-c0UW0ffqMxSCtoVbmVt2lERJLkEqgoOn2ejPsWXzr0ZrqRbl3uruGgwHzhtqXxi6K/ei3Ey7zimOqSwXgzazPg==",
+ "requires": {
+ "lodash._renative": "~2.3.0",
+ "lodash._shimkeys": "~2.3.0",
+ "lodash.isobject": "~2.3.0"
+ }
+ },
+ "lodash.noop": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.noop/-/lodash.noop-2.3.0.tgz",
+ "integrity": "sha512-NpSm8HRm1WkBBWHUveDukLF4Kfb5P5E3fjHc9Qre9A11nNubozLWD2wH3UBTZbu+KSuX8aSUvy9b+PUyEceJ8g=="
+ },
+ "lodash.sortby": {
+ "version": "4.7.0",
+ "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz",
+ "integrity": "sha512-HDWXG8isMntAyRF5vZ7xKuEvOhT4AhlRt/3czTSjvGUxjYCBVRQY48ViDHyfYz9VIoBkW4TMGQNapx+l3RUwdA=="
+ },
+ "lodash.support": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.support/-/lodash.support-2.3.0.tgz",
+ "integrity": "sha512-etc7VWbB0U3Iya8ixj2xy4sDBN3jvPX7ODi8iXtn4KkkjNpdngrdc7Vlt5jub/Vgqx6/dWtp7Ml9awhCQPYKGQ==",
+ "requires": {
+ "lodash._renative": "~2.3.0"
+ }
+ },
+ "loglevel": {
+ "version": "1.8.1",
+ "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.8.1.tgz",
+ "integrity": "sha512-tCRIJM51SHjAayKwC+QAg8hT8vg6z7GSgLJKGvzuPb1Wc+hLzqtuVLxp6/HzSPOozuK+8ErAhy7U/sVzw8Dgfg=="
+ },
+ "long": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz",
+ "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA=="
+ },
+ "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==",
+ "requires": {
+ "yallist": "^4.0.0"
+ }
+ },
+ "lru-queue": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/lru-queue/-/lru-queue-0.1.0.tgz",
+ "integrity": "sha512-BpdYkt9EvGl8OfWHDQPISVpcl5xZthb+XPsbELj5AQXxIC8IriDZIQYjBJPEm5rS420sjZ0TLEzRcq5KdBhYrQ==",
+ "requires": {
+ "es5-ext": "~0.10.2"
+ }
+ },
+ "media-typer": {
+ "version": "0.3.0",
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
+ "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ=="
+ },
+ "memoizee": {
+ "version": "0.4.15",
+ "resolved": "https://registry.npmjs.org/memoizee/-/memoizee-0.4.15.tgz",
+ "integrity": "sha512-UBWmJpLZd5STPm7PMUlOw/TSy972M+z8gcyQ5veOnSDRREz/0bmpyTfKt3/51DhEBqCZQn1udM/5flcSPYhkdQ==",
+ "requires": {
+ "d": "^1.0.1",
+ "es5-ext": "^0.10.53",
+ "es6-weak-map": "^2.0.3",
+ "event-emitter": "^0.3.5",
+ "is-promise": "^2.2.2",
+ "lru-queue": "^0.1.0",
+ "next-tick": "^1.1.0",
+ "timers-ext": "^0.1.7"
+ }
+ },
+ "merge-descriptors": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
+ "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w=="
+ },
+ "methods": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
+ "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w=="
+ },
+ "mime": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
+ "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
+ },
+ "mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg=="
+ },
+ "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==",
+ "requires": {
+ "mime-db": "1.52.0"
+ }
+ },
+ "mimic-response": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-2.1.0.tgz",
+ "integrity": "sha512-wXqjST+SLt7R009ySCglWBCFpjUygmCIfD790/kVbiGmUgfYGuB14PiTd5DwVxSV4NcYHjzMkoj5LjQZwTQLEA=="
+ },
+ "minimist": {
+ "version": "1.2.7",
+ "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.7.tgz",
+ "integrity": "sha512-bzfL1YUZsP41gmu/qjrEk0Q6i2ix/cVeAhbCbqH9u3zYutS1cLg00qhrD0M2MVdCcx4Sc0UpP2eBWo9rotpq6g=="
+ },
+ "morgan": {
+ "version": "1.10.0",
+ "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.10.0.tgz",
+ "integrity": "sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==",
+ "requires": {
+ "basic-auth": "~2.0.1",
+ "debug": "2.6.9",
+ "depd": "~2.0.0",
+ "on-finished": "~2.3.0",
+ "on-headers": "~1.0.2"
+ },
+ "dependencies": {
+ "on-finished": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
+ "integrity": "sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww==",
+ "requires": {
+ "ee-first": "1.1.1"
+ }
+ }
+ }
+ },
+ "ms": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+ "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="
+ },
+ "negotiator": {
+ "version": "0.6.3",
+ "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
+ "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg=="
+ },
+ "next-tick": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz",
+ "integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ=="
+ },
+ "node-fetch": {
+ "version": "2.6.9",
+ "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.9.tgz",
+ "integrity": "sha512-DJm/CJkZkRjKKj4Zi4BsKVZh3ValV5IR5s7LVZnW+6YMh0W1BfNA8XSs6DLMGYlId5F3KnA70uu2qepcR08Qqg==",
+ "requires": {
+ "whatwg-url": "^5.0.0"
+ }
+ },
+ "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=="
+ },
+ "object-inspect": {
+ "version": "1.12.3",
+ "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz",
+ "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g=="
+ },
+ "object-keys": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz",
+ "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA=="
+ },
+ "object-path": {
+ "version": "0.11.8",
+ "resolved": "https://registry.npmjs.org/object-path/-/object-path-0.11.8.tgz",
+ "integrity": "sha512-YJjNZrlXJFM42wTBn6zgOJVar9KFJvzx6sTWDte8sWZF//cnjl0BxHNpfZx+ZffXX63A9q0b1zsFiBX4g4X5KA=="
+ },
+ "object.assign": {
+ "version": "4.1.4",
+ "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz",
+ "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.4",
+ "has-symbols": "^1.0.3",
+ "object-keys": "^1.1.1"
+ }
+ },
+ "object.getownpropertydescriptors": {
+ "version": "2.1.5",
+ "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.5.tgz",
+ "integrity": "sha512-yDNzckpM6ntyQiGTik1fKV1DcVDRS+w8bvpWNCBanvH5LfRX9O8WTHqQzG4RZwRAM4I0oU7TV11Lj5v0g20ibw==",
+ "requires": {
+ "array.prototype.reduce": "^1.0.5",
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.4",
+ "es-abstract": "^1.20.4"
+ }
+ },
+ "on-finished": {
+ "version": "2.4.1",
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
+ "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
+ "requires": {
+ "ee-first": "1.1.1"
+ }
+ },
+ "on-headers": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz",
+ "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA=="
+ },
+ "once": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+ "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
+ "requires": {
+ "wrappy": "1"
+ }
+ },
+ "optionator": {
+ "version": "0.8.3",
+ "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz",
+ "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==",
+ "requires": {
+ "deep-is": "~0.1.3",
+ "fast-levenshtein": "~2.0.6",
+ "levn": "~0.3.0",
+ "prelude-ls": "~1.1.2",
+ "type-check": "~0.3.2",
+ "word-wrap": "~1.2.3"
+ }
+ },
+ "p-is-promise": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-3.0.0.tgz",
+ "integrity": "sha512-Wo8VsW4IRQSKVXsJCn7TomUaVtyfjVDn3nUP7kE967BQk0CwFpdbZs0X0uk5sW9mkBa9eNM7hCMaG93WUAwxYQ=="
+ },
+ "paralleljs": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/paralleljs/-/paralleljs-0.2.1.tgz",
+ "integrity": "sha512-msRXP0u+C6tIzBYWuYR+uDWHh5U6W/qM8vE9qIqMOU1xkyxtdMQtWA4tjTcw3uqMZAVN36zdaOwYAoKXkg2Uzg=="
+ },
+ "parseurl": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
+ "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
+ },
+ "path-to-regexp": {
+ "version": "0.1.7",
+ "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
+ "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ=="
+ },
+ "prelude-ls": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz",
+ "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w=="
+ },
+ "process-nextick-args": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
+ "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
+ },
+ "proxy-addr": {
+ "version": "2.0.7",
+ "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
+ "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
+ "requires": {
+ "forwarded": "0.2.0",
+ "ipaddr.js": "1.9.1"
+ }
+ },
+ "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=="
+ },
+ "pump": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
+ "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
+ "requires": {
+ "end-of-stream": "^1.1.0",
+ "once": "^1.3.1"
+ }
+ },
+ "qew": {
+ "version": "0.9.13",
+ "resolved": "https://registry.npmjs.org/qew/-/qew-0.9.13.tgz",
+ "integrity": "sha512-DGWHoEIzgzU7JHPlRh2FE72vnmGeT0RSjMDMGdZth5FZAgYhib4bO3zvkQsxld7/SPC6uXHeSCuPrCWGuGa6qQ==",
+ "requires": {
+ "typescript": "^2.4.2"
+ }
+ },
+ "qs": {
+ "version": "6.11.0",
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz",
+ "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==",
+ "requires": {
+ "side-channel": "^1.0.4"
+ }
+ },
+ "ramda": {
+ "version": "0.26.1",
+ "resolved": "https://registry.npmjs.org/ramda/-/ramda-0.26.1.tgz",
+ "integrity": "sha512-hLWjpy7EnsDBb0p+Z3B7rPi3GDeRG5ZtiI33kJhTt+ORCd38AbAIjB/9zRIUoeTbE/AVX5ZkU7m6bznsvrf8eQ=="
+ },
+ "range-parser": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
+ "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
+ },
+ "raw-body": {
+ "version": "2.5.1",
+ "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz",
+ "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==",
+ "requires": {
+ "bytes": "3.1.2",
+ "http-errors": "2.0.0",
+ "iconv-lite": "0.4.24",
+ "unpipe": "1.0.0"
+ }
+ },
+ "readable-stream": {
+ "version": "2.3.7",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz",
+ "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==",
+ "requires": {
+ "core-util-is": "~1.0.0",
+ "inherits": "~2.0.3",
+ "isarray": "~1.0.0",
+ "process-nextick-args": "~2.0.0",
+ "safe-buffer": "~5.1.1",
+ "string_decoder": "~1.1.1",
+ "util-deprecate": "~1.0.1"
+ },
+ "dependencies": {
+ "safe-buffer": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+ "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
+ }
+ }
+ },
+ "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=="
+ },
+ "regexp.prototype.flags": {
+ "version": "1.4.3",
+ "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz",
+ "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.3",
+ "functions-have-names": "^1.2.2"
+ }
+ },
+ "retry": {
+ "version": "0.13.1",
+ "resolved": "https://registry.npmjs.org/retry/-/retry-0.13.1.tgz",
+ "integrity": "sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg=="
+ },
+ "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=="
+ },
+ "safe-regex-test": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz",
+ "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "get-intrinsic": "^1.1.3",
+ "is-regex": "^1.1.4"
+ }
+ },
+ "safer-buffer": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
+ },
+ "secure-json-parse": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/secure-json-parse/-/secure-json-parse-2.7.0.tgz",
+ "integrity": "sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw=="
+ },
+ "send": {
+ "version": "0.18.0",
+ "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz",
+ "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==",
+ "requires": {
+ "debug": "2.6.9",
+ "depd": "2.0.0",
+ "destroy": "1.2.0",
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "etag": "~1.8.1",
+ "fresh": "0.5.2",
+ "http-errors": "2.0.0",
+ "mime": "1.6.0",
+ "ms": "2.1.3",
+ "on-finished": "2.4.1",
+ "range-parser": "~1.2.1",
+ "statuses": "2.0.1"
+ },
+ "dependencies": {
+ "ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
+ }
+ }
+ },
+ "serve-static": {
+ "version": "1.15.0",
+ "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz",
+ "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==",
+ "requires": {
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "parseurl": "~1.3.3",
+ "send": "0.18.0"
+ }
+ },
+ "setprototypeof": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
+ "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="
+ },
+ "sha.js": {
+ "version": "2.4.11",
+ "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz",
+ "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==",
+ "requires": {
+ "inherits": "^2.0.1",
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "side-channel": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz",
+ "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==",
+ "requires": {
+ "call-bind": "^1.0.0",
+ "get-intrinsic": "^1.0.2",
+ "object-inspect": "^1.9.0"
+ }
+ },
+ "source-map": {
+ "version": "0.6.1",
+ "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
+ "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
+ "optional": true
+ },
+ "stack-trace": {
+ "version": "0.0.10",
+ "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz",
+ "integrity": "sha512-KGzahc7puUKkzyMt+IqAep+TVNbKP+k2Lmwhub39m1AsTSkaDutx56aDCo+HLDzf/D26BIHTJWNiTG1KAJiQCg=="
+ },
+ "static-eval": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/static-eval/-/static-eval-2.0.2.tgz",
+ "integrity": "sha512-N/D219Hcr2bPjLxPiV+TQE++Tsmrady7TqAJugLy7Xk1EumfDWS/f5dtBbkRCGE7wKKXuYockQoj8Rm2/pVKyg==",
+ "requires": {
+ "escodegen": "^1.8.1"
+ }
+ },
+ "statuses": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
+ "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ=="
+ },
+ "stoppable": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/stoppable/-/stoppable-1.1.0.tgz",
+ "integrity": "sha512-KXDYZ9dszj6bzvnEMRYvxgeTHU74QBFL54XKtP3nyMuJ81CFYtABZ3bAzL2EdFUaEwJOBOgENyFj3R7oTzDyyw=="
+ },
+ "streamsearch": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz",
+ "integrity": "sha512-jos8u++JKm0ARcSUTAZXOVC0mSox7Bhn6sBgty73P1f3JGf7yG2clTbBNHUdde/kdvP2FESam+vM6l8jBrNxHA=="
+ },
+ "string.prototype.trimend": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz",
+ "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.4",
+ "es-abstract": "^1.20.4"
+ }
+ },
+ "string.prototype.trimstart": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz",
+ "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "define-properties": "^1.1.4",
+ "es-abstract": "^1.20.4"
+ }
+ },
+ "string_decoder": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
+ "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
+ "requires": {
+ "safe-buffer": "~5.1.0"
+ },
+ "dependencies": {
+ "safe-buffer": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+ "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
+ }
+ }
+ },
+ "subscriptions-transport-ws": {
+ "version": "0.9.19",
+ "resolved": "https://registry.npmjs.org/subscriptions-transport-ws/-/subscriptions-transport-ws-0.9.19.tgz",
+ "integrity": "sha512-dxdemxFFB0ppCLg10FTtRqH/31FNRL1y1BQv8209MK5I4CwALb7iihQg+7p65lFcIl8MHatINWBLOqpgU4Kyyw==",
+ "requires": {
+ "backo2": "^1.0.2",
+ "eventemitter3": "^3.1.0",
+ "iterall": "^1.2.1",
+ "symbol-observable": "^1.0.4",
+ "ws": "^5.2.0 || ^6.0.0 || ^7.0.0"
+ }
+ },
+ "supports-color": {
+ "version": "5.5.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+ "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+ "requires": {
+ "has-flag": "^3.0.0"
+ }
+ },
+ "symbol-observable": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz",
+ "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ=="
+ },
+ "tar-stream": {
+ "version": "1.6.2",
+ "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-1.6.2.tgz",
+ "integrity": "sha512-rzS0heiNf8Xn7/mpdSVVSMAWAoy9bfb1WOTYC78Z0UQKeKa/CWS8FOq0lKGNa8DWKAn9gxjCvMLYc5PGXYlK2A==",
+ "requires": {
+ "bl": "^1.0.0",
+ "buffer-alloc": "^1.2.0",
+ "end-of-stream": "^1.0.0",
+ "fs-constants": "^1.0.0",
+ "readable-stream": "^2.3.0",
+ "to-buffer": "^1.1.1",
+ "xtend": "^4.0.0"
+ }
+ },
+ "through2": {
+ "version": "2.0.5",
+ "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz",
+ "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==",
+ "requires": {
+ "readable-stream": "~2.3.6",
+ "xtend": "~4.0.1"
+ }
+ },
+ "timers-ext": {
+ "version": "0.1.7",
+ "resolved": "https://registry.npmjs.org/timers-ext/-/timers-ext-0.1.7.tgz",
+ "integrity": "sha512-b85NUNzTSdodShTIbky6ZF02e8STtVVfD+fu4aXXShEELpozH+bCpJLYMPZbsABN2wDH7fJpqIoXxJpzbf0NqQ==",
+ "requires": {
+ "es5-ext": "~0.10.46",
+ "next-tick": "1"
+ }
+ },
+ "to-buffer": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/to-buffer/-/to-buffer-1.1.1.tgz",
+ "integrity": "sha512-lx9B5iv7msuFYE3dytT+KE5tap+rNYw+K4jVkb9R/asAb+pbBSM17jtunHplhBe6RRJdZx3Pn2Jph24O32mOVg=="
+ },
+ "toidentifier": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
+ "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA=="
+ },
+ "tr46": {
+ "version": "0.0.3",
+ "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz",
+ "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw=="
+ },
+ "ts-invariant": {
+ "version": "0.4.4",
+ "resolved": "https://registry.npmjs.org/ts-invariant/-/ts-invariant-0.4.4.tgz",
+ "integrity": "sha512-uEtWkFM/sdZvRNNDL3Ehu4WVpwaulhwQszV8mrtcdeE8nN00BV9mAmQ88RkrBhFgl9gMgvjJLAQcZbnPXI9mlA==",
+ "requires": {
+ "tslib": "^1.9.3"
+ }
+ },
+ "tslib": {
+ "version": "1.14.1",
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz",
+ "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg=="
+ },
+ "type": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz",
+ "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg=="
+ },
+ "type-check": {
+ "version": "0.3.2",
+ "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz",
+ "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==",
+ "requires": {
+ "prelude-ls": "~1.1.2"
+ }
+ },
+ "type-is": {
+ "version": "1.6.18",
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
+ "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
+ "requires": {
+ "media-typer": "0.3.0",
+ "mime-types": "~2.1.24"
+ }
+ },
+ "typed-array-length": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz",
+ "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "for-each": "^0.3.3",
+ "is-typed-array": "^1.1.9"
+ }
+ },
+ "typescript": {
+ "version": "2.9.2",
+ "resolved": "https://registry.npmjs.org/typescript/-/typescript-2.9.2.tgz",
+ "integrity": "sha512-Gr4p6nFNaoufRIY4NMdpQRNmgxVIGMs4Fcu/ujdYk3nAZqk7supzBE9idmvfZIlH/Cuj//dvi+019qEue9lV0w=="
+ },
+ "unbox-primitive": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz",
+ "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==",
+ "requires": {
+ "call-bind": "^1.0.2",
+ "has-bigints": "^1.0.2",
+ "has-symbols": "^1.0.3",
+ "which-boxed-primitive": "^1.0.2"
+ }
+ },
+ "underscore": {
+ "version": "1.12.1",
+ "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.12.1.tgz",
+ "integrity": "sha512-hEQt0+ZLDVUMhebKxL4x1BTtDY7bavVofhZ9KZ4aI26X9SRaE+Y3m83XUL1UP2jn8ynjndwCCpEHdUG+9pP1Tw=="
+ },
+ "unpipe": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
+ "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ=="
+ },
+ "url-join": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/url-join/-/url-join-4.0.1.tgz",
+ "integrity": "sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA=="
+ },
+ "utf8": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/utf8/-/utf8-3.0.0.tgz",
+ "integrity": "sha512-E8VjFIQ/TyQgp+TZfS6l8yp/xWppSAHzidGiRrqe4bK4XP9pTRyKFgGJpO3SN7zdX4DeomTrwaseCHovfpFcqQ=="
+ },
+ "util-deprecate": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw=="
+ },
+ "util.promisify": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.1.1.tgz",
+ "integrity": "sha512-/s3UsZUrIfa6xDhr7zZhnE9SLQ5RIXyYfiVnMMyMDzOc8WhWN4Nbh36H842OyurKbCDAesZOJaVyvmSl6fhGQw==",
+ "requires": {
+ "call-bind": "^1.0.0",
+ "define-properties": "^1.1.3",
+ "for-each": "^0.3.3",
+ "has-symbols": "^1.0.1",
+ "object.getownpropertydescriptors": "^2.1.1"
+ }
+ },
+ "utils-merge": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
+ "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA=="
+ },
+ "uuid": {
+ "version": "3.4.0",
+ "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz",
+ "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A=="
+ },
+ "vary": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
+ "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg=="
+ },
+ "webidl-conversions": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",
+ "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ=="
+ },
+ "whatwg-url": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz",
+ "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==",
+ "requires": {
+ "tr46": "~0.0.3",
+ "webidl-conversions": "^3.0.0"
+ }
+ },
+ "which-boxed-primitive": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz",
+ "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==",
+ "requires": {
+ "is-bigint": "^1.0.1",
+ "is-boolean-object": "^1.1.0",
+ "is-number-object": "^1.0.4",
+ "is-string": "^1.0.5",
+ "is-symbol": "^1.0.3"
+ }
+ },
+ "which-typed-array": {
+ "version": "1.1.9",
+ "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.9.tgz",
+ "integrity": "sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==",
+ "requires": {
+ "available-typed-arrays": "^1.0.5",
+ "call-bind": "^1.0.2",
+ "for-each": "^0.3.3",
+ "gopd": "^1.0.1",
+ "has-tostringtag": "^1.0.0",
+ "is-typed-array": "^1.1.10"
+ }
+ },
+ "winston": {
+ "version": "2.4.7",
+ "resolved": "https://registry.npmjs.org/winston/-/winston-2.4.7.tgz",
+ "integrity": "sha512-vLB4BqzCKDnnZH9PHGoS2ycawueX4HLqENXQitvFHczhgW2vFpSOn31LZtVr1KU8YTw7DS4tM+cqyovxo8taVg==",
+ "requires": {
+ "async": "^2.6.4",
+ "colors": "1.0.x",
+ "cycle": "1.0.x",
+ "eyes": "0.1.x",
+ "isstream": "0.1.x",
+ "stack-trace": "0.0.x"
+ }
+ },
+ "word-wrap": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz",
+ "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ=="
+ },
+ "wrappy": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+ "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="
+ },
+ "ws": {
+ "version": "7.5.9",
+ "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.9.tgz",
+ "integrity": "sha512-F+P9Jil7UiSKSkppIiD94dN07AwvFixvLIj1Og1Rl9GGMuNipJnV9JzjD6XuqmAeiswGvUmNLjr5cFuXwNS77Q=="
+ },
+ "xss": {
+ "version": "1.0.14",
+ "resolved": "https://registry.npmjs.org/xss/-/xss-1.0.14.tgz",
+ "integrity": "sha512-og7TEJhXvn1a7kzZGQ7ETjdQVS2UfZyTlsEdDOqvQF7GoxNfY+0YLCzBy1kPdsDDx4QuNAonQPddpsn6Xl/7sw==",
+ "requires": {
+ "commander": "^2.20.3",
+ "cssfilter": "0.0.10"
+ }
+ },
+ "xtend": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
+ "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ=="
+ },
+ "yallist": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
+ "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A=="
+ },
+ "zen-observable": {
+ "version": "0.8.15",
+ "resolved": "https://registry.npmjs.org/zen-observable/-/zen-observable-0.8.15.tgz",
+ "integrity": "sha512-PQ2PC7R9rslx84ndNBZB/Dkv8V8fZEpk83RLgXtYd0fwUgEjseMn1Dgajh2x6S8QbZAFa9p2qVCEuYZNgve0dQ=="
+ },
+ "zen-observable-ts": {
+ "version": "0.8.21",
+ "resolved": "https://registry.npmjs.org/zen-observable-ts/-/zen-observable-ts-0.8.21.tgz",
+ "integrity": "sha512-Yj3yXweRc8LdRMrCC8nIc4kkjWecPAUVh0TI0OUrWXx6aX790vLcDlWca6I4vsyCGH3LpWxq0dJRcMOFoVqmeg==",
+ "requires": {
+ "tslib": "^1.9.3",
+ "zen-observable": "^0.8.0"
+ }
+ }
+ }
+}
diff --git a/modules/nf-server-filter/package.json b/modules/nf-server-filter/package.json
new file mode 100644
index 000000000..f1a4511c9
--- /dev/null
+++ b/modules/nf-server-filter/package.json
@@ -0,0 +1,31 @@
+{
+ "name": "server-filter",
+ "version": "1.0.0",
+ "description": "",
+ "main": "index.js",
+ "type": "module",
+ "scripts": {},
+ "keywords": [],
+ "author": "",
+ "license": "ISC",
+ "dependencies": {
+ "@arranger/schema": "^2.19.4",
+ "@arranger/server": "2.19.3",
+ "@elastic/elasticsearch": "^7.17.0",
+ "@types/express-serve-static-core": "4.17.24",
+ "apollo-server-express": "^2.14.2",
+ "axios": "^1.3.4",
+ "cors": "^2.8.5",
+ "date-fns": "^2.29.2",
+ "express": "^4.18.2",
+ "graphql": "^14.5.3",
+ "graphql-playground-middleware-express": "^1.7.12",
+ "jsonpath": "^1.0.0",
+ "jwt-decode": "^3.1.2",
+ "kind-of": "^6.0.3",
+ "lodash": "^4.17.20",
+ "memoizee": "^0.4.15",
+ "minimist": "^1.2.5",
+ "through2": "^2.0.3"
+ }
+}
diff --git a/modules/nf-server-filter/server.js b/modules/nf-server-filter/server.js
new file mode 100644
index 000000000..1d549adfd
--- /dev/null
+++ b/modules/nf-server-filter/server.js
@@ -0,0 +1,104 @@
+import express from 'express';
+import cors from 'cors';
+import { Client } from '@elastic/elasticsearch';
+import { ApolloServer } from 'apollo-server-express';
+import { downloader } from './export/export-file.js';
+import { processRequest } from './authFilter.js';
+import { ARRANGER_PROJECT_ID, ELASTICSEARCH, SERVER_PORT } from './config.js';
+
+// We need to pull the createProjectSchema method out of the arranger server dist because this gives us access to the `getServerSideFilter` method
+import { createProjectSchema } from '@arranger/server/dist/startProject.js';
+
+// Manually create ES Client, as we need to give arranger access to it directly
+const esClient = new Client({
+ node: ELASTICSEARCH,
+});
+
+/**
+ * Execute server filter on the incoming request
+ * @param req request from express server
+ * @param res response from express server
+ * @param next next middleware function from express server
+ */
+
+async function serverFilter(req, res, next) {
+ try {
+ req.sqon = await processRequest();
+ } catch (error) {
+ console.error(`Failed to execute server filter: ${error.message} `);
+ }
+ next();
+}
+
+/**
+ * Retrieve sqon from RBAC filter
+ * @param context The context provided by the ApolloServer based on the incoming request
+ * @returns a SQON filter to apply to the entire request and to all aggregations
+ */
+function SQONFilter(context) {
+ return context.req.sqon;
+}
+
+// Schema for server-side filtering
+const arrangerSchema = await createProjectSchema({
+ es: esClient,
+ id: ARRANGER_PROJECT_ID,
+ graphqlOptions: {},
+ enableAdmin: false,
+ getServerSideFilter: SQONFilter,
+});
+
+// Schema for download functionality
+const arrangerSchemaDownload = await createProjectSchema({
+ es: esClient,
+ id: ARRANGER_PROJECT_ID,
+ graphqlOptions: {},
+ enableAdmin: false,
+});
+
+// Setup Apollo Server
+const server = new ApolloServer({
+ schema: arrangerSchema['schema'],
+ context: ({ req }) => {
+ return {
+ es: esClient,
+ req,
+ projectId: ARRANGER_PROJECT_ID,
+ };
+ },
+});
+
+// Setup express server
+const app = express();
+app.use(express.json()); // support json encoded bodies
+
+// Use serverFilter for incoming requests
+app.use(serverFilter);
+
+// Enable all CORS requests
+app.use(cors());
+
+// Add Arranger middleware
+server.applyMiddleware({
+ app,
+ path: `/${ARRANGER_PROJECT_ID}/graphql`,
+});
+server.applyMiddleware({
+ app,
+ path: `/${ARRANGER_PROJECT_ID}/graphql/*`,
+});
+
+// Enable export file
+app.use(
+ `/${ARRANGER_PROJECT_ID}/download`,
+ downloader({
+ projectId: ARRANGER_PROJECT_ID,
+ es: esClient,
+ arranger_schema: arrangerSchemaDownload,
+ }),
+);
+
+// Start express server
+app.listen(SERVER_PORT, () => {
+ console.log(`Arranger server running on ${SERVER_PORT}`);
+});
diff --git a/modules/nf-server-filter/sqon/base_sqon.js b/modules/nf-server-filter/sqon/base_sqon.js
new file mode 100644
index 000000000..78465573f
--- /dev/null
+++ b/modules/nf-server-filter/sqon/base_sqon.js
@@ -0,0 +1,19 @@
+/**
+ * Class to manage base SQONs
+ */
+export class BaseSQON {
+ /**
+ * Create base SQON
+ */
+ static generate() {
+ return {
+ op: 'and',
+ content: [
+ {
+ op: 'or',
+ content: [],
+ },
+ ],
+ };
+ }
+}
diff --git a/modules/server-filter/.env.schema b/modules/server-filter/.env.schema
index d16e3e4aa..8e5a62e89 100644
--- a/modules/server-filter/.env.schema
+++ b/modules/server-filter/.env.schema
@@ -13,6 +13,13 @@ AUTH_SERVICE=
# AUTH SERVICE PAGE SIZE
AUTH_SERVICE_PAGE_SIZE=
+# METADATA SERVICE HOST:PORT
+METADATA_SERVICE=
+
+# METADATA SERVICE PAGE SIZE
+METADATA_SERVICE_PAGE_SIZE=
+
+
# DOWNLOAD FILE STREAM BUFFER SIZE (export)
DOWNLOAD_STREAM_BUFFER_SIZE=
diff --git a/modules/server-filter/README.md b/modules/server-filter/README.md
index bb2353975..21be1cb16 100644
--- a/modules/server-filter/README.md
+++ b/modules/server-filter/README.md
@@ -12,7 +12,7 @@ NodeJs version: `v14.21.0`; NPM version: `6.14.17`. Recommended to use node vers
1. ``cd server-filter``
2. ``npm install``
-### Start-up
+### Start up
1. Set ``.env`` to point to desired server port, arranger project ID, and elasticsearch instance.
2. Run `node server.js`
@@ -23,3 +23,4 @@ NodeJs version: `v14.21.0`; NPM version: `6.14.17`. Recommended to use node vers
3. Export file logic exists under `/export`.
+
diff --git a/modules/server-filter/authFilter.js b/modules/server-filter/authFilter.js
index 58ba715fb..c85dd1faf 100644
--- a/modules/server-filter/authFilter.js
+++ b/modules/server-filter/authFilter.js
@@ -6,9 +6,13 @@ import memoize from 'memoizee';
import {
AUTH_SERVICE,
AUTH_SERVICE_PAGE_SIZE,
- MEMOIZE_TIMEOUT
+ MEMOIZE_TIMEOUT, METADATA_SERVICE, METADATA_SERVICE_PAGE_SIZE
} from './config.js';
+import {
+ ProjectFolder
+} from './sqonModels.js'
+
/**
* Function will read sqon json files
* @param path The path to sqon file
@@ -26,7 +30,8 @@ async function readFile(path) {
});
} catch (error) {
- console.error(`Failed to read sqon model: ${error}`);
+ console.error(error)
+ throw Error('Failed to read sqon model')
}
}
@@ -38,8 +43,14 @@ async function readFile(path) {
*/
export function decodeToken(request) {
try {
- // get user roles
+ // decode token
const decoded = jwt_decode(request.headers.authorization);
+
+ if ((!decoded['preferred_username']) || (!decoded['realm_access']['roles'])) {
+ throw Error('Invalid decoded token format')
+ }
+
+ // get user roles
const user_roles = decoded['realm_access']['roles'];
// get username
@@ -51,7 +62,8 @@ export function decodeToken(request) {
}
} catch (error) {
- console.error(`Failed to decode token: ${error}`);
+ console.error(error)
+ throw Error('Failed to decode token')
}
}
@@ -80,11 +92,63 @@ function UserRoleCheck(project_code, user_roles) {
}
} catch (error) {
- console.error(`Failed to check realm roles against project: ${error}`)
+ console.error(error)
+ throw Error(`Failed to check realm roles against project ${project_code}`)
+
}
}
+/**
+ * Function will call metadata service to obtain project folder names that can be accessed by user
+ * @param project_code project code to query project folders from
+ * @param token user-specific bearer token
+ * @param page The page number used for pagination
+ * @param data The resulting data for a single api call
+ * @returns An array of non user-accessible project folders for a project
+ */
+
+const getProjectFolders = async (project_code = '', token = '',page = 0, data = []) => {
+ try {
+ const zones = {0:'greenroom', 1:'core'}
+ const request_data = {
+ params: {
+ type: 'project_folder',
+ container_code: project_code,
+ status: 'ACTIVE',
+ parent_path: 'shared',
+ page: page,
+ page_size: METADATA_SERVICE_PAGE_SIZE
+ },
+ paramsSerializer: {
+ indexes: null // by default: false
+ },
+ headers: {
+ "Authorization" : `${token}`
+
+ }
+ }
+
+ const url = `${METADATA_SERVICE}/v1/items/search`;
+ const response = await axios.get(url, request_data);
+ page++;
+ if (response.data.result.length !== 0){
+ for (const entry of response.data.result) {
+ data.push({"name":entry.name, "zone":zones[entry.zone]});
+ }
+ }
+ if (response.data.num_of_pages > page) {
+ return getProjectFolders(project_code, token, page, data);
+ }
+ return data;
+ } catch (error) {
+ console.error(error.message)
+ throw Error(`Failed to call metadata service`);
+ }
+};
+
+
+
/**
* Function will call auth service to obtain rbac permissions for a project
* @param project_code The project code for a project
@@ -104,7 +168,8 @@ const getPermissions = async (project_code, page = 0, data = []) => {
}
return data;
} catch (error) {
- console.error(`Failed to call auth service: ${error}`);
+ console.error(error.message)
+ throw Error(`Failed to call auth service`);
}
};
@@ -112,15 +177,17 @@ const getPermissions = async (project_code, page = 0, data = []) => {
* Function will obtain rbac permissions for a project and filter based on realm role
* @param project_code The project code for a project
* @param user_roles The realm roles obtained by decoded token
+ * @param token Bearer token of user
* @returns Role and respective RBAC permissions for a project
*/
-const getRBAC = async (project_code, user_roles) => {
+const getRBAC = async (project_code, user_roles, token) => {
try {
// define base RBAC
const permitted = {
file_in_own_namefolder: [],
- file_any: []
+ file_any: [],
+ project_folders_accessible: []
};
// check if user has role associated with project
@@ -141,13 +208,23 @@ const getRBAC = async (project_code, user_roles) => {
}
}
}
+
+ // obtain project folders a user has access to
+ const projectFolders = await getProjectFolders(project_code, token);
+ if (projectFolders.length !== 0) {
+ for (const folder of projectFolders) {
+ permitted['project_folders_accessible'].push(folder)
+ }
+ }
+
return {
"role": validated_role,
"permissions": permitted
};
} catch (error) {
- console.log(`Failed to get RBAC info for project ${project_code}: ${error}`)
+ console.error(error)
+ throw Error(`Cannot retrieve RBAC info for project ${project_code}`)
}
};
@@ -170,7 +247,7 @@ const buildSQON = async (role_metadata, project_code, username) => {
const base_sqon = await readFile('./models/base_sqon.json');
// if user does not have any permissions, return no data
- if (permissions['file_any'].length === 0 && permissions['file_in_own_namefolder'].length === 0) {
+ if (permissions['file_any'].length === 0 && permissions['file_in_own_namefolder'].length === 0 && permissions['project_folders_accessible'].length === 0) {
base_sqon['content'][0]['content']['field'] = 'no_permissions'
return base_sqon
}
@@ -185,25 +262,36 @@ const buildSQON = async (role_metadata, project_code, username) => {
if (permissions['file_any'].length !== 0) {
for (const p of permissions['file_any']) {
let others_sqon = await readFile('./models/others_sqon.json');
- others_sqon['content']['value'] = [p];
+ others_sqon['content'][1]['content']['value'] = [p];
base_sqon['content'][1]['content'].push(others_sqon);
}
}
if (permissions['file_in_own_namefolder'].length !== 0) {
for (const p of permissions['file_in_own_namefolder']) {
let owner_sqon = await readFile('./models/owner_sqon.json');
- owner_sqon['content'][0]['content']['value'] = [`${username}*`];
+ owner_sqon['content'][0]['content']['value'] = [`users/${username}*`];
owner_sqon['content'][1]['content']['value'] = [p];
base_sqon['content'][1]['content'].push(owner_sqon);
}
}
+
+ // project folders accessible, select these only. User does not have access to all name folders in any zone
+ if (permissions['project_folders_accessible'].length !== 0) {
+ const projectFolder = new ProjectFolder();
+ for (const p of permissions['project_folders_accessible']){
+ const sqon = projectFolder.generateAccessibleSQON(p.zone, p.name);
+ base_sqon['content'][1]['content'].push(sqon);
+ }
+ }
+
// return sqon with user permissions in place
return base_sqon;
} catch (error) {
- console.error(`Failed to build SQON for project ${project_code}: ${error}`)
+ console.error(error)
+ throw Error(`Cannot build SQON for project ${project_code}`)
}
};
@@ -214,23 +302,16 @@ const buildSQON = async (role_metadata, project_code, username) => {
* @param project_code project code for faceted search interface
* @param username username of requester
* @param user_roles user realm roles
+ * @param token bearer token of user
* @returns a JSON encoded SQON filter to apply to graphql query, including aggregations
*/
-export async function arrangerAuthFilter(project_code, username, user_roles) {
- try {
-
- // get rbac permissions (if any)
- const rbac = await getRBAC(project_code, user_roles);
-
- // build sqon
- return await buildSQON(rbac, project_code, username);
-
-
- } catch (error) {
- console.error(`Failed to execute auth filter: ${error} `)
+export async function arrangerAuthFilter(project_code, username, user_roles, token) {
+ // get rbac permissions (if any)
+ const rbac = await getRBAC(project_code, user_roles, token);
- }
+ // build sqon
+ return await buildSQON(rbac, project_code, username);
}
@@ -240,17 +321,12 @@ export async function arrangerAuthFilter(project_code, username, user_roles) {
* @param username username of requester
* @param request_body JSON of the incoming request body
* @param realm_roles keycloak realm roles for user
+ * @param token bearer token of user
* @returns a JSON encoded SQON filter to apply to graphql query, including aggregations
*/
-export async function processRequest(project_code, username, request_body, realm_roles) {
-
- try {
- return await arrangerAuthFilter(project_code, username, realm_roles)
-
- } catch (error) {
- console.log(`Failed to process request: ${error}`)
- }
+export async function processRequest(project_code, username, request_body, realm_roles, token) {
+ return await arrangerAuthFilter(project_code, username, realm_roles, token)
}
diff --git a/modules/server-filter/config.js b/modules/server-filter/config.js
index 121614df7..18efb9057 100644
--- a/modules/server-filter/config.js
+++ b/modules/server-filter/config.js
@@ -3,5 +3,7 @@ export const ARRANGER_PROJECT_ID = process.env.ARRANGER_PROJECT_ID || 'pilotdev'
export const ELASTICSEARCH = process.env.ELASTICSEARCH || 'http://127.0.0.1:9200';
export const AUTH_SERVICE = process.env.AUTH_SERVICE || 'http://127.0.0.1:5061';
export const AUTH_SERVICE_PAGE_SIZE = process.env.AUTH_SERVICE_PAGE_SIZE || 80;
+export const METADATA_SERVICE = process.env.METADATA_SERVICE || 'http://127.0.0.1:5066';
+export const METADATA_SERVICE_PAGE_SIZE = process.env.METADATA_SERVICE_PAGE_SIZE || 80;
export const DOWNLOAD_STREAM_BUFFER_SIZE = process.env.DOWNLOAD_STREAM_BUFFER_SIZE || 2000;
export const MEMOIZE_TIMEOUT = process.env.MEMOIZE_TIMEOUT || 10000;
diff --git a/modules/server-filter/export/utils/authFilterDownload.js b/modules/server-filter/export/utils/authFilterDownload.js
index 024cccb82..a599badaa 100644
--- a/modules/server-filter/export/utils/authFilterDownload.js
+++ b/modules/server-filter/export/utils/authFilterDownload.js
@@ -6,9 +6,8 @@
*/
export function arrangerAuthFilterDownload(req, params, sqon) {
try {
-
// add user-selected identifiers
- if (params['identifiers'].length > 0) {
+ if (params['identifiers'] && params['identifiers'].length > 0) {
const sqon_id = {
"op": "in",
"content": {
@@ -21,6 +20,17 @@ export function arrangerAuthFilterDownload(req, params, sqon) {
}
req.sqon.content.push(sqon_id)
}
+ if (sqon !== null) {
+ if (sqon.content){
+ // add facet selections to sqon
+ for (const filter of sqon.content){
+ req.sqon.content.push(filter)
+
+ }
+ }
+
+ }
+
// return custom sqon
return req.sqon
diff --git a/modules/server-filter/models/others_sqon.json b/modules/server-filter/models/others_sqon.json
index 47d5be19e..296f58af6 100755
--- a/modules/server-filter/models/others_sqon.json
+++ b/modules/server-filter/models/others_sqon.json
@@ -1,9 +1,23 @@
{
- "op":"in",
- "content":{
- "field":"zonefilter",
- "value":[
- "zone"
- ]
- }
+ "op":"and",
+ "content":[
+ {
+ "op":"in",
+ "content":{
+ "field":"parent_path",
+ "value":[
+ "users/*"
+ ]
+ }
+ },
+ {
+ "op":"in",
+ "content":{
+ "field":"zonefilter",
+ "value":[
+ "zone"
+ ]
+ }
+ }
+ ]
}
diff --git a/modules/server-filter/models/owner_sqon.json b/modules/server-filter/models/owner_sqon.json
index 5ea04745b..c83c8e6a9 100644
--- a/modules/server-filter/models/owner_sqon.json
+++ b/modules/server-filter/models/owner_sqon.json
@@ -6,7 +6,7 @@
"content":{
"field":"parent_path",
"value":[
- "user*"
+ "users/user*"
]
}
},
diff --git a/modules/server-filter/package.json b/modules/server-filter/package.json
index f38393683..f1a4511c9 100644
--- a/modules/server-filter/package.json
+++ b/modules/server-filter/package.json
@@ -15,6 +15,7 @@
"@types/express-serve-static-core": "4.17.24",
"apollo-server-express": "^2.14.2",
"axios": "^1.3.4",
+ "cors": "^2.8.5",
"date-fns": "^2.29.2",
"express": "^4.18.2",
"graphql": "^14.5.3",
diff --git a/modules/server-filter/server.js b/modules/server-filter/server.js
index 565109c36..30f095408 100644
--- a/modules/server-filter/server.js
+++ b/modules/server-filter/server.js
@@ -1,4 +1,5 @@
import express from 'express';
+import cors from 'cors';
import { Client } from '@elastic/elasticsearch';
import { ApolloServer } from 'apollo-server-express';
import { downloader } from './export/export-file.js';
@@ -9,7 +10,9 @@ import { ARRANGER_PROJECT_ID, ELASTICSEARCH, SERVER_PORT } from './config.js';
import { createProjectSchema } from '@arranger/server/dist/startProject.js';
// Manually create ES Client, as we need to give arranger access to it directly
-const esClient = new Client({node: ELASTICSEARCH});
+const esClient = new Client({
+ node: ELASTICSEARCH
+});
/**
* Execute server filter on the incoming request
@@ -18,13 +21,33 @@ const esClient = new Client({node: ELASTICSEARCH});
* @param next next middleware function from express server
*/
-async function serverFilter (req, res, next) {
- const decoded = decodeToken(req)
- const project_code = req.body['project_code'];
- const username = decoded['username']
- const realm_roles = decoded['roles']
- req.sqon = await memoizedProcess(project_code, username, JSON.stringify(req.body), realm_roles)
- next()
+async function serverFilter(req, res, next) {
+ const invalid_sqon = {
+ "op": "and",
+ "content": [{
+ "op": "in",
+ "content": {
+ "field": "container_code",
+ "value": ["no_permissions"]
+ }
+ }]
+ }
+ try {
+ if ((!req.headers.authorization) || (!req.body.project_code)) {
+ req.sqon = invalid_sqon;
+ } else {
+ const decoded = decodeToken(req)
+ const project_code = req.body['project_code'];
+ const username = decoded['username']
+ const realm_roles = decoded['roles']
+ req.sqon = await memoizedProcess(project_code, username, JSON.stringify(req.body), realm_roles, req.headers.authorization)
+
+ }
+ } catch (error) {
+ console.error(`Failed to execute server filter: ${error.message} `)
+ req.sqon = invalid_sqon
+ }
+ next()
}
/**
@@ -32,49 +55,69 @@ async function serverFilter (req, res, next) {
* @param context The context provided by the ApolloServer based on the incoming request
* @returns a SQON filter to apply to the entire request and to all aggregations
*/
-function SQONFilter(context){
- return context.req.sqon
+function SQONFilter(context) {
+ return context.req.sqon
}
// Schema for server-side filtering
const arrangerSchema = await createProjectSchema({
- es: esClient,
- id: ARRANGER_PROJECT_ID,
- graphqlOptions: {},
- enableAdmin: false,
- getServerSideFilter: SQONFilter,
+ es: esClient,
+ id: ARRANGER_PROJECT_ID,
+ graphqlOptions: {},
+ enableAdmin: false,
+ getServerSideFilter: SQONFilter,
})
// Schema for download functionality
const arrangerSchemaDownload = await createProjectSchema({
- es: esClient,
- id: ARRANGER_PROJECT_ID,
- graphqlOptions: {},
- enableAdmin: false,
+ es: esClient,
+ id: ARRANGER_PROJECT_ID,
+ graphqlOptions: {},
+ enableAdmin: false,
})
// Setup Apollo Server
const server = new ApolloServer({
- schema: arrangerSchema['schema'],
- context: ({ req }) => {
- return {es: esClient, req, projectId: ARRANGER_PROJECT_ID};
- }
+ schema: arrangerSchema['schema'],
+ context: ({
+ req
+ }) => {
+ return {
+ es: esClient,
+ req,
+ projectId: ARRANGER_PROJECT_ID
+ };
+ }
})
// Setup express server
const app = express();
app.use(express.json()); // support json encoded bodies
+
+// Use serverFilter for incoming requests
app.use(serverFilter)
+// Enable all CORS requests
+app.use(cors())
+
// Add Arranger middleware
-server.applyMiddleware({ app, path: `/${ARRANGER_PROJECT_ID}/graphql`});
-server.applyMiddleware({ app, path: `/${ARRANGER_PROJECT_ID}/graphql/*`});
+server.applyMiddleware({
+ app,
+ path: `/${ARRANGER_PROJECT_ID}/graphql`
+});
+server.applyMiddleware({
+ app,
+ path: `/${ARRANGER_PROJECT_ID}/graphql/*`
+});
// Enable export file
-app.use(`/${ARRANGER_PROJECT_ID}/download`, downloader({ projectId: ARRANGER_PROJECT_ID, es: esClient,
-arranger_schema: arrangerSchemaDownload}))
+app.use(`/${ARRANGER_PROJECT_ID}/download`, downloader({
+ projectId: ARRANGER_PROJECT_ID,
+ es: esClient,
+ arranger_schema: arrangerSchemaDownload
+}))
// Start express server
app.listen(SERVER_PORT, () => {
- console.log(`Arranger server running on ${SERVER_PORT}`);
+ console.log(`Arranger server running on ${SERVER_PORT}`);
});
diff --git a/modules/server-filter/sqonModels.js b/modules/server-filter/sqonModels.js
new file mode 100644
index 000000000..f1f3e9ced
--- /dev/null
+++ b/modules/server-filter/sqonModels.js
@@ -0,0 +1,57 @@
+/**
+ * Class to manage and create shared project folder SQONs
+ */
+export class ProjectFolder {
+ /**
+ * Generate SQON for project folders and their respective files/folders that are user accessible
+ * @param zone zone of project folder
+ * @param folderName name of project folder
+ */
+ generateAccessibleSQON(zone, folderName){
+ return {
+ 'op': 'and',
+ 'content': [{
+ 'op': 'in',
+ 'content': {
+ 'field': 'zonefilter',
+ 'value': [
+ zone,
+ ],
+ },
+ }, {
+ 'op': 'or',
+ 'content': [{
+ 'op': 'in',
+ 'content': {
+ 'field': 'parent_path.keyword',
+ 'value': [
+ `shared/${folderName}*`,
+ ],
+ },
+ },
+ {
+ "op":"and",
+ "content":[
+ {
+ "op":"in",
+ "content":{
+ "field":"type",
+ "value":[
+ "project_folder"
+ ]
+ }
+ },
+ {
+ "op":"in",
+ "content":{
+ "field":"name",
+ "value":[
+ `${folderName}`
+ ]
+ }
+ }
+ ]
+ }]}],
+ };
+ }
+}
diff --git a/package.json b/package.json
index b46439c41..21d159106 100644
--- a/package.json
+++ b/package.json
@@ -1,5 +1,5 @@
{
- "version": "1.0.0",
+ "version": "1.0.6",
"repository": {
"type": "git",
"url": "https://github.com/overture-stack/arranger.git"