├── .gitignore ├── .npmignore ├── CODE_OF_CONDUCT.md ├── CONTRIBUTING.md ├── LICENSE ├── README.md ├── asset ├── architecture.png ├── progress.png └── trigger.png ├── bin └── forecast-mlops-pipeline.ts ├── cdk.json ├── glue ├── postprocess.py └── preprocess.py ├── jest.config.js ├── lambda ├── create-dataset-group │ └── index.py ├── create-dataset-import-job │ └── index.py ├── create-dataset │ └── index.py ├── create-forecast │ └── index.py ├── create-predictor │ └── index.py ├── delete-forecast-resource │ └── index.py └── trigger │ └── index.py ├── lib ├── construct │ ├── glue.ts │ ├── lambda.ts │ ├── state-machine.ts │ └── trigger.ts └── forecast-mlops-pipeline-stack.ts ├── package-lock.json ├── package.json ├── sample-data ├── data.zip └── forecast-config.json ├── test ├── forecast-mlops-pipeline.test.ts └── test.py └── tsconfig.json /.gitignore: -------------------------------------------------------------------------------- 1 | *.js 2 | !jest.config.js 3 | *.d.ts 4 | node_modules 5 | 6 | # CDK asset staging directory 7 | .cdk.staging 8 | cdk.out 9 | 10 | .devcontainer 11 | .DS_Store 12 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | *.ts 2 | !*.d.ts 3 | 4 | # CDK asset staging directory 5 | .cdk.staging 6 | cdk.out 7 | -------------------------------------------------------------------------------- /CODE_OF_CONDUCT.md: -------------------------------------------------------------------------------- 1 | ## Code of Conduct 2 | This project has adopted the [Amazon Open Source Code of Conduct](https://aws.github.io/code-of-conduct). 3 | For more information see the [Code of Conduct FAQ](https://aws.github.io/code-of-conduct-faq) or contact 4 | opensource-codeofconduct@amazon.com with any additional questions or comments. 5 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Contributing Guidelines 2 | 3 | Thank you for your interest in contributing to our project. Whether it's a bug report, new feature, correction, or additional 4 | documentation, we greatly value feedback and contributions from our community. 5 | 6 | Please read through this document before submitting any issues or pull requests to ensure we have all the necessary 7 | information to effectively respond to your bug report or contribution. 8 | 9 | 10 | ## Reporting Bugs/Feature Requests 11 | 12 | We welcome you to use the GitHub issue tracker to report bugs or suggest features. 13 | 14 | When filing an issue, please check existing open, or recently closed, issues to make sure somebody else hasn't already 15 | reported the issue. Please try to include as much information as you can. Details like these are incredibly useful: 16 | 17 | * A reproducible test case or series of steps 18 | * The version of our code being used 19 | * Any modifications you've made relevant to the bug 20 | * Anything unusual about your environment or deployment 21 | 22 | 23 | ## Contributing via Pull Requests 24 | Contributions via pull requests are much appreciated. Before sending us a pull request, please ensure that: 25 | 26 | 1. You are working against the latest source on the *main* branch. 27 | 2. You check existing open, and recently merged, pull requests to make sure someone else hasn't addressed the problem already. 28 | 3. You open an issue to discuss any significant work - we would hate for your time to be wasted. 29 | 30 | To send us a pull request, please: 31 | 32 | 1. Fork the repository. 33 | 2. Modify the source; please focus on the specific change you are contributing. If you also reformat all the code, it will be hard for us to focus on your change. 34 | 3. Ensure local tests pass. 35 | 4. Commit to your fork using clear commit messages. 36 | 5. Send us a pull request, answering any default questions in the pull request interface. 37 | 6. Pay attention to any automated CI failures reported in the pull request, and stay involved in the conversation. 38 | 39 | GitHub provides additional document on [forking a repository](https://help.github.com/articles/fork-a-repo/) and 40 | [creating a pull request](https://help.github.com/articles/creating-a-pull-request/). 41 | 42 | 43 | ## Finding contributions to work on 44 | Looking at the existing issues is a great way to find something to contribute on. As our projects, by default, use the default GitHub issue labels (enhancement/bug/duplicate/help wanted/invalid/question/wontfix), looking at any 'help wanted' issues is a great place to start. 45 | 46 | 47 | ## Code of Conduct 48 | This project has adopted the [Amazon Open Source Code of Conduct](https://aws.github.io/code-of-conduct). 49 | For more information see the [Code of Conduct FAQ](https://aws.github.io/code-of-conduct-faq) or contact 50 | opensource-codeofconduct@amazon.com with any additional questions or comments. 51 | 52 | 53 | ## Security issue notifications 54 | If you discover a potential security issue in this project we ask that you notify AWS/Amazon Security via our [vulnerability reporting page](http://aws.amazon.com/security/vulnerability-reporting/). Please do **not** create a public github issue. 55 | 56 | 57 | ## Licensing 58 | 59 | See the [LICENSE](LICENSE) file for our project's licensing. We will ask you to confirm the licensing of your contribution. 60 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining a copy of 4 | this software and associated documentation files (the "Software"), to deal in 5 | the Software without restriction, including without limitation the rights to 6 | use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of 7 | the Software, and to permit persons to whom the Software is furnished to do so. 8 | 9 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 10 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS 11 | FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR 12 | COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER 13 | IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 14 | CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 15 | 16 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # amazon-forecast-mlops-pipeline-cdk 2 | 3 | > This CDK implementation demonstrates how Amazon Forecast workflow can be automated with AWS Serverless Services. 4 | 5 | ## Prerequisites 6 | 7 | AWS CDK v2 is required to deploy this application. 8 | See [Installation Guide](https://docs.aws.amazon.com/cdk/latest/guide/work-with-cdk-typescript.html) to install AWS CDK in TypeScript. 9 | 10 | Once complete with the previous step, run below command on the root of this project to deploy the pipeline. 11 | 12 | ```bash 13 | cdk deploy ForecastMlopsPipelineStack 14 | ``` 15 | 16 | If successfully deployed, we are ready to launch 🚀🚀🚀 17 | 18 | 19 | ## Solutions Architecture 20 | 21 | ![architecture](asset/architecture.png) 22 | 23 | This package contains a CDK implementation of a MLOps Pipeline for Amazon Forecast. The pipeline automatically performs Amazon Forecast operations and relevant operations. Basically, each of the operations are either implemente with AWS Lambda or AWS Glue, and the entire operations are orchestrated by AWS Step Functions. 24 | 25 | The pipeline consist of the following steps: 26 | 27 | ### Data Preprocessing 28 | Implemented in Glue Python Job. Performs data operations defined in python scripts. The current code does nothing but printing out dataset files, but you can add your own preprocessing logics using pandas, NumPy, sklearn, etc. 29 | 30 | ### DatasetGroup Creation 31 | Implemented in Lambda Function. Creates Amazon Forecast DatasetGroup. 32 | 33 | ### Dataset Creation 34 | Implemented in Lambda Function. Creates Amazon Forecast Datasets. 35 | 36 | ### DatasetImportJob Creataion 37 | Implemented in Lambda Function. Creates Amazon Forecast Dataset Import Job for each Dataset. 38 | 39 | ### Predictor Creation 40 | Implemented in Lambda Function. Creates Amazon Forecast Predictor for training. 41 | 42 | ### Forecast & ForecastExportJob Creation 43 | Implemented in Lambda Function. Creates Amazon Forecast Forecast and ForecastExportJob. After Forecast & ForecastExport is created, forecast result will be saved in Resource Bucket as csv files. 44 | 45 | ### Data Postprocessing 46 | Implemented in Glue Python Job. By default, Amazon forecast exports results in separate files, but this is inconvenient in some cases. Thus, Data Postprocessing Glue Job merges forecast results into a single file. As mentioned in preprocessing step, you can add your own postprocessing logics in this step too, using pandas, NumPy, sklearn, etc. 47 | 48 | ### Resource Cleanup 49 | Implemented in Lambda Function. Cleans up Amazon Forecast Resources created in above steps. Result files in S3 Bucket will not be deleted even if Forecast Resources are delete. You can either skip this step by configuring `PerformDelete` to `false` in __forecast-config.json__. 50 | 51 | 52 | ## How to Run 53 | 54 | This section explains how to trigger the pipeline to run Amazon Forecast from end to end. Follow the instructions in order. 55 | 56 | ### Prepare Dataset 57 | 58 | Prepare your own dataset in csv files formatted for Amazon Forecast. 59 | See this [link](https://docs.aws.amazon.com/forecast/latest/dg/howitworks-datasets-groups.html) for details aboout Amazon Forecast dataset format. 60 | 61 | For those who want to execute the pipeline before preparing your own dataset, sample dataset is included in this project. Please see __sample-data/data.zip__. 62 | 63 | > In __sample-data/data.zip__, `TTS.csv` contains monthly sales history of instruments in Amazon Forecast Target Time Series format. `IM.csv` contains metadata, which instrument category of each item, in Amazon Forecast Item Metadata format. 64 | 65 | When you prepare your own dataset, be sure to match the directory structure of the dataset file __data.zip__. The directory structure should look like below. 66 | 67 | ```bash 68 | data.zip 69 | │────TTS.csv # mandatory 70 | │────IM.csv # optional, may not exist 71 | └────RTS.csv # optional, may not exist 72 | ``` 73 | 74 | > Please note that the names of csv files should be one of `TTS.csv` / `IM.csv` / `RTS.csv`. 75 | 76 | 77 | ### Prepare Configuration JSON 78 | 79 | To use Amazon Forecast, you have to find out proper values for a bunch of configurations. For example, which algorithm to train, how to fill missing value, etc. 80 | 81 | To tell the pipeline all the detailed configurations, you have to write a json file and put it into Resource Bucket. 82 | 83 | As well as the dataset, an example for configurations is also provided within this project. Please see __sample-data/forecast-config.json__. This example is for training a CNNQR model with the dataset given in __sample-data/data.zip__. 84 | 85 | Otherwise, you can use this pipeline on your own dataset. If you want to do so, make sure to properly setup __forecast-config.json__ by referring to __sample-data/forecast-config.json__. 86 | 87 | Since this pipeline uses boto3 package for Amazon Forecast API calls, the schema of __forecast-config.json__ corresponds to boto3 API parameter schema. To add other configuration variable that doesn't appear on __sample-data/forecast-config.json__, see [Boto3 Developer Guide](https://boto3.amazonaws.com/v1/documentation/api/latest/reference/services/forecast.html). 88 | 89 | 90 | ### Upload to Launch 91 | 92 | Finally, you are ready to launch the pipeline. After you deploy this pipeline, 93 | you will see an S3 bucket named `forecast-mlops-pipeline-resource-bucket-{YOUR-12DIGIT-AWS-ACCOUNT-ID}`. All you have to do now is to upload files prepared from the previous steps. 94 | 95 | First, you have to upload your __forecast-config.json__ to the bucket by either using AWS CLI or AWS Console: 96 | 97 | ```bash 98 | aws s3 cp forecast-config.json s3://forecast-mlops-pipeline-resource-bucket-{YOUR-12DIGIT-AWS-ACCOUNT-ID} 99 | ``` 100 | 101 | And then upload your on data.zip file to __raw/__ directory in the same bucket. 102 | 103 | ```bash 104 | aws s3 cp data.zip s3://forecast-mlops-pipeline-resource-bucket-{YOUR-12DIGIT-AWS-ACCOUNT-ID}/raw 105 | ``` 106 | 107 | > You should upload `data.zip` as the final step because `raw/*.zip` is registered as the upload trigger key of the pipeline. Therefore the pipeline will be triggered immediately after `raw/data.zip` is uploaded. To avoid any unintended failure, upload `forecast-config.json` files before the dataset, `raw/data.zip`. 108 | 109 | ![trigger](asset/trigger.png) 110 | 111 | Navigate to the StepFunctions console to monitor the excecution. You will see the progress like below while execution. 112 | 113 | ![progress](asset/progress.png) 114 | 115 | When the steps are completed, the __Resource Bucket__ will have the following structure: 116 | 117 | ```bash 118 | forecast-config.json # Your pipeline configuration file. 119 | raw/ # Where you uploaded the raw dataset 120 | input/ # Where preprocessed csv files are stored 121 | export/ # Where raw forecast csvs generated by Amazon Forecast are stored 122 | forecast/ # Where postprocessed and merged csv file is stored 123 | ``` 124 | 125 | 126 | ## Project Structure 127 | This project follows typical AWS CDK project structure. The below directories are important from the user perspective. 128 | 129 | ```bash 130 | bin/ # Entrypoint of the entire project 131 | lib/ # CDK Constructs and Stacks are defined 132 | lambda/ # Python scripts for Lambda Functions 133 | glue/ # Python scripts for Glue Jobs 134 | ``` 135 | 136 | 137 | ## 👀 Got Problems? 138 | 139 | Please open an issue to let the maintainer know. Or PR would be also appreciated. 140 | The maintainer will reach out! 141 | 142 | ## Security 143 | 144 | See [CONTRIBUTING](CONTRIBUTING.md#security-issue-notifications) for more information. 145 | 146 | ## License 147 | 148 | This library is licensed under the MIT-0 License. See the LICENSE file. 149 | -------------------------------------------------------------------------------- /asset/architecture.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/aws-samples/amazon-forecast-mlops-pipeline-cdk/4c2bd451e6abc1809150d86f778da2516ef0a89a/asset/architecture.png -------------------------------------------------------------------------------- /asset/progress.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/aws-samples/amazon-forecast-mlops-pipeline-cdk/4c2bd451e6abc1809150d86f778da2516ef0a89a/asset/progress.png -------------------------------------------------------------------------------- /asset/trigger.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/aws-samples/amazon-forecast-mlops-pipeline-cdk/4c2bd451e6abc1809150d86f778da2516ef0a89a/asset/trigger.png -------------------------------------------------------------------------------- /bin/forecast-mlops-pipeline.ts: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | import 'source-map-support/register'; 4 | import * as cdk from 'aws-cdk-lib'; 5 | import { ForecastMlopsPipelineStack } from '../lib/forecast-mlops-pipeline-stack'; 6 | 7 | const app = new cdk.App(); 8 | const fcstMlopsPplStack = new ForecastMlopsPipelineStack( 9 | app, 'ForecastMlopsPipelineStack', {} 10 | ); -------------------------------------------------------------------------------- /cdk.json: -------------------------------------------------------------------------------- 1 | { 2 | "app": "npx ts-node --prefer-ts-exts bin/forecast-mlops-pipeline.ts", 3 | "watch": { 4 | "include": [ 5 | "**" 6 | ], 7 | "exclude": [ 8 | "README.md", 9 | "cdk*.json", 10 | "**/*.d.ts", 11 | "**/*.js", 12 | "tsconfig.json", 13 | "package*.json", 14 | "yarn.lock", 15 | "node_modules", 16 | "test" 17 | ] 18 | }, 19 | "context": { 20 | "@aws-cdk/aws-apigateway:usagePlanKeyOrderInsensitiveId": true, 21 | "@aws-cdk/core:stackRelativeExports": true, 22 | "@aws-cdk/aws-rds:lowercaseDbIdentifier": true, 23 | "@aws-cdk/aws-lambda:recognizeVersionProps": true, 24 | "@aws-cdk/aws-lambda:recognizeLayerVersion": true, 25 | "@aws-cdk/aws-cloudfront:defaultSecurityPolicyTLSv1.2_2021": true, 26 | "@aws-cdk-containers/ecs-service-extensions:enableDefaultLogDriver": true, 27 | "@aws-cdk/aws-ec2:uniqueImdsv2TemplateName": true, 28 | "@aws-cdk/core:checkSecretUsage": true, 29 | "@aws-cdk/aws-iam:minimizePolicies": true, 30 | "@aws-cdk/core:target-partitions": [ 31 | "aws", 32 | "aws-cn" 33 | ] 34 | } 35 | } 36 | -------------------------------------------------------------------------------- /glue/postprocess.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | # coding: utf-8 3 | 4 | # Import pandas library for csv import 5 | import zipfile 6 | import boto3 7 | import pandas as pd 8 | import sys 9 | import os 10 | from awsglue.utils import getResolvedOptions 11 | 12 | 13 | s3 = boto3.client('s3') 14 | 15 | 16 | def download_csvs(bucket, export_dir_key, local_dst): 17 | os.makedirs(local_dst, exist_ok=True) 18 | 19 | obj_list = s3.list_objects(Bucket=bucket, Prefix=export_dir_key)['Contents'] 20 | 21 | for obj in obj_list: 22 | if obj['Key'].endswith('.csv'): 23 | basename = os.path.basename(obj['Key']) 24 | s3.download_file(bucket, obj['Key'], os.path.join(local_dst, basename)) 25 | 26 | 27 | # merge split csvs into a single one 28 | def merge(csv_dir, dst_path): 29 | split_list = os.listdir(local_dst) 30 | 31 | df_list = [] 32 | for split_csv in split_list: 33 | appendant = pd.read_csv(os.path.join(local_dst, split_csv)) 34 | df_list.append(appendant) 35 | 36 | merged_df = pd.concat(df_list, axis=1, join='inner').sort_index() 37 | merged_df.to_csv(dst_path, index=False) 38 | 39 | 40 | def save_to_s3(bucket, merged_csv_path, s3_key): 41 | s3.upload_file(merged_csv_path, bucket, s3_key) 42 | 43 | 44 | # main 45 | args = getResolvedOptions(sys.argv, ['bucket', 'uid']) 46 | bucket = args['bucket'] 47 | uid = args['uid'] 48 | 49 | export_dir_key = f'export/{uid}' 50 | local_dst = '/tmp/export' 51 | 52 | download_csvs(bucket, export_dir_key, local_dst) 53 | merged_csv_path = '/tmp/merged.csv' 54 | merge(local_dst, merged_csv_path) 55 | 56 | s3_key = f'forecast/{uid}.csv' 57 | save_to_s3(bucket, merged_csv_path, s3_key) 58 | -------------------------------------------------------------------------------- /glue/preprocess.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | # coding: utf-8 3 | 4 | # Import pandas library for csv import 5 | import zipfile 6 | import boto3 7 | import pandas as pd 8 | import sys 9 | import os 10 | from awsglue.utils import getResolvedOptions 11 | 12 | s3 = boto3.client('s3') 13 | 14 | 15 | def download_and_extract(bucket, fileuri, csv_dir): 16 | dst_path = os.path.join('/tmp', fileuri) 17 | os.makedirs(os.path.dirname(dst_path), exist_ok=True) 18 | 19 | s3.download_file(bucket, fileuri, dst_path) 20 | 21 | with zipfile.ZipFile(dst_path, 'r') as zipf: 22 | zipf.extractall(csv_dir) 23 | 24 | 25 | def preprocess(csv_dir): 26 | # Load csv 27 | tts_df = pd.read_csv(os.path.join(csv_dir, 'TTS.csv')) 28 | im_df = pd.read_csv(os.path.join(csv_dir, 'IM.csv')) 29 | # and do nothing because our sample dataset has been refined already 30 | # you can implement your own preprocessing logic here if you want to 31 | 32 | def save_to_s3(bucket, csv_dir): 33 | csv_list = os.listdir(csv_dir) 34 | 35 | for single_csv in csv_list: 36 | object_key = os.path.join('input/', os.path.basename(single_csv)) 37 | s3.upload_file(os.path.join(csv_dir, single_csv), bucket, object_key) 38 | 39 | 40 | # main 41 | args = getResolvedOptions(sys.argv, ['bucket', 'fileuri']) 42 | bucket = args['bucket'] 43 | fileuri = args['fileuri'] 44 | 45 | csv_dir = 'input/' 46 | download_and_extract(bucket, fileuri, csv_dir) 47 | preprocess(csv_dir) 48 | save_to_s3(bucket, csv_dir) 49 | -------------------------------------------------------------------------------- /jest.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | testEnvironment: 'node', 3 | roots: ['/test'], 4 | testMatch: ['**/*.test.ts'], 5 | transform: { 6 | '^.+\\.tsx?$': 'ts-jest' 7 | } 8 | }; 9 | -------------------------------------------------------------------------------- /lambda/create-dataset-group/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | 4 | forecast = boto3.client('forecast') 5 | 6 | def handler(event, context): 7 | account_id = context.invoked_function_arn.split(":")[4] 8 | region = os.environ['AWS_REGION'] 9 | 10 | dataset_group = event['config']['DatasetGroup'] 11 | dataset_group_arn = f'arn:aws:forecast:{region}:{account_id}:dataset-group/{dataset_group["DatasetGroupName"]}' 12 | 13 | dataset_dict = event['config']['Dataset'] 14 | 15 | dataset_group['DatasetArns'] = [] 16 | for kind, dataset in dataset_dict.items(): 17 | dataset_group['DatasetArns'].append(dataset['Arn']) 18 | 19 | status = None 20 | 21 | try: 22 | status = forecast.describe_dataset_group( 23 | DatasetGroupArn=dataset_group_arn 24 | )['Status'] 25 | except forecast.exceptions.ResourceNotFoundException: 26 | print(f'Dataset Group not found! Creating: {dataset_group_arn}') 27 | 28 | forecast.create_dataset_group( 29 | **dataset_group 30 | ) 31 | 32 | status = forecast.describe_dataset_group( 33 | DatasetGroupArn=dataset_group_arn 34 | )['Status'] 35 | 36 | dataset_group['Arn'] = dataset_group_arn 37 | 38 | if status in ['CREATE_PENDING', 'CREATE_IN_PROGRESS']: 39 | raise ResourcePending 40 | 41 | return event 42 | 43 | 44 | class ResourcePending(Exception): 45 | pass 46 | -------------------------------------------------------------------------------- /lambda/create-dataset-import-job/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | 4 | forecast = boto3.client('forecast') 5 | 6 | def handler(event, context): 7 | uid = event['uid'] 8 | account_id = context.invoked_function_arn.split(":")[4] 9 | region = os.environ['AWS_REGION'] 10 | 11 | config = event['config'] 12 | dataset_dict = config['Dataset'] 13 | status = None 14 | 15 | for kind, dataset in dataset_dict.items(): 16 | import_job_arn = f'arn:aws:forecast:{region}:{account_id}:dataset-import-job/{dataset["DatasetName"]}/{dataset["DatasetName"]}_{uid}' 17 | 18 | try: 19 | status = forecast.describe_dataset_import_job( 20 | DatasetImportJobArn=import_job_arn 21 | )['Status'] 22 | except forecast.exceptions.ResourceNotFoundException: 23 | print( 24 | f'Dataset Import Job not found! Creating {kind} Import Job: {import_job_arn}' 25 | ) 26 | 27 | forecast.create_dataset_import_job( 28 | DatasetImportJobName=f'{dataset["DatasetName"]}_{uid}', 29 | DatasetArn=dataset['Arn'], 30 | DataSource={ 31 | 'S3Config': { 32 | 'Path': dataset['CsvS3Uri'], 33 | 'RoleArn': os.environ['FORECAST_ROLE'], 34 | } 35 | }, 36 | TimestampFormat=config['TimestampFormat'] 37 | ) 38 | 39 | status = forecast.describe_dataset_import_job( 40 | DatasetImportJobArn=import_job_arn 41 | )['Status'] 42 | 43 | dataset['ImportJobArn'] = import_job_arn 44 | 45 | if status in ['CREATE_PENDING', 'CREATE_IN_PROGRESS']: 46 | raise ResourcePending 47 | 48 | return event 49 | 50 | 51 | class ResourcePending(Exception): 52 | pass 53 | -------------------------------------------------------------------------------- /lambda/create-dataset/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | 4 | forecast = boto3.client('forecast') 5 | RESOURCE_BUCKET = os.environ['RESOURCE_BUCKET'] 6 | 7 | def handler(event, context): 8 | account_id = context.invoked_function_arn.split(":")[4] 9 | region = os.environ['AWS_REGION'] 10 | 11 | dataset_dict = event['config']['Dataset'] 12 | status = None 13 | 14 | for kind, dataset in dataset_dict.items(): 15 | csv_s3_uri = f's3://{RESOURCE_BUCKET}/input/{kind}.csv' 16 | dataset_arn = f'arn:aws:forecast:{region}:{account_id}:dataset/{dataset["DatasetName"]}' 17 | 18 | try: 19 | status = forecast.describe_dataset( 20 | DatasetArn=dataset_arn 21 | )['Status'] 22 | except forecast.exceptions.ResourceNotFoundException: 23 | print(f'Dataset not found! Creating {kind}: {dataset_arn}') 24 | 25 | forecast.create_dataset(**dataset) 26 | 27 | status = forecast.describe_dataset( 28 | DatasetArn=dataset_arn 29 | )['Status'] 30 | 31 | dataset['Arn'] = dataset_arn 32 | dataset['CsvS3Uri'] = csv_s3_uri 33 | 34 | if status in ['CREATE_PENDING', 'CREATE_IN_PROGRESS']: 35 | raise ResourcePending 36 | 37 | return event 38 | 39 | 40 | class ResourcePending(Exception): 41 | pass 42 | -------------------------------------------------------------------------------- /lambda/create-forecast/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | 4 | forecast = boto3.client('forecast') 5 | cw = boto3.client('cloudwatch') 6 | 7 | def post_metric(metrics): 8 | for metric in metrics['PredictorEvaluationResults']: 9 | cw.put_metric_data( 10 | Namespace='FORECAST', 11 | MetricData=[ 12 | { 13 | 'Dimensions': 14 | [ 15 | { 16 | 'Name': 'Algorithm', 17 | 'Value': metric['AlgorithmArn'] 18 | }, { 19 | 'Name': 'Quantile', 20 | 'Value': str(quantile['Quantile']) 21 | } 22 | ], 23 | 'MetricName': 'WQL', 24 | 'Unit': 'None', 25 | 'Value': quantile['LossValue'] 26 | } for quantile in metric['TestWindows'][0]['Metrics'] 27 | ['WeightedQuantileLosses'] 28 | ] + [ 29 | { 30 | 'Dimensions': 31 | [ 32 | { 33 | 'Name': 'Algorithm', 34 | 'Value': metric['AlgorithmArn'] 35 | } 36 | ], 37 | 'MetricName': 'RMSE', 38 | 'Unit': 'None', 39 | 'Value': metric['TestWindows'][0]['Metrics']['RMSE'] 40 | } 41 | ] 42 | ) 43 | 44 | 45 | def handler(event, context): 46 | account_id = context.invoked_function_arn.split(":")[4] 47 | region = os.environ['AWS_REGION'] 48 | 49 | uid = event['uid'] 50 | predictor_arn = event['config']['Predictor']['Arn'] 51 | forecast_config = event['config']['Forecast'] 52 | forecast_config['ForecastName'] += '_' + uid 53 | forecast_arn = f'arn:aws:forecast:{region}:{account_id}:forecast/{forecast_config["ForecastName"]}' 54 | 55 | status = None 56 | 57 | # create forecast 58 | try: 59 | status = forecast.describe_forecast( 60 | ForecastArn=forecast_arn 61 | )['Status'] 62 | except forecast.exceptions.ResourceNotFoundException: 63 | post_metric( 64 | forecast.get_accuracy_metrics( 65 | PredictorArn=predictor_arn 66 | ) 67 | ) 68 | 69 | print(f'Forecast not found! Creating: {forecast_arn}') 70 | 71 | forecast.create_forecast( 72 | **forecast_config, PredictorArn=predictor_arn 73 | ) 74 | 75 | status = forecast.describe_forecast( 76 | ForecastArn=forecast_arn 77 | )['Status'] 78 | 79 | forecast_config['Arn'] = forecast_arn 80 | 81 | if status in ['CREATE_PENDING', 'CREATE_IN_PROGRESS']: 82 | raise ResourcePending 83 | 84 | # put metrics int context 85 | metrics = forecast.get_accuracy_metrics( 86 | PredictorArn=predictor_arn 87 | )['PredictorEvaluationResults'][0] 88 | 89 | for row in metrics['TestWindows']: 90 | if row['EvaluationType'] == 'COMPUTED': 91 | row['TestWindowStart'] = row['TestWindowStart'].strftime("%Y%m%dT%H%M%S") 92 | row['TestWindowEnd'] = row['TestWindowEnd'].strftime("%Y%m%dT%H%M%S") 93 | 94 | event['metrics'] = metrics 95 | 96 | # create forecast export job 97 | export_job_arn = f'arn:aws:forecast:{region}:{account_id}:forecast-export-job/{forecast_config["ForecastName"]}/{forecast_config["ForecastName"]}' 98 | 99 | try: 100 | status = forecast.describe_forecast_export_job( 101 | ForecastExportJobArn=export_job_arn 102 | )['Status'] 103 | except forecast.exceptions.ResourceNotFoundException: 104 | print(f'Forecast export job not found! Creating: {export_job_arn}') 105 | 106 | forecast.create_forecast_export_job( 107 | ForecastExportJobName=forecast_config['ForecastName'], 108 | ForecastArn=forecast_arn, 109 | Destination={ 110 | 'S3Config': 111 | { 112 | 'Path': f's3://{os.environ["RESOURCE_BUCKET"]}/export/{uid}/', 113 | 'RoleArn': os.environ['FORECAST_ROLE'] 114 | } 115 | } 116 | ) 117 | status = forecast.describe_forecast_export_job( 118 | ForecastExportJobArn=export_job_arn 119 | )['Status'] 120 | 121 | forecast_config['ExportJobArn'] = export_job_arn 122 | 123 | if status in ['CREATE_PENDING', 'CREATE_IN_PROGRESS']: 124 | raise ResourcePending 125 | 126 | return event 127 | 128 | 129 | class ResourcePending(Exception): 130 | pass 131 | -------------------------------------------------------------------------------- /lambda/create-predictor/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | 4 | forecast = boto3.client('forecast') 5 | 6 | def handler(event, context): 7 | account_id = context.invoked_function_arn.split(":")[4] 8 | region = os.environ['AWS_REGION'] 9 | 10 | uid = event['uid'] 11 | predictor = event['config']['Predictor'] 12 | predictor['PredictorName'] += '_' + uid 13 | predictor_arn = f'arn:aws:forecast:{region}:{account_id}:predictor/{predictor["PredictorName"]}' 14 | 15 | status = None 16 | 17 | try: 18 | status = forecast.describe_predictor( 19 | PredictorArn=predictor_arn 20 | )['Status'] 21 | except forecast.exceptions.ResourceNotFoundException: 22 | print(f'Predictor not found! Creating: {predictor_arn}') 23 | 24 | if 'InputDataConfig' in predictor.keys(): 25 | predictor['InputDataConfig']['DatasetGroupArn'] = event['config']['DatasetGroup']['Arn'] 26 | else: 27 | predictor['InputDataConfig'] = { 28 | 'DatasetGroupArn': event['config']['DatasetGroup']['Arn'] 29 | } 30 | 31 | forecast.create_predictor( 32 | **predictor 33 | ) 34 | 35 | status = forecast.describe_predictor( 36 | PredictorArn=predictor_arn 37 | )['Status'] 38 | 39 | predictor['Arn'] = predictor_arn 40 | 41 | if status in ['CREATE_PENDING', 'CREATE_IN_PROGRESS']: 42 | raise ResourcePending 43 | 44 | return event 45 | 46 | 47 | class ResourcePending(Exception): 48 | pass 49 | -------------------------------------------------------------------------------- /lambda/delete-forecast-resource/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | import time 4 | 5 | 6 | forecast = boto3.client('forecast') 7 | 8 | 9 | def delete_dataset_import_jobs(dataset_dict): 10 | for kind, dataset in dataset_dict.items(): 11 | try: 12 | status = forecast.describe_dataset_import_job( 13 | DatasetImportJobArn=dataset['ImportJobArn'] 14 | )['Status'] 15 | 16 | if status in ['ACTIVE']: 17 | forecast.delete_dataset_import_job( 18 | DatasetImportJobArn=dataset['ImportJobArn'] 19 | ) 20 | 21 | raise ResourcePending 22 | 23 | elif status in ['DELETE_PENDING', 'DELETE_IN_PROGRESS']: 24 | raise ResourcePending 25 | 26 | except forecast.exceptions.ResourceNotFoundException: 27 | pass 28 | 29 | 30 | def delete_datasets(dataset_dict): 31 | for kind, dataset in dataset_dict.items(): 32 | try: 33 | status = forecast.describe_dataset( 34 | DatasetArn=dataset['Arn'] 35 | )['Status'] 36 | 37 | if status in ['ACTIVE']: 38 | forecast.delete_dataset( 39 | DatasetArn=dataset['Arn'] 40 | ) 41 | raise ResourcePending 42 | 43 | elif status in ['DELETE_PENDING', 'DELETE_IN_PROGRESS']: 44 | raise ResourcePending 45 | 46 | except forecast.exceptions.ResourceNotFoundException: 47 | pass 48 | 49 | 50 | def delete_resource_tree(dataset_group_arn): 51 | try: 52 | status = forecast.describe_dataset_group( 53 | DatasetGroupArn=dataset_group_arn 54 | )['Status'] 55 | 56 | if status in ['ACTIVE']: 57 | forecast.delete_resource_tree( 58 | ResourceArn=dataset_group_arn 59 | ) 60 | raise ResourcePending 61 | 62 | elif status in ['DELETE_PENDING', 'DELETE_IN_PROGRESS']: 63 | raise ResourcePending 64 | 65 | except forecast.exceptions.ResourceNotFoundException: 66 | pass 67 | 68 | 69 | def handler(event, context): 70 | dataset_dict = event['config']['Dataset'] 71 | dataset_group_arn = event['config']['DatasetGroup']['Arn'] 72 | 73 | delete_resource_tree(dataset_group_arn) 74 | 75 | delete_dataset_import_jobs(dataset_dict) 76 | delete_datasets(dataset_dict) 77 | 78 | return event 79 | 80 | 81 | class ResourcePending(Exception): 82 | pass 83 | -------------------------------------------------------------------------------- /lambda/trigger/index.py: -------------------------------------------------------------------------------- 1 | import os 2 | import boto3 3 | from json import loads, dumps 4 | from datetime import datetime, timedelta 5 | 6 | sfn = boto3.client('stepfunctions') 7 | s3 = boto3. client('s3') 8 | 9 | 10 | def get_config(bucket_name, key_name): 11 | config = loads( 12 | s3.get_object( 13 | Bucket=bucket_name, Key=key_name 14 | )['Body'].read().decode('utf-8') 15 | ) 16 | # validate(params, SCHEMA_DEF) 17 | return config 18 | 19 | 20 | def handler(event, context): 21 | bucket_name = event['Records'][0]['s3']['bucket']['name'] 22 | 23 | ts = datetime.now() # note TZ is UTC 24 | ts = ts.strftime("%Y%m%dT%H%M%S") 25 | 26 | return dumps( 27 | sfn.start_execution( 28 | stateMachineArn=os.environ['STEP_FUNCTIONS_ARN'], 29 | name=ts, 30 | input=dumps( 31 | { 32 | 'uid': ts, 33 | 'config': get_config(bucket_name, 'forecast-config.json') 34 | } 35 | ) 36 | ), 37 | default=str 38 | ) 39 | -------------------------------------------------------------------------------- /lib/construct/glue.ts: -------------------------------------------------------------------------------- 1 | import { Construct } from 'constructs'; 2 | import { 3 | aws_iam as iam, 4 | aws_s3 as s3, 5 | aws_s3_deployment as s3Deployment, 6 | aws_stepfunctions as sfn, 7 | aws_stepfunctions_tasks as tasks 8 | } from 'aws-cdk-lib'; 9 | 10 | import * as glue from "@aws-cdk/aws-glue-alpha"; 11 | 12 | export interface GlueConstructProps { 13 | taskName: string, 14 | pythonFilePath: string, 15 | defaultArguments?: { 16 | [key:string]: string; 17 | }, 18 | arguments?: { 19 | [key:string]: any; 20 | } 21 | } 22 | 23 | export class GlueConstruct extends Construct { 24 | public readonly role: iam.Role; 25 | public readonly task: tasks.GlueStartJobRun; 26 | 27 | constructor(scope: Construct, id: string, props: GlueConstructProps) { 28 | super(scope, id); 29 | 30 | // IAM Role 31 | this.role = new iam.Role(this, '${props.taskName}GlueRole', { 32 | assumedBy: new iam.ServicePrincipal('glue.amazonaws.com'), 33 | roleName: `${props.taskName}-Glue-Role`, 34 | managedPolicies: [ 35 | {managedPolicyArn: 'arn:aws:iam::aws:policy/AmazonS3FullAccess'}, 36 | {managedPolicyArn: 'arn:aws:iam::aws:policy/service-role/AWSGlueServiceRole'}, 37 | ], 38 | }); 39 | 40 | // Glue Python Job 41 | const pythonJob = new glue.Job(this, `${props.taskName}Job`, { 42 | executable: glue.JobExecutable.pythonShell({ 43 | glueVersion: glue.GlueVersion.V1_0, 44 | pythonVersion: glue.PythonVersion.THREE, 45 | script: glue.Code.fromAsset(props.pythonFilePath), 46 | }), 47 | role: this.role, 48 | jobName: props.taskName, 49 | maxCapacity: 1, 50 | defaultArguments: props.defaultArguments, 51 | }); 52 | 53 | // StepFunction Task 54 | this.task = new tasks.GlueStartJobRun( 55 | this, 56 | props.taskName, 57 | { 58 | glueJobName: pythonJob.jobName, 59 | integrationPattern: sfn.IntegrationPattern.RUN_JOB, 60 | resultPath: sfn.JsonPath.stringAt('$.result'), 61 | arguments: sfn.TaskInput.fromObject(props.arguments!), 62 | } 63 | ); 64 | 65 | } 66 | } -------------------------------------------------------------------------------- /lib/construct/lambda.ts: -------------------------------------------------------------------------------- 1 | import { Construct } from 'constructs'; 2 | import { 3 | aws_glue as glue, 4 | aws_iam as iam, 5 | aws_lambda as lambda, 6 | aws_s3 as s3, 7 | aws_s3_deployment as s3Deployment, 8 | aws_stepfunctions as sfn, 9 | aws_stepfunctions_tasks as tasks, 10 | Duration 11 | } from 'aws-cdk-lib'; 12 | 13 | export interface LambdaConstructProps { 14 | taskName: string; 15 | lambdaCodePath: string; 16 | timeout: Duration; 17 | environment?: { 18 | [key:string]: string; 19 | }; 20 | } 21 | 22 | export class LambdaConstruct extends Construct { 23 | 24 | public readonly role: iam.Role; 25 | public readonly lambda: lambda.Function; 26 | public readonly task: sfn.TaskStateBase; 27 | 28 | constructor(scope: Construct, id: string, props: LambdaConstructProps) { 29 | super(scope, id); 30 | 31 | // IAM Role 32 | this.role = new iam.Role(this, `${props.taskName}LambdaRole`, { 33 | assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'), 34 | roleName: `${props.taskName}-Lambda-Role`, 35 | managedPolicies: [ 36 | {managedPolicyArn: 'arn:aws:iam::aws:policy/AmazonS3FullAccess'}, 37 | {managedPolicyArn: 'arn:aws:iam::aws:policy/CloudWatchFullAccess'}, 38 | {managedPolicyArn: 'arn:aws:iam::aws:policy/AmazonForecastFullAccess'} 39 | ], 40 | }); 41 | 42 | // Lambda Function 43 | this.lambda = new lambda.Function( 44 | this, `${props.taskName}Lambda`, 45 | { 46 | code: lambda.Code.fromAsset(props.lambdaCodePath), 47 | handler: 'index.handler', 48 | functionName: props.taskName, 49 | runtime: lambda.Runtime.PYTHON_3_7, 50 | timeout: props.timeout, 51 | role: this.role, 52 | environment: props.environment, 53 | }); 54 | 55 | // StepFunctions Task 56 | this.task = new tasks.LambdaInvoke( 57 | this, 58 | `${props.taskName}`, 59 | { 60 | lambdaFunction: this.lambda, 61 | integrationPattern: sfn.IntegrationPattern.REQUEST_RESPONSE, 62 | resultPath: sfn.JsonPath.stringAt('$'), 63 | outputPath: sfn.JsonPath.stringAt('$.Payload'), 64 | } 65 | ); 66 | 67 | this.task.addRetry({ 68 | backoffRate: 1.0, 69 | errors: ["ResourcePending"], 70 | interval: Duration.seconds(30), 71 | maxAttempts: 600 72 | }); 73 | 74 | } 75 | 76 | } -------------------------------------------------------------------------------- /lib/construct/state-machine.ts: -------------------------------------------------------------------------------- 1 | import { Construct } from 'constructs'; 2 | import { Duration } from 'aws-cdk-lib'; 3 | import { 4 | aws_ec2 as ec2, 5 | aws_iam as iam, 6 | aws_logs as logs, 7 | aws_s3 as s3, 8 | aws_stepfunctions as sfn, 9 | aws_stepfunctions_tasks as tasks 10 | } from 'aws-cdk-lib'; 11 | 12 | import { GlueConstruct } from './glue'; 13 | import { LambdaConstruct } from './lambda'; 14 | import { TriggerConstruct } from './trigger'; 15 | 16 | export interface StateMachineProps { 17 | resourceBucket: s3.Bucket; 18 | } 19 | 20 | export class StateMachine extends Construct { 21 | constructor(scope: Construct, id: string, props: StateMachineProps) { 22 | super(scope, id); 23 | 24 | const resourceBucket = props.resourceBucket; 25 | 26 | // IAM Role to pass to Forecast 27 | const forecastRole = new iam.Role(this, 'ForecastRole', { 28 | assumedBy: new iam.ServicePrincipal('forecast.amazonaws.com'), 29 | roleName: 'ForecastMLOpsPipeline-ForecastRole', 30 | managedPolicies: [ 31 | {managedPolicyArn: 'arn:aws:iam::aws:policy/CloudWatchFullAccess'}, 32 | {managedPolicyArn: 'arn:aws:iam::aws:policy/AmazonS3FullAccess'}, 33 | ] 34 | }); 35 | 36 | // Preprocessing 37 | const preprocess = new GlueConstruct(this, 'ForecastMLOpsPreprocess', { 38 | taskName: 'Forecast-MLOps-Preprocess', 39 | pythonFilePath: 'glue/preprocess.py', 40 | defaultArguments: { 41 | '--bucket': props.resourceBucket.bucketName, 42 | '--fileuri': 'raw/data.zip', 43 | }, 44 | }); 45 | 46 | // Create Dataset 47 | const createDataset = new LambdaConstruct( 48 | this, 'ForecastMLOpsCreateDataset', 49 | { 50 | taskName: 'Forecast-MLOps-Create-Dataset', 51 | lambdaCodePath: 'lambda/create-dataset', 52 | timeout: Duration.seconds(30), 53 | environment: { 54 | RESOURCE_BUCKET: props.resourceBucket.bucketName, 55 | } 56 | } 57 | ); 58 | 59 | // Create DatasetGroup 60 | const createDatasetGroup = new LambdaConstruct( 61 | this, 'ForecastMLOpsCreateDatasetGroup', 62 | { 63 | taskName: 'Forecast-MLOps-Create-Dataset-Group', 64 | lambdaCodePath: 'lambda/create-dataset-group', 65 | timeout: Duration.seconds(30), 66 | } 67 | ); 68 | 69 | // Create DatasetImportJob 70 | const createDatasetImportJob = new LambdaConstruct( 71 | this, 'ForecastMLOpsCreateDatasetImportJob', 72 | { 73 | taskName: 'Forecast-MLOps-Create-Dataset-Import-Job', 74 | lambdaCodePath: 'lambda/create-dataset-import-job', 75 | timeout: Duration.seconds(30), 76 | environment: { 77 | FORECAST_ROLE: forecastRole.roleArn, 78 | } 79 | } 80 | ); 81 | 82 | // Create Predictor 83 | const createPredictor = new LambdaConstruct( 84 | this, 'ForecastMLOpsCreatePredictor', 85 | { 86 | taskName: 'Forecast-MLOps-Create-Predictor', 87 | lambdaCodePath: 'lambda/create-predictor', 88 | timeout: Duration.seconds(30), 89 | } 90 | ); 91 | 92 | // Create Forecast 93 | const createForecast = new LambdaConstruct( 94 | this, 'ForecastMLOpsCreateForecast', 95 | { 96 | taskName: 'Forecast-MLOps-Create-Forecast', 97 | lambdaCodePath: 'lambda/create-forecast', 98 | timeout: Duration.seconds(30), 99 | environment: { 100 | FORECAST_ROLE: forecastRole.roleArn, 101 | RESOURCE_BUCKET: props.resourceBucket.bucketName, 102 | } 103 | } 104 | ); 105 | 106 | // Postprocessing 107 | const postprocess = new GlueConstruct(this, 'ForecastMLOpsPostprocess', { 108 | taskName: 'Forecast-MLOps-Postprocess', 109 | pythonFilePath: 'glue/postprocess.py', 110 | arguments: { 111 | '--uid': sfn.JsonPath.stringAt("$.uid"), 112 | '--bucket': props.resourceBucket.bucketName, 113 | } 114 | }); 115 | 116 | // Delete Forecast Resources 117 | const deleteForecastResource = new LambdaConstruct( 118 | this, 'ForecastMLOpsDeleteForecastResource', 119 | { 120 | taskName: 'Forecast-MLOps-Delete-Forecast-Resource', 121 | lambdaCodePath: 'lambda/delete-forecast-resource', 122 | timeout: Duration.seconds(30), 123 | environment: { 124 | FORECAST_ROLE: forecastRole.roleArn, 125 | } 126 | } 127 | ); 128 | 129 | // common 130 | const skipDeletion = new sfn.Pass(this, 'ForecastMLOpsPipelineSkipDeletion'); 131 | const deleteOrNot = new sfn.Choice(this, 'ForecastMLOpsPipelineStrategyChoice', { 132 | }).when( 133 | sfn.Condition.booleanEquals('$.config.PerformDelete', true), 134 | deleteForecastResource.task 135 | ).otherwise( 136 | skipDeletion 137 | ); 138 | 139 | /****** StateMachine - Begin ******/ 140 | // IAM Role for StateMachine 141 | const statesMachineExecutionRole = new iam.Role( 142 | this, 'ForecastMLOpsPipelineStateMachineExecutionRole', { 143 | assumedBy: new iam.ServicePrincipal('states.amazonaws.com'), 144 | roleName: 'ForecastMLOpsPipelineStateMachineExecutionRole', 145 | managedPolicies: [ 146 | {managedPolicyArn: 'arn:aws:iam::aws:policy/service-role/AWSLambdaRole'}, 147 | {managedPolicyArn: 'arn:aws:iam::aws:policy/service-role/AWSGlueServiceRole'}, 148 | ], 149 | } 150 | ); 151 | 152 | // StateMachine Definition 153 | const definition = preprocess.task 154 | .next(createDataset.task) 155 | .next(createDatasetGroup.task) 156 | .next(createDatasetImportJob.task) 157 | .next(createPredictor.task) 158 | .next(createForecast.task) 159 | .next(postprocess.task) 160 | .next(deleteOrNot); 161 | 162 | const stateMachine = new sfn.StateMachine( 163 | this, 'ForecastMLOpsPipelineStateMachine', { 164 | role: statesMachineExecutionRole, 165 | definition: definition, 166 | stateMachineName: 'Forecast-MLOps-Pipeline', 167 | } 168 | ); 169 | /****** StateMachine - End ******/ 170 | 171 | // Configure S3 Upload Trigger for StateMachine 172 | const trainTrigger = new TriggerConstruct( 173 | this, 174 | 'TrainTrigger', { 175 | resourceBucket: resourceBucket, 176 | stateMachine: stateMachine, 177 | s3Prefix: 'raw/', 178 | s3Suffix: '.zip', 179 | } 180 | ); 181 | 182 | } 183 | 184 | } 185 | -------------------------------------------------------------------------------- /lib/construct/trigger.ts: -------------------------------------------------------------------------------- 1 | import { Construct } from 'constructs'; 2 | import { 3 | aws_glue as glue, 4 | aws_iam as iam, 5 | aws_lambda as lambda, 6 | aws_s3 as s3, 7 | aws_s3_deployment as s3Deployment, 8 | aws_stepfunctions as sfn, 9 | aws_stepfunctions_tasks as tasks 10 | } from 'aws-cdk-lib'; 11 | import { S3EventSource } from 'aws-cdk-lib/aws-lambda-event-sources'; 12 | 13 | export interface TriggerConstructProps { 14 | stateMachine: sfn.StateMachine; 15 | resourceBucket: s3.Bucket; 16 | s3Prefix: string, 17 | s3Suffix: string, 18 | } 19 | 20 | export class TriggerConstruct extends Construct { 21 | public readonly role: iam.Role; 22 | public readonly lambda: lambda.Function; 23 | public readonly task: sfn.TaskStateBase; 24 | 25 | constructor(scope: Construct, id: string, props: TriggerConstructProps) { 26 | super(scope, id); 27 | 28 | // IAM Role 29 | this.role = new iam.Role(this, 'ForecastMLOpsPipelineTriggerRole', { 30 | assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'), 31 | roleName: `ForecastMLOpsPipelineTriggerRole`, 32 | managedPolicies: [ 33 | {managedPolicyArn: 'arn:aws:iam::aws:policy/AWSStepFunctionsFullAccess'}, 34 | {managedPolicyArn: 'arn:aws:iam::aws:policy/CloudWatchFullAccess'}, 35 | {managedPolicyArn: 'arn:aws:iam::aws:policy/AmazonS3ReadOnlyAccess'}, 36 | {managedPolicyArn: 'arn:aws:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'}, 37 | {managedPolicyArn: 'arn:aws:iam::aws:policy/service-role/AWSLambdaRole'}, 38 | ], 39 | }); 40 | 41 | // Lambda Function 42 | this.lambda = new lambda.Function( 43 | this, 'UploadTriggerLambda', 44 | { 45 | code: lambda.Code.fromAsset('lambda/trigger'), 46 | handler: 'index.handler', 47 | functionName: `Forecast-MLOps-Pipeline-Upload-Trigger`, 48 | runtime: lambda.Runtime.PYTHON_3_7, 49 | role: this.role, 50 | environment: { 51 | STEP_FUNCTIONS_ARN: props.stateMachine.stateMachineArn, 52 | }, 53 | }); 54 | 55 | // add S3Bucket as a Lambda's Event Source 56 | this.lambda.addEventSource(new S3EventSource( 57 | props.resourceBucket, { 58 | events: [ 59 | s3.EventType.OBJECT_CREATED, 60 | ], 61 | filters: [ 62 | { 63 | prefix: props.s3Prefix, 64 | suffix: props.s3Suffix 65 | }, 66 | ] 67 | } 68 | )); 69 | } 70 | 71 | 72 | } -------------------------------------------------------------------------------- /lib/forecast-mlops-pipeline-stack.ts: -------------------------------------------------------------------------------- 1 | import { Stack, StackProps, RemovalPolicy } from 'aws-cdk-lib'; 2 | import { Construct } from 'constructs'; 3 | import { 4 | aws_s3 as s3, 5 | } from 'aws-cdk-lib'; 6 | 7 | import { StateMachine } from './construct/state-machine'; 8 | 9 | export class ForecastMlopsPipelineStack extends Stack { 10 | constructor(scope: Construct, id: string, props?: StackProps) { 11 | super(scope, id, props); 12 | 13 | // S3 Bucket for pipeline resources 14 | const resourceBucket = new s3.Bucket(this, `ForecastMLOpsPipelineResourceBucket`, { 15 | bucketName: `forecast-mlops-pipeline-resource-bucket-${Stack.of(this).account}`, 16 | versioned: false, 17 | autoDeleteObjects: true, 18 | removalPolicy: RemovalPolicy.DESTROY 19 | }); 20 | 21 | const stateMachine = new StateMachine( 22 | this, 'ForecastMLOpsPiplineStateMachine', 23 | { 24 | resourceBucket: resourceBucket, 25 | } 26 | ); 27 | 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "forecast-mlops-pipeline", 3 | "version": "0.1.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@ampproject/remapping": { 8 | "version": "2.2.0", 9 | "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.2.0.tgz", 10 | "integrity": "sha512-qRmjj8nj9qmLTQXXmaR1cck3UXSRMPrbsLJAasZpF+t3riI71BXed5ebIOYwQntykeZuhjsdweEc9BxH5Jc26w==", 11 | "dev": true, 12 | "requires": { 13 | "@jridgewell/gen-mapping": "^0.1.0", 14 | "@jridgewell/trace-mapping": "^0.3.9" 15 | } 16 | }, 17 | "@aws-cdk/asset-awscli-v1": { 18 | "version": "2.2.196", 19 | "resolved": "https://registry.npmjs.org/@aws-cdk/asset-awscli-v1/-/asset-awscli-v1-2.2.196.tgz", 20 | "integrity": "sha512-F8hU1rEzYS7z5Dt2s+ttd0/jMvPuUE9BcXexgq+dIOLuZsRpDwNnkMBtjNaJXJS48ZGJ2X4b8VlklseepdtoSA==" 21 | }, 22 | "@aws-cdk/asset-kubectl-v20": { 23 | "version": "2.1.2", 24 | "resolved": "https://registry.npmjs.org/@aws-cdk/asset-kubectl-v20/-/asset-kubectl-v20-2.1.2.tgz", 25 | "integrity": "sha512-3M2tELJOxQv0apCIiuKQ4pAbncz9GuLwnKFqxifWfe77wuMxyTRPmxssYHs42ePqzap1LT6GDcPygGs+hHstLg==" 26 | }, 27 | "@aws-cdk/asset-node-proxy-agent-v5": { 28 | "version": "2.0.165", 29 | "resolved": "https://registry.npmjs.org/@aws-cdk/asset-node-proxy-agent-v5/-/asset-node-proxy-agent-v5-2.0.165.tgz", 30 | "integrity": "sha512-bsyLQD/vqXQcc9RDmlM1XqiFNO/yewgVFXmkMcQkndJbmE/jgYkzewwYGrBlfL725hGLQipXq19+jwWwdsXQqg==" 31 | }, 32 | "@aws-cdk/aws-glue-alpha": { 33 | "version": "2.28.0-alpha.0", 34 | "resolved": "https://registry.npmjs.org/@aws-cdk/aws-glue-alpha/-/aws-glue-alpha-2.28.0-alpha.0.tgz", 35 | "integrity": "sha512-pctVCdi3qsHBjb5n0iZsyzjAcZaPrDmzyotna+XW38n94vH28jJ+BSMkeIzUuaqlZNnYBsPQQZwVoxpxVmL6ww==" 36 | }, 37 | "@babel/code-frame": { 38 | "version": "7.16.7", 39 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.16.7.tgz", 40 | "integrity": "sha512-iAXqUn8IIeBTNd72xsFlgaXHkMBMt6y4HJp1tIaK465CWLT/fG1aqB7ykr95gHHmlBdGbFeWWfyB4NJJ0nmeIg==", 41 | "dev": true, 42 | "requires": { 43 | "@babel/highlight": "^7.16.7" 44 | } 45 | }, 46 | "@babel/compat-data": { 47 | "version": "7.17.10", 48 | "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.17.10.tgz", 49 | "integrity": "sha512-GZt/TCsG70Ms19gfZO1tM4CVnXsPgEPBCpJu+Qz3L0LUDsY5nZqFZglIoPC1kIYOtNBZlrnFT+klg12vFGZXrw==", 50 | "dev": true 51 | }, 52 | "@babel/core": { 53 | "version": "7.18.2", 54 | "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.18.2.tgz", 55 | "integrity": "sha512-A8pri1YJiC5UnkdrWcmfZTJTV85b4UXTAfImGmCfYmax4TR9Cw8sDS0MOk++Gp2mE/BefVJ5nwy5yzqNJbP/DQ==", 56 | "dev": true, 57 | "requires": { 58 | "@ampproject/remapping": "^2.1.0", 59 | "@babel/code-frame": "^7.16.7", 60 | "@babel/generator": "^7.18.2", 61 | "@babel/helper-compilation-targets": "^7.18.2", 62 | "@babel/helper-module-transforms": "^7.18.0", 63 | "@babel/helpers": "^7.18.2", 64 | "@babel/parser": "^7.18.0", 65 | "@babel/template": "^7.16.7", 66 | "@babel/traverse": "^7.18.2", 67 | "@babel/types": "^7.18.2", 68 | "convert-source-map": "^1.7.0", 69 | "debug": "^4.1.0", 70 | "gensync": "^1.0.0-beta.2", 71 | "json5": "^2.2.1", 72 | "semver": "^6.3.0" 73 | } 74 | }, 75 | "@babel/generator": { 76 | "version": "7.18.2", 77 | "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.18.2.tgz", 78 | "integrity": "sha512-W1lG5vUwFvfMd8HVXqdfbuG7RuaSrTCCD8cl8fP8wOivdbtbIg2Db3IWUcgvfxKbbn6ZBGYRW/Zk1MIwK49mgw==", 79 | "dev": true, 80 | "requires": { 81 | "@babel/types": "^7.18.2", 82 | "@jridgewell/gen-mapping": "^0.3.0", 83 | "jsesc": "^2.5.1" 84 | }, 85 | "dependencies": { 86 | "@jridgewell/gen-mapping": { 87 | "version": "0.3.1", 88 | "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.1.tgz", 89 | "integrity": "sha512-GcHwniMlA2z+WFPWuY8lp3fsza0I8xPFMWL5+n8LYyP6PSvPrXf4+n8stDHZY2DM0zy9sVkRDy1jDI4XGzYVqg==", 90 | "dev": true, 91 | "requires": { 92 | "@jridgewell/set-array": "^1.0.0", 93 | "@jridgewell/sourcemap-codec": "^1.4.10", 94 | "@jridgewell/trace-mapping": "^0.3.9" 95 | } 96 | } 97 | } 98 | }, 99 | "@babel/helper-compilation-targets": { 100 | "version": "7.18.2", 101 | "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.18.2.tgz", 102 | "integrity": "sha512-s1jnPotJS9uQnzFtiZVBUxe67CuBa679oWFHpxYYnTpRL/1ffhyX44R9uYiXoa/pLXcY9H2moJta0iaanlk/rQ==", 103 | "dev": true, 104 | "requires": { 105 | "@babel/compat-data": "^7.17.10", 106 | "@babel/helper-validator-option": "^7.16.7", 107 | "browserslist": "^4.20.2", 108 | "semver": "^6.3.0" 109 | } 110 | }, 111 | "@babel/helper-environment-visitor": { 112 | "version": "7.18.2", 113 | "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.18.2.tgz", 114 | "integrity": "sha512-14GQKWkX9oJzPiQQ7/J36FTXcD4kSp8egKjO9nINlSKiHITRA9q/R74qu8S9xlc/b/yjsJItQUeeh3xnGN0voQ==", 115 | "dev": true 116 | }, 117 | "@babel/helper-module-imports": { 118 | "version": "7.16.7", 119 | "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.16.7.tgz", 120 | "integrity": "sha512-LVtS6TqjJHFc+nYeITRo6VLXve70xmq7wPhWTqDJusJEgGmkAACWwMiTNrvfoQo6hEhFwAIixNkvB0jPXDL8Wg==", 121 | "dev": true, 122 | "requires": { 123 | "@babel/types": "^7.16.7" 124 | } 125 | }, 126 | "@babel/helper-module-transforms": { 127 | "version": "7.18.0", 128 | "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.18.0.tgz", 129 | "integrity": "sha512-kclUYSUBIjlvnzN2++K9f2qzYKFgjmnmjwL4zlmU5f8ZtzgWe8s0rUPSTGy2HmK4P8T52MQsS+HTQAgZd3dMEA==", 130 | "dev": true, 131 | "requires": { 132 | "@babel/helper-environment-visitor": "^7.16.7", 133 | "@babel/helper-module-imports": "^7.16.7", 134 | "@babel/helper-simple-access": "^7.17.7", 135 | "@babel/helper-split-export-declaration": "^7.16.7", 136 | "@babel/helper-validator-identifier": "^7.16.7", 137 | "@babel/template": "^7.16.7", 138 | "@babel/traverse": "^7.18.0", 139 | "@babel/types": "^7.18.0" 140 | } 141 | }, 142 | "@babel/helper-plugin-utils": { 143 | "version": "7.17.12", 144 | "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.17.12.tgz", 145 | "integrity": "sha512-JDkf04mqtN3y4iAbO1hv9U2ARpPyPL1zqyWs/2WG1pgSq9llHFjStX5jdxb84himgJm+8Ng+x0oiWF/nw/XQKA==", 146 | "dev": true 147 | }, 148 | "@babel/helper-simple-access": { 149 | "version": "7.18.2", 150 | "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.18.2.tgz", 151 | "integrity": "sha512-7LIrjYzndorDY88MycupkpQLKS1AFfsVRm2k/9PtKScSy5tZq0McZTj+DiMRynboZfIqOKvo03pmhTaUgiD6fQ==", 152 | "dev": true, 153 | "requires": { 154 | "@babel/types": "^7.18.2" 155 | } 156 | }, 157 | "@babel/helper-split-export-declaration": { 158 | "version": "7.16.7", 159 | "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.16.7.tgz", 160 | "integrity": "sha512-xbWoy/PFoxSWazIToT9Sif+jJTlrMcndIsaOKvTA6u7QEo7ilkRZpjew18/W3c7nm8fXdUDXh02VXTbZ0pGDNw==", 161 | "dev": true, 162 | "requires": { 163 | "@babel/types": "^7.16.7" 164 | } 165 | }, 166 | "@babel/helper-string-parser": { 167 | "version": "7.22.5", 168 | "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.22.5.tgz", 169 | "integrity": "sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw==", 170 | "dev": true 171 | }, 172 | "@babel/helper-validator-identifier": { 173 | "version": "7.16.7", 174 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.16.7.tgz", 175 | "integrity": "sha512-hsEnFemeiW4D08A5gUAZxLBTXpZ39P+a+DGDsHw1yxqyQ/jzFEnxf5uTEGp+3bzAbNOxU1paTgYS4ECU/IgfDw==", 176 | "dev": true 177 | }, 178 | "@babel/helper-validator-option": { 179 | "version": "7.16.7", 180 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.16.7.tgz", 181 | "integrity": "sha512-TRtenOuRUVo9oIQGPC5G9DgK4743cdxvtOw0weQNpZXaS16SCBi5MNjZF8vba3ETURjZpTbVn7Vvcf2eAwFozQ==", 182 | "dev": true 183 | }, 184 | "@babel/helpers": { 185 | "version": "7.18.2", 186 | "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.18.2.tgz", 187 | "integrity": "sha512-j+d+u5xT5utcQSzrh9p+PaJX94h++KN+ng9b9WEJq7pkUPAd61FGqhjuUEdfknb3E/uDBb7ruwEeKkIxNJPIrg==", 188 | "dev": true, 189 | "requires": { 190 | "@babel/template": "^7.16.7", 191 | "@babel/traverse": "^7.18.2", 192 | "@babel/types": "^7.18.2" 193 | } 194 | }, 195 | "@babel/highlight": { 196 | "version": "7.17.12", 197 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.17.12.tgz", 198 | "integrity": "sha512-7yykMVF3hfZY2jsHZEEgLc+3x4o1O+fYyULu11GynEUQNwB6lua+IIQn1FiJxNucd5UlyJryrwsOh8PL9Sn8Qg==", 199 | "dev": true, 200 | "requires": { 201 | "@babel/helper-validator-identifier": "^7.16.7", 202 | "chalk": "^2.0.0", 203 | "js-tokens": "^4.0.0" 204 | }, 205 | "dependencies": { 206 | "ansi-styles": { 207 | "version": "3.2.1", 208 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 209 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 210 | "dev": true, 211 | "requires": { 212 | "color-convert": "^1.9.0" 213 | } 214 | }, 215 | "chalk": { 216 | "version": "2.4.2", 217 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 218 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 219 | "dev": true, 220 | "requires": { 221 | "ansi-styles": "^3.2.1", 222 | "escape-string-regexp": "^1.0.5", 223 | "supports-color": "^5.3.0" 224 | } 225 | }, 226 | "color-convert": { 227 | "version": "1.9.3", 228 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 229 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 230 | "dev": true, 231 | "requires": { 232 | "color-name": "1.1.3" 233 | } 234 | }, 235 | "color-name": { 236 | "version": "1.1.3", 237 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 238 | "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", 239 | "dev": true 240 | }, 241 | "has-flag": { 242 | "version": "3.0.0", 243 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 244 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 245 | "dev": true 246 | }, 247 | "supports-color": { 248 | "version": "5.5.0", 249 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 250 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 251 | "dev": true, 252 | "requires": { 253 | "has-flag": "^3.0.0" 254 | } 255 | } 256 | } 257 | }, 258 | "@babel/parser": { 259 | "version": "7.18.4", 260 | "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.18.4.tgz", 261 | "integrity": "sha512-FDge0dFazETFcxGw/EXzOkN8uJp0PC7Qbm+Pe9T+av2zlBpOgunFHkQPPn+eRuClU73JF+98D531UgayY89tow==", 262 | "dev": true 263 | }, 264 | "@babel/plugin-syntax-async-generators": { 265 | "version": "7.8.4", 266 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", 267 | "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", 268 | "dev": true, 269 | "requires": { 270 | "@babel/helper-plugin-utils": "^7.8.0" 271 | } 272 | }, 273 | "@babel/plugin-syntax-bigint": { 274 | "version": "7.8.3", 275 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", 276 | "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", 277 | "dev": true, 278 | "requires": { 279 | "@babel/helper-plugin-utils": "^7.8.0" 280 | } 281 | }, 282 | "@babel/plugin-syntax-class-properties": { 283 | "version": "7.12.13", 284 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", 285 | "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", 286 | "dev": true, 287 | "requires": { 288 | "@babel/helper-plugin-utils": "^7.12.13" 289 | } 290 | }, 291 | "@babel/plugin-syntax-import-meta": { 292 | "version": "7.10.4", 293 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", 294 | "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", 295 | "dev": true, 296 | "requires": { 297 | "@babel/helper-plugin-utils": "^7.10.4" 298 | } 299 | }, 300 | "@babel/plugin-syntax-json-strings": { 301 | "version": "7.8.3", 302 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", 303 | "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", 304 | "dev": true, 305 | "requires": { 306 | "@babel/helper-plugin-utils": "^7.8.0" 307 | } 308 | }, 309 | "@babel/plugin-syntax-logical-assignment-operators": { 310 | "version": "7.10.4", 311 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", 312 | "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", 313 | "dev": true, 314 | "requires": { 315 | "@babel/helper-plugin-utils": "^7.10.4" 316 | } 317 | }, 318 | "@babel/plugin-syntax-nullish-coalescing-operator": { 319 | "version": "7.8.3", 320 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", 321 | "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", 322 | "dev": true, 323 | "requires": { 324 | "@babel/helper-plugin-utils": "^7.8.0" 325 | } 326 | }, 327 | "@babel/plugin-syntax-numeric-separator": { 328 | "version": "7.10.4", 329 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", 330 | "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", 331 | "dev": true, 332 | "requires": { 333 | "@babel/helper-plugin-utils": "^7.10.4" 334 | } 335 | }, 336 | "@babel/plugin-syntax-object-rest-spread": { 337 | "version": "7.8.3", 338 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", 339 | "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", 340 | "dev": true, 341 | "requires": { 342 | "@babel/helper-plugin-utils": "^7.8.0" 343 | } 344 | }, 345 | "@babel/plugin-syntax-optional-catch-binding": { 346 | "version": "7.8.3", 347 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", 348 | "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", 349 | "dev": true, 350 | "requires": { 351 | "@babel/helper-plugin-utils": "^7.8.0" 352 | } 353 | }, 354 | "@babel/plugin-syntax-optional-chaining": { 355 | "version": "7.8.3", 356 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", 357 | "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", 358 | "dev": true, 359 | "requires": { 360 | "@babel/helper-plugin-utils": "^7.8.0" 361 | } 362 | }, 363 | "@babel/plugin-syntax-top-level-await": { 364 | "version": "7.14.5", 365 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", 366 | "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", 367 | "dev": true, 368 | "requires": { 369 | "@babel/helper-plugin-utils": "^7.14.5" 370 | } 371 | }, 372 | "@babel/plugin-syntax-typescript": { 373 | "version": "7.17.12", 374 | "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.17.12.tgz", 375 | "integrity": "sha512-TYY0SXFiO31YXtNg3HtFwNJHjLsAyIIhAhNWkQ5whPPS7HWUFlg9z0Ta4qAQNjQbP1wsSt/oKkmZ/4/WWdMUpw==", 376 | "dev": true, 377 | "requires": { 378 | "@babel/helper-plugin-utils": "^7.17.12" 379 | } 380 | }, 381 | "@babel/template": { 382 | "version": "7.16.7", 383 | "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.16.7.tgz", 384 | "integrity": "sha512-I8j/x8kHUrbYRTUxXrrMbfCa7jxkE7tZre39x3kjr9hvI82cK1FfqLygotcWN5kdPGWcLdWMHpSBavse5tWw3w==", 385 | "dev": true, 386 | "requires": { 387 | "@babel/code-frame": "^7.16.7", 388 | "@babel/parser": "^7.16.7", 389 | "@babel/types": "^7.16.7" 390 | } 391 | }, 392 | "@babel/traverse": { 393 | "version": "7.23.2", 394 | "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.23.2.tgz", 395 | "integrity": "sha512-azpe59SQ48qG6nu2CzcMLbxUudtN+dOM9kDbUqGq3HXUJRlo7i8fvPoxQUzYgLZ4cMVmuZgm8vvBpNeRhd6XSw==", 396 | "dev": true, 397 | "requires": { 398 | "@babel/code-frame": "^7.22.13", 399 | "@babel/generator": "^7.23.0", 400 | "@babel/helper-environment-visitor": "^7.22.20", 401 | "@babel/helper-function-name": "^7.23.0", 402 | "@babel/helper-hoist-variables": "^7.22.5", 403 | "@babel/helper-split-export-declaration": "^7.22.6", 404 | "@babel/parser": "^7.23.0", 405 | "@babel/types": "^7.23.0", 406 | "debug": "^4.1.0", 407 | "globals": "^11.1.0" 408 | }, 409 | "dependencies": { 410 | "@babel/code-frame": { 411 | "version": "7.22.13", 412 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.22.13.tgz", 413 | "integrity": "sha512-XktuhWlJ5g+3TJXc5upd9Ks1HutSArik6jf2eAjYFyIOf4ej3RN+184cZbzDvbPnuTJIUhPKKJE3cIsYTiAT3w==", 414 | "dev": true, 415 | "requires": { 416 | "@babel/highlight": "^7.22.13", 417 | "chalk": "^2.4.2" 418 | } 419 | }, 420 | "@babel/generator": { 421 | "version": "7.23.0", 422 | "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.23.0.tgz", 423 | "integrity": "sha512-lN85QRR+5IbYrMWM6Y4pE/noaQtg4pNiqeNGX60eqOfo6gtEj6uw/JagelB8vVztSd7R6M5n1+PQkDbHbBRU4g==", 424 | "dev": true, 425 | "requires": { 426 | "@babel/types": "^7.23.0", 427 | "@jridgewell/gen-mapping": "^0.3.2", 428 | "@jridgewell/trace-mapping": "^0.3.17", 429 | "jsesc": "^2.5.1" 430 | } 431 | }, 432 | "@babel/helper-environment-visitor": { 433 | "version": "7.22.20", 434 | "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.20.tgz", 435 | "integrity": "sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==", 436 | "dev": true 437 | }, 438 | "@babel/helper-function-name": { 439 | "version": "7.23.0", 440 | "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.23.0.tgz", 441 | "integrity": "sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==", 442 | "dev": true, 443 | "requires": { 444 | "@babel/template": "^7.22.15", 445 | "@babel/types": "^7.23.0" 446 | } 447 | }, 448 | "@babel/helper-hoist-variables": { 449 | "version": "7.22.5", 450 | "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz", 451 | "integrity": "sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==", 452 | "dev": true, 453 | "requires": { 454 | "@babel/types": "^7.22.5" 455 | } 456 | }, 457 | "@babel/helper-split-export-declaration": { 458 | "version": "7.22.6", 459 | "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.6.tgz", 460 | "integrity": "sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==", 461 | "dev": true, 462 | "requires": { 463 | "@babel/types": "^7.22.5" 464 | } 465 | }, 466 | "@babel/helper-validator-identifier": { 467 | "version": "7.22.20", 468 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.20.tgz", 469 | "integrity": "sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==", 470 | "dev": true 471 | }, 472 | "@babel/highlight": { 473 | "version": "7.22.20", 474 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.22.20.tgz", 475 | "integrity": "sha512-dkdMCN3py0+ksCgYmGG8jKeGA/8Tk+gJwSYYlFGxG5lmhfKNoAy004YpLxpS1W2J8m/EK2Ew+yOs9pVRwO89mg==", 476 | "dev": true, 477 | "requires": { 478 | "@babel/helper-validator-identifier": "^7.22.20", 479 | "chalk": "^2.4.2", 480 | "js-tokens": "^4.0.0" 481 | } 482 | }, 483 | "@babel/parser": { 484 | "version": "7.23.0", 485 | "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.23.0.tgz", 486 | "integrity": "sha512-vvPKKdMemU85V9WE/l5wZEmImpCtLqbnTvqDS2U1fJ96KrxoW7KrXhNsNCblQlg8Ck4b85yxdTyelsMUgFUXiw==", 487 | "dev": true 488 | }, 489 | "@babel/template": { 490 | "version": "7.22.15", 491 | "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.22.15.tgz", 492 | "integrity": "sha512-QPErUVm4uyJa60rkI73qneDacvdvzxshT3kksGqlGWYdOTIUOwJ7RDUL8sGqslY1uXWSL6xMFKEXDS3ox2uF0w==", 493 | "dev": true, 494 | "requires": { 495 | "@babel/code-frame": "^7.22.13", 496 | "@babel/parser": "^7.22.15", 497 | "@babel/types": "^7.22.15" 498 | } 499 | }, 500 | "@babel/types": { 501 | "version": "7.23.0", 502 | "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.23.0.tgz", 503 | "integrity": "sha512-0oIyUfKoI3mSqMvsxBdclDwxXKXAUA8v/apZbc+iSyARYou1o8ZGDxbUYyLFoW2arqS2jDGqJuZvv1d/io1axg==", 504 | "dev": true, 505 | "requires": { 506 | "@babel/helper-string-parser": "^7.22.5", 507 | "@babel/helper-validator-identifier": "^7.22.20", 508 | "to-fast-properties": "^2.0.0" 509 | } 510 | }, 511 | "@jridgewell/gen-mapping": { 512 | "version": "0.3.3", 513 | "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz", 514 | "integrity": "sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ==", 515 | "dev": true, 516 | "requires": { 517 | "@jridgewell/set-array": "^1.0.1", 518 | "@jridgewell/sourcemap-codec": "^1.4.10", 519 | "@jridgewell/trace-mapping": "^0.3.9" 520 | } 521 | }, 522 | "@jridgewell/resolve-uri": { 523 | "version": "3.1.1", 524 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz", 525 | "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==", 526 | "dev": true 527 | }, 528 | "@jridgewell/trace-mapping": { 529 | "version": "0.3.20", 530 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.20.tgz", 531 | "integrity": "sha512-R8LcPeWZol2zR8mmH3JeKQ6QRCFb7XgUhV9ZlGhHLGyg4wpPiPZNQOOWhFZhxKw8u//yTbNGI42Bx/3paXEQ+Q==", 532 | "dev": true, 533 | "requires": { 534 | "@jridgewell/resolve-uri": "^3.1.0", 535 | "@jridgewell/sourcemap-codec": "^1.4.14" 536 | }, 537 | "dependencies": { 538 | "@jridgewell/sourcemap-codec": { 539 | "version": "1.4.15", 540 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", 541 | "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", 542 | "dev": true 543 | } 544 | } 545 | }, 546 | "ansi-styles": { 547 | "version": "3.2.1", 548 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 549 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 550 | "dev": true, 551 | "requires": { 552 | "color-convert": "^1.9.0" 553 | } 554 | }, 555 | "chalk": { 556 | "version": "2.4.2", 557 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 558 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 559 | "dev": true, 560 | "requires": { 561 | "ansi-styles": "^3.2.1", 562 | "escape-string-regexp": "^1.0.5", 563 | "supports-color": "^5.3.0" 564 | } 565 | }, 566 | "color-convert": { 567 | "version": "1.9.3", 568 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 569 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 570 | "dev": true, 571 | "requires": { 572 | "color-name": "1.1.3" 573 | } 574 | }, 575 | "color-name": { 576 | "version": "1.1.3", 577 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 578 | "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", 579 | "dev": true 580 | }, 581 | "has-flag": { 582 | "version": "3.0.0", 583 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 584 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 585 | "dev": true 586 | }, 587 | "supports-color": { 588 | "version": "5.5.0", 589 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 590 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 591 | "dev": true, 592 | "requires": { 593 | "has-flag": "^3.0.0" 594 | } 595 | } 596 | } 597 | }, 598 | "@babel/types": { 599 | "version": "7.18.4", 600 | "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.18.4.tgz", 601 | "integrity": "sha512-ThN1mBcMq5pG/Vm2IcBmPPfyPXbd8S02rS+OBIDENdufvqC7Z/jHPCv9IcP01277aKtDI8g/2XysBN4hA8niiw==", 602 | "dev": true, 603 | "requires": { 604 | "@babel/helper-validator-identifier": "^7.16.7", 605 | "to-fast-properties": "^2.0.0" 606 | } 607 | }, 608 | "@bcoe/v8-coverage": { 609 | "version": "0.2.3", 610 | "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", 611 | "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", 612 | "dev": true 613 | }, 614 | "@cspotcode/source-map-support": { 615 | "version": "0.8.1", 616 | "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz", 617 | "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", 618 | "dev": true, 619 | "requires": { 620 | "@jridgewell/trace-mapping": "0.3.9" 621 | }, 622 | "dependencies": { 623 | "@jridgewell/trace-mapping": { 624 | "version": "0.3.9", 625 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", 626 | "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", 627 | "dev": true, 628 | "requires": { 629 | "@jridgewell/resolve-uri": "^3.0.3", 630 | "@jridgewell/sourcemap-codec": "^1.4.10" 631 | } 632 | } 633 | } 634 | }, 635 | "@istanbuljs/load-nyc-config": { 636 | "version": "1.1.0", 637 | "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", 638 | "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", 639 | "dev": true, 640 | "requires": { 641 | "camelcase": "^5.3.1", 642 | "find-up": "^4.1.0", 643 | "get-package-type": "^0.1.0", 644 | "js-yaml": "^3.13.1", 645 | "resolve-from": "^5.0.0" 646 | } 647 | }, 648 | "@istanbuljs/schema": { 649 | "version": "0.1.3", 650 | "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", 651 | "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", 652 | "dev": true 653 | }, 654 | "@jest/console": { 655 | "version": "27.5.1", 656 | "resolved": "https://registry.npmjs.org/@jest/console/-/console-27.5.1.tgz", 657 | "integrity": "sha512-kZ/tNpS3NXn0mlXXXPNuDZnb4c0oZ20r4K5eemM2k30ZC3G0T02nXUvyhf5YdbXWHPEJLc9qGLxEZ216MdL+Zg==", 658 | "dev": true, 659 | "requires": { 660 | "@jest/types": "^27.5.1", 661 | "@types/node": "*", 662 | "chalk": "^4.0.0", 663 | "jest-message-util": "^27.5.1", 664 | "jest-util": "^27.5.1", 665 | "slash": "^3.0.0" 666 | } 667 | }, 668 | "@jest/core": { 669 | "version": "27.5.1", 670 | "resolved": "https://registry.npmjs.org/@jest/core/-/core-27.5.1.tgz", 671 | "integrity": "sha512-AK6/UTrvQD0Cd24NSqmIA6rKsu0tKIxfiCducZvqxYdmMisOYAsdItspT+fQDQYARPf8XgjAFZi0ogW2agH5nQ==", 672 | "dev": true, 673 | "requires": { 674 | "@jest/console": "^27.5.1", 675 | "@jest/reporters": "^27.5.1", 676 | "@jest/test-result": "^27.5.1", 677 | "@jest/transform": "^27.5.1", 678 | "@jest/types": "^27.5.1", 679 | "@types/node": "*", 680 | "ansi-escapes": "^4.2.1", 681 | "chalk": "^4.0.0", 682 | "emittery": "^0.8.1", 683 | "exit": "^0.1.2", 684 | "graceful-fs": "^4.2.9", 685 | "jest-changed-files": "^27.5.1", 686 | "jest-config": "^27.5.1", 687 | "jest-haste-map": "^27.5.1", 688 | "jest-message-util": "^27.5.1", 689 | "jest-regex-util": "^27.5.1", 690 | "jest-resolve": "^27.5.1", 691 | "jest-resolve-dependencies": "^27.5.1", 692 | "jest-runner": "^27.5.1", 693 | "jest-runtime": "^27.5.1", 694 | "jest-snapshot": "^27.5.1", 695 | "jest-util": "^27.5.1", 696 | "jest-validate": "^27.5.1", 697 | "jest-watcher": "^27.5.1", 698 | "micromatch": "^4.0.4", 699 | "rimraf": "^3.0.0", 700 | "slash": "^3.0.0", 701 | "strip-ansi": "^6.0.0" 702 | } 703 | }, 704 | "@jest/environment": { 705 | "version": "27.5.1", 706 | "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-27.5.1.tgz", 707 | "integrity": "sha512-/WQjhPJe3/ghaol/4Bq480JKXV/Rfw8nQdN7f41fM8VDHLcxKXou6QyXAh3EFr9/bVG3x74z1NWDkP87EiY8gA==", 708 | "dev": true, 709 | "requires": { 710 | "@jest/fake-timers": "^27.5.1", 711 | "@jest/types": "^27.5.1", 712 | "@types/node": "*", 713 | "jest-mock": "^27.5.1" 714 | } 715 | }, 716 | "@jest/fake-timers": { 717 | "version": "27.5.1", 718 | "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-27.5.1.tgz", 719 | "integrity": "sha512-/aPowoolwa07k7/oM3aASneNeBGCmGQsc3ugN4u6s4C/+s5M64MFo/+djTdiwcbQlRfFElGuDXWzaWj6QgKObQ==", 720 | "dev": true, 721 | "requires": { 722 | "@jest/types": "^27.5.1", 723 | "@sinonjs/fake-timers": "^8.0.1", 724 | "@types/node": "*", 725 | "jest-message-util": "^27.5.1", 726 | "jest-mock": "^27.5.1", 727 | "jest-util": "^27.5.1" 728 | } 729 | }, 730 | "@jest/globals": { 731 | "version": "27.5.1", 732 | "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-27.5.1.tgz", 733 | "integrity": "sha512-ZEJNB41OBQQgGzgyInAv0UUfDDj3upmHydjieSxFvTRuZElrx7tXg/uVQ5hYVEwiXs3+aMsAeEc9X7xiSKCm4Q==", 734 | "dev": true, 735 | "requires": { 736 | "@jest/environment": "^27.5.1", 737 | "@jest/types": "^27.5.1", 738 | "expect": "^27.5.1" 739 | } 740 | }, 741 | "@jest/reporters": { 742 | "version": "27.5.1", 743 | "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-27.5.1.tgz", 744 | "integrity": "sha512-cPXh9hWIlVJMQkVk84aIvXuBB4uQQmFqZiacloFuGiP3ah1sbCxCosidXFDfqG8+6fO1oR2dTJTlsOy4VFmUfw==", 745 | "dev": true, 746 | "requires": { 747 | "@bcoe/v8-coverage": "^0.2.3", 748 | "@jest/console": "^27.5.1", 749 | "@jest/test-result": "^27.5.1", 750 | "@jest/transform": "^27.5.1", 751 | "@jest/types": "^27.5.1", 752 | "@types/node": "*", 753 | "chalk": "^4.0.0", 754 | "collect-v8-coverage": "^1.0.0", 755 | "exit": "^0.1.2", 756 | "glob": "^7.1.2", 757 | "graceful-fs": "^4.2.9", 758 | "istanbul-lib-coverage": "^3.0.0", 759 | "istanbul-lib-instrument": "^5.1.0", 760 | "istanbul-lib-report": "^3.0.0", 761 | "istanbul-lib-source-maps": "^4.0.0", 762 | "istanbul-reports": "^3.1.3", 763 | "jest-haste-map": "^27.5.1", 764 | "jest-resolve": "^27.5.1", 765 | "jest-util": "^27.5.1", 766 | "jest-worker": "^27.5.1", 767 | "slash": "^3.0.0", 768 | "source-map": "^0.6.0", 769 | "string-length": "^4.0.1", 770 | "terminal-link": "^2.0.0", 771 | "v8-to-istanbul": "^8.1.0" 772 | } 773 | }, 774 | "@jest/source-map": { 775 | "version": "27.5.1", 776 | "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-27.5.1.tgz", 777 | "integrity": "sha512-y9NIHUYF3PJRlHk98NdC/N1gl88BL08aQQgu4k4ZopQkCw9t9cV8mtl3TV8b/YCB8XaVTFrmUTAJvjsntDireg==", 778 | "dev": true, 779 | "requires": { 780 | "callsites": "^3.0.0", 781 | "graceful-fs": "^4.2.9", 782 | "source-map": "^0.6.0" 783 | } 784 | }, 785 | "@jest/test-result": { 786 | "version": "27.5.1", 787 | "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-27.5.1.tgz", 788 | "integrity": "sha512-EW35l2RYFUcUQxFJz5Cv5MTOxlJIQs4I7gxzi2zVU7PJhOwfYq1MdC5nhSmYjX1gmMmLPvB3sIaC+BkcHRBfag==", 789 | "dev": true, 790 | "requires": { 791 | "@jest/console": "^27.5.1", 792 | "@jest/types": "^27.5.1", 793 | "@types/istanbul-lib-coverage": "^2.0.0", 794 | "collect-v8-coverage": "^1.0.0" 795 | } 796 | }, 797 | "@jest/test-sequencer": { 798 | "version": "27.5.1", 799 | "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-27.5.1.tgz", 800 | "integrity": "sha512-LCheJF7WB2+9JuCS7VB/EmGIdQuhtqjRNI9A43idHv3E4KltCTsPsLxvdaubFHSYwY/fNjMWjl6vNRhDiN7vpQ==", 801 | "dev": true, 802 | "requires": { 803 | "@jest/test-result": "^27.5.1", 804 | "graceful-fs": "^4.2.9", 805 | "jest-haste-map": "^27.5.1", 806 | "jest-runtime": "^27.5.1" 807 | } 808 | }, 809 | "@jest/transform": { 810 | "version": "27.5.1", 811 | "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-27.5.1.tgz", 812 | "integrity": "sha512-ipON6WtYgl/1329g5AIJVbUuEh0wZVbdpGwC99Jw4LwuoBNS95MVphU6zOeD9pDkon+LLbFL7lOQRapbB8SCHw==", 813 | "dev": true, 814 | "requires": { 815 | "@babel/core": "^7.1.0", 816 | "@jest/types": "^27.5.1", 817 | "babel-plugin-istanbul": "^6.1.1", 818 | "chalk": "^4.0.0", 819 | "convert-source-map": "^1.4.0", 820 | "fast-json-stable-stringify": "^2.0.0", 821 | "graceful-fs": "^4.2.9", 822 | "jest-haste-map": "^27.5.1", 823 | "jest-regex-util": "^27.5.1", 824 | "jest-util": "^27.5.1", 825 | "micromatch": "^4.0.4", 826 | "pirates": "^4.0.4", 827 | "slash": "^3.0.0", 828 | "source-map": "^0.6.1", 829 | "write-file-atomic": "^3.0.0" 830 | } 831 | }, 832 | "@jest/types": { 833 | "version": "27.5.1", 834 | "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.5.1.tgz", 835 | "integrity": "sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw==", 836 | "dev": true, 837 | "requires": { 838 | "@types/istanbul-lib-coverage": "^2.0.0", 839 | "@types/istanbul-reports": "^3.0.0", 840 | "@types/node": "*", 841 | "@types/yargs": "^16.0.0", 842 | "chalk": "^4.0.0" 843 | } 844 | }, 845 | "@jridgewell/gen-mapping": { 846 | "version": "0.1.1", 847 | "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.1.1.tgz", 848 | "integrity": "sha512-sQXCasFk+U8lWYEe66WxRDOE9PjVz4vSM51fTu3Hw+ClTpUSQb718772vH3pyS5pShp6lvQM7SxgIDXXXmOX7w==", 849 | "dev": true, 850 | "requires": { 851 | "@jridgewell/set-array": "^1.0.0", 852 | "@jridgewell/sourcemap-codec": "^1.4.10" 853 | } 854 | }, 855 | "@jridgewell/resolve-uri": { 856 | "version": "3.0.7", 857 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.0.7.tgz", 858 | "integrity": "sha512-8cXDaBBHOr2pQ7j77Y6Vp5VDT2sIqWyWQ56TjEq4ih/a4iST3dItRe8Q9fp0rrIl9DoKhWQtUQz/YpOxLkXbNA==", 859 | "dev": true 860 | }, 861 | "@jridgewell/set-array": { 862 | "version": "1.1.1", 863 | "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.1.tgz", 864 | "integrity": "sha512-Ct5MqZkLGEXTVmQYbGtx9SVqD2fqwvdubdps5D3djjAkgkKwT918VNOz65pEHFaYTeWcukmJmH5SwsA9Tn2ObQ==", 865 | "dev": true 866 | }, 867 | "@jridgewell/sourcemap-codec": { 868 | "version": "1.4.13", 869 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.13.tgz", 870 | "integrity": "sha512-GryiOJmNcWbovBxTfZSF71V/mXbgcV3MewDe3kIMCLyIh5e7SKAeUZs+rMnJ8jkMolZ/4/VsdBmMrw3l+VdZ3w==", 871 | "dev": true 872 | }, 873 | "@jridgewell/trace-mapping": { 874 | "version": "0.3.13", 875 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.13.tgz", 876 | "integrity": "sha512-o1xbKhp9qnIAoHJSWd6KlCZfqslL4valSF81H8ImioOAxluWYWOpWkpyktY2vnt4tbrX9XYaxovq6cgowaJp2w==", 877 | "dev": true, 878 | "requires": { 879 | "@jridgewell/resolve-uri": "^3.0.3", 880 | "@jridgewell/sourcemap-codec": "^1.4.10" 881 | } 882 | }, 883 | "@sinonjs/commons": { 884 | "version": "1.8.3", 885 | "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.3.tgz", 886 | "integrity": "sha512-xkNcLAn/wZaX14RPlwizcKicDk9G3F8m2nU3L7Ukm5zBgTwiT0wsoFAHx9Jq56fJA1z/7uKGtCRu16sOUCLIHQ==", 887 | "dev": true, 888 | "requires": { 889 | "type-detect": "4.0.8" 890 | } 891 | }, 892 | "@sinonjs/fake-timers": { 893 | "version": "8.1.0", 894 | "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-8.1.0.tgz", 895 | "integrity": "sha512-OAPJUAtgeINhh/TAlUID4QTs53Njm7xzddaVlEs/SXwgtiD1tW22zAB/W1wdqfrpmikgaWQ9Fw6Ws+hsiRm5Vg==", 896 | "dev": true, 897 | "requires": { 898 | "@sinonjs/commons": "^1.7.0" 899 | } 900 | }, 901 | "@tootallnate/once": { 902 | "version": "1.1.2", 903 | "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", 904 | "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", 905 | "dev": true 906 | }, 907 | "@tsconfig/node10": { 908 | "version": "1.0.8", 909 | "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.8.tgz", 910 | "integrity": "sha512-6XFfSQmMgq0CFLY1MslA/CPUfhIL919M1rMsa5lP2P097N2Wd1sSX0tx1u4olM16fLNhtHZpRhedZJphNJqmZg==", 911 | "dev": true 912 | }, 913 | "@tsconfig/node12": { 914 | "version": "1.0.9", 915 | "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.9.tgz", 916 | "integrity": "sha512-/yBMcem+fbvhSREH+s14YJi18sp7J9jpuhYByADT2rypfajMZZN4WQ6zBGgBKp53NKmqI36wFYDb3yaMPurITw==", 917 | "dev": true 918 | }, 919 | "@tsconfig/node14": { 920 | "version": "1.0.1", 921 | "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.1.tgz", 922 | "integrity": "sha512-509r2+yARFfHHE7T6Puu2jjkoycftovhXRqW328PDXTVGKihlb1P8Z9mMZH04ebyajfRY7dedfGynlrFHJUQCg==", 923 | "dev": true 924 | }, 925 | "@tsconfig/node16": { 926 | "version": "1.0.2", 927 | "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.2.tgz", 928 | "integrity": "sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA==", 929 | "dev": true 930 | }, 931 | "@types/babel__core": { 932 | "version": "7.1.19", 933 | "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.19.tgz", 934 | "integrity": "sha512-WEOTgRsbYkvA/KCsDwVEGkd7WAr1e3g31VHQ8zy5gul/V1qKullU/BU5I68X5v7V3GnB9eotmom4v5a5gjxorw==", 935 | "dev": true, 936 | "requires": { 937 | "@babel/parser": "^7.1.0", 938 | "@babel/types": "^7.0.0", 939 | "@types/babel__generator": "*", 940 | "@types/babel__template": "*", 941 | "@types/babel__traverse": "*" 942 | } 943 | }, 944 | "@types/babel__generator": { 945 | "version": "7.6.4", 946 | "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.4.tgz", 947 | "integrity": "sha512-tFkciB9j2K755yrTALxD44McOrk+gfpIpvC3sxHjRawj6PfnQxrse4Clq5y/Rq+G3mrBurMax/lG8Qn2t9mSsg==", 948 | "dev": true, 949 | "requires": { 950 | "@babel/types": "^7.0.0" 951 | } 952 | }, 953 | "@types/babel__template": { 954 | "version": "7.4.1", 955 | "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.1.tgz", 956 | "integrity": "sha512-azBFKemX6kMg5Io+/rdGT0dkGreboUVR0Cdm3fz9QJWpaQGJRQXl7C+6hOTCZcMll7KFyEQpgbYI2lHdsS4U7g==", 957 | "dev": true, 958 | "requires": { 959 | "@babel/parser": "^7.1.0", 960 | "@babel/types": "^7.0.0" 961 | } 962 | }, 963 | "@types/babel__traverse": { 964 | "version": "7.17.1", 965 | "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.17.1.tgz", 966 | "integrity": "sha512-kVzjari1s2YVi77D3w1yuvohV2idweYXMCDzqBiVNN63TcDWrIlTVOYpqVrvbbyOE/IyzBoTKF0fdnLPEORFxA==", 967 | "dev": true, 968 | "requires": { 969 | "@babel/types": "^7.3.0" 970 | } 971 | }, 972 | "@types/graceful-fs": { 973 | "version": "4.1.5", 974 | "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.5.tgz", 975 | "integrity": "sha512-anKkLmZZ+xm4p8JWBf4hElkM4XR+EZeA2M9BAkkTldmcyDY4mbdIJnRghDJH3Ov5ooY7/UAoENtmdMSkaAd7Cw==", 976 | "dev": true, 977 | "requires": { 978 | "@types/node": "*" 979 | } 980 | }, 981 | "@types/istanbul-lib-coverage": { 982 | "version": "2.0.4", 983 | "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", 984 | "integrity": "sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==", 985 | "dev": true 986 | }, 987 | "@types/istanbul-lib-report": { 988 | "version": "3.0.0", 989 | "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", 990 | "integrity": "sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==", 991 | "dev": true, 992 | "requires": { 993 | "@types/istanbul-lib-coverage": "*" 994 | } 995 | }, 996 | "@types/istanbul-reports": { 997 | "version": "3.0.1", 998 | "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.1.tgz", 999 | "integrity": "sha512-c3mAZEuK0lvBp8tmuL74XRKn1+y2dcwOUpH7x4WrF6gk1GIgiluDRgMYQtw2OFcBvAJWlt6ASU3tSqxp0Uu0Aw==", 1000 | "dev": true, 1001 | "requires": { 1002 | "@types/istanbul-lib-report": "*" 1003 | } 1004 | }, 1005 | "@types/jest": { 1006 | "version": "27.5.2", 1007 | "resolved": "https://registry.npmjs.org/@types/jest/-/jest-27.5.2.tgz", 1008 | "integrity": "sha512-mpT8LJJ4CMeeahobofYWIjFo0xonRS/HfxnVEPMPFSQdGUt1uHCnoPT7Zhb+sjDU2wz0oKV0OLUR0WzrHNgfeA==", 1009 | "dev": true, 1010 | "requires": { 1011 | "jest-matcher-utils": "^27.0.0", 1012 | "pretty-format": "^27.0.0" 1013 | } 1014 | }, 1015 | "@types/node": { 1016 | "version": "10.17.27", 1017 | "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.27.tgz", 1018 | "integrity": "sha512-J0oqm9ZfAXaPdwNXMMgAhylw5fhmXkToJd06vuDUSAgEDZ/n/69/69UmyBZbc+zT34UnShuDSBqvim3SPnozJg==", 1019 | "dev": true 1020 | }, 1021 | "@types/prettier": { 1022 | "version": "2.6.0", 1023 | "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.6.0.tgz", 1024 | "integrity": "sha512-G/AdOadiZhnJp0jXCaBQU449W2h716OW/EoXeYkCytxKL06X1WCXB4DZpp8TpZ8eyIJVS1cw4lrlkkSYU21cDw==", 1025 | "dev": true 1026 | }, 1027 | "@types/stack-utils": { 1028 | "version": "2.0.1", 1029 | "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz", 1030 | "integrity": "sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw==", 1031 | "dev": true 1032 | }, 1033 | "@types/yargs": { 1034 | "version": "16.0.4", 1035 | "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-16.0.4.tgz", 1036 | "integrity": "sha512-T8Yc9wt/5LbJyCaLiHPReJa0kApcIgJ7Bn735GjItUfh08Z1pJvu8QZqb9s+mMvKV6WUQRV7K2R46YbjMXTTJw==", 1037 | "dev": true, 1038 | "requires": { 1039 | "@types/yargs-parser": "*" 1040 | } 1041 | }, 1042 | "@types/yargs-parser": { 1043 | "version": "21.0.0", 1044 | "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.0.tgz", 1045 | "integrity": "sha512-iO9ZQHkZxHn4mSakYV0vFHAVDyEOIJQrV2uZ06HxEPcx+mt8swXoZHIbaaJ2crJYFfErySgktuTZ3BeLz+XmFA==", 1046 | "dev": true 1047 | }, 1048 | "abab": { 1049 | "version": "2.0.6", 1050 | "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz", 1051 | "integrity": "sha512-j2afSsaIENvHZN2B8GOpF566vZ5WVk5opAiMTvWgaQT8DkbOqsTfvNAvHoRGU2zzP8cPoqys+xHTRDWW8L+/BA==", 1052 | "dev": true 1053 | }, 1054 | "acorn": { 1055 | "version": "8.7.1", 1056 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.1.tgz", 1057 | "integrity": "sha512-Xx54uLJQZ19lKygFXOWsscKUbsBZW0CPykPhVQdhIeIwrbPmJzqeASDInc8nKBnp/JT6igTs82qPXz069H8I/A==", 1058 | "dev": true 1059 | }, 1060 | "acorn-globals": { 1061 | "version": "6.0.0", 1062 | "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz", 1063 | "integrity": "sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg==", 1064 | "dev": true, 1065 | "requires": { 1066 | "acorn": "^7.1.1", 1067 | "acorn-walk": "^7.1.1" 1068 | }, 1069 | "dependencies": { 1070 | "acorn": { 1071 | "version": "7.4.1", 1072 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", 1073 | "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", 1074 | "dev": true 1075 | } 1076 | } 1077 | }, 1078 | "acorn-walk": { 1079 | "version": "7.2.0", 1080 | "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", 1081 | "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", 1082 | "dev": true 1083 | }, 1084 | "agent-base": { 1085 | "version": "6.0.2", 1086 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", 1087 | "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", 1088 | "dev": true, 1089 | "requires": { 1090 | "debug": "4" 1091 | } 1092 | }, 1093 | "ansi-escapes": { 1094 | "version": "4.3.2", 1095 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", 1096 | "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", 1097 | "dev": true, 1098 | "requires": { 1099 | "type-fest": "^0.21.3" 1100 | } 1101 | }, 1102 | "ansi-regex": { 1103 | "version": "5.0.1", 1104 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 1105 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 1106 | "dev": true 1107 | }, 1108 | "ansi-styles": { 1109 | "version": "4.3.0", 1110 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1111 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1112 | "dev": true, 1113 | "requires": { 1114 | "color-convert": "^2.0.1" 1115 | } 1116 | }, 1117 | "anymatch": { 1118 | "version": "3.1.2", 1119 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 1120 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 1121 | "dev": true, 1122 | "requires": { 1123 | "normalize-path": "^3.0.0", 1124 | "picomatch": "^2.0.4" 1125 | } 1126 | }, 1127 | "arg": { 1128 | "version": "4.1.3", 1129 | "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", 1130 | "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", 1131 | "dev": true 1132 | }, 1133 | "argparse": { 1134 | "version": "1.0.10", 1135 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 1136 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 1137 | "dev": true, 1138 | "requires": { 1139 | "sprintf-js": "~1.0.2" 1140 | } 1141 | }, 1142 | "asynckit": { 1143 | "version": "0.4.0", 1144 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 1145 | "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", 1146 | "dev": true 1147 | }, 1148 | "aws-cdk": { 1149 | "version": "2.27.0", 1150 | "resolved": "https://registry.npmjs.org/aws-cdk/-/aws-cdk-2.27.0.tgz", 1151 | "integrity": "sha512-TAKSP4ViFqj+jktFwJl4IE4jZbQCsH32t7L/VcGY71VZV9VqvuzkjF4gGx+hrES+A/dFBtLmPXsumyJu5y0elQ==", 1152 | "dev": true, 1153 | "requires": { 1154 | "fsevents": "2.3.2" 1155 | } 1156 | }, 1157 | "aws-cdk-lib": { 1158 | "version": "2.80.0", 1159 | "resolved": "https://registry.npmjs.org/aws-cdk-lib/-/aws-cdk-lib-2.80.0.tgz", 1160 | "integrity": "sha512-PoqD3Yms5I0ajuTi071nTW/hpkH3XsdyZzn5gYsPv0qD7mqP3h6Qr+6RiGx+yQ1KcVFyxWdX15uK+DsC0KwvcQ==", 1161 | "requires": { 1162 | "@aws-cdk/asset-awscli-v1": "^2.2.177", 1163 | "@aws-cdk/asset-kubectl-v20": "^2.1.1", 1164 | "@aws-cdk/asset-node-proxy-agent-v5": "^2.0.148", 1165 | "@balena/dockerignore": "^1.0.2", 1166 | "case": "1.6.3", 1167 | "fs-extra": "^11.1.1", 1168 | "ignore": "^5.2.4", 1169 | "jsonschema": "^1.4.1", 1170 | "minimatch": "^3.1.2", 1171 | "punycode": "^2.3.0", 1172 | "semver": "^7.5.1", 1173 | "table": "^6.8.1", 1174 | "yaml": "1.10.2" 1175 | }, 1176 | "dependencies": { 1177 | "@balena/dockerignore": { 1178 | "version": "1.0.2", 1179 | "bundled": true 1180 | }, 1181 | "ajv": { 1182 | "version": "8.12.0", 1183 | "bundled": true, 1184 | "requires": { 1185 | "fast-deep-equal": "^3.1.1", 1186 | "json-schema-traverse": "^1.0.0", 1187 | "require-from-string": "^2.0.2", 1188 | "uri-js": "^4.2.2" 1189 | } 1190 | }, 1191 | "ansi-regex": { 1192 | "version": "5.0.1", 1193 | "bundled": true 1194 | }, 1195 | "ansi-styles": { 1196 | "version": "4.3.0", 1197 | "bundled": true, 1198 | "requires": { 1199 | "color-convert": "^2.0.1" 1200 | } 1201 | }, 1202 | "astral-regex": { 1203 | "version": "2.0.0", 1204 | "bundled": true 1205 | }, 1206 | "balanced-match": { 1207 | "version": "1.0.2", 1208 | "bundled": true 1209 | }, 1210 | "brace-expansion": { 1211 | "version": "1.1.11", 1212 | "bundled": true, 1213 | "requires": { 1214 | "balanced-match": "^1.0.0", 1215 | "concat-map": "0.0.1" 1216 | } 1217 | }, 1218 | "case": { 1219 | "version": "1.6.3", 1220 | "bundled": true 1221 | }, 1222 | "color-convert": { 1223 | "version": "2.0.1", 1224 | "bundled": true, 1225 | "requires": { 1226 | "color-name": "~1.1.4" 1227 | } 1228 | }, 1229 | "color-name": { 1230 | "version": "1.1.4", 1231 | "bundled": true 1232 | }, 1233 | "concat-map": { 1234 | "version": "0.0.1", 1235 | "bundled": true 1236 | }, 1237 | "emoji-regex": { 1238 | "version": "8.0.0", 1239 | "bundled": true 1240 | }, 1241 | "fast-deep-equal": { 1242 | "version": "3.1.3", 1243 | "bundled": true 1244 | }, 1245 | "fs-extra": { 1246 | "version": "11.1.1", 1247 | "bundled": true, 1248 | "requires": { 1249 | "graceful-fs": "^4.2.0", 1250 | "jsonfile": "^6.0.1", 1251 | "universalify": "^2.0.0" 1252 | } 1253 | }, 1254 | "graceful-fs": { 1255 | "version": "4.2.11", 1256 | "bundled": true 1257 | }, 1258 | "ignore": { 1259 | "version": "5.2.4", 1260 | "bundled": true 1261 | }, 1262 | "is-fullwidth-code-point": { 1263 | "version": "3.0.0", 1264 | "bundled": true 1265 | }, 1266 | "json-schema-traverse": { 1267 | "version": "1.0.0", 1268 | "bundled": true 1269 | }, 1270 | "jsonfile": { 1271 | "version": "6.1.0", 1272 | "bundled": true, 1273 | "requires": { 1274 | "graceful-fs": "^4.1.6", 1275 | "universalify": "^2.0.0" 1276 | } 1277 | }, 1278 | "jsonschema": { 1279 | "version": "1.4.1", 1280 | "bundled": true 1281 | }, 1282 | "lodash.truncate": { 1283 | "version": "4.4.2", 1284 | "bundled": true 1285 | }, 1286 | "lru-cache": { 1287 | "version": "6.0.0", 1288 | "bundled": true, 1289 | "requires": { 1290 | "yallist": "^4.0.0" 1291 | } 1292 | }, 1293 | "minimatch": { 1294 | "version": "3.1.2", 1295 | "bundled": true, 1296 | "requires": { 1297 | "brace-expansion": "^1.1.7" 1298 | } 1299 | }, 1300 | "punycode": { 1301 | "version": "2.3.0", 1302 | "bundled": true 1303 | }, 1304 | "require-from-string": { 1305 | "version": "2.0.2", 1306 | "bundled": true 1307 | }, 1308 | "semver": { 1309 | "version": "7.5.1", 1310 | "bundled": true, 1311 | "requires": { 1312 | "lru-cache": "^6.0.0" 1313 | } 1314 | }, 1315 | "slice-ansi": { 1316 | "version": "4.0.0", 1317 | "bundled": true, 1318 | "requires": { 1319 | "ansi-styles": "^4.0.0", 1320 | "astral-regex": "^2.0.0", 1321 | "is-fullwidth-code-point": "^3.0.0" 1322 | } 1323 | }, 1324 | "string-width": { 1325 | "version": "4.2.3", 1326 | "bundled": true, 1327 | "requires": { 1328 | "emoji-regex": "^8.0.0", 1329 | "is-fullwidth-code-point": "^3.0.0", 1330 | "strip-ansi": "^6.0.1" 1331 | } 1332 | }, 1333 | "strip-ansi": { 1334 | "version": "6.0.1", 1335 | "bundled": true, 1336 | "requires": { 1337 | "ansi-regex": "^5.0.1" 1338 | } 1339 | }, 1340 | "table": { 1341 | "version": "6.8.1", 1342 | "bundled": true, 1343 | "requires": { 1344 | "ajv": "^8.0.1", 1345 | "lodash.truncate": "^4.4.2", 1346 | "slice-ansi": "^4.0.0", 1347 | "string-width": "^4.2.3", 1348 | "strip-ansi": "^6.0.1" 1349 | } 1350 | }, 1351 | "universalify": { 1352 | "version": "2.0.0", 1353 | "bundled": true 1354 | }, 1355 | "uri-js": { 1356 | "version": "4.4.1", 1357 | "bundled": true, 1358 | "requires": { 1359 | "punycode": "^2.1.0" 1360 | } 1361 | }, 1362 | "yallist": { 1363 | "version": "4.0.0", 1364 | "bundled": true 1365 | }, 1366 | "yaml": { 1367 | "version": "1.10.2", 1368 | "bundled": true 1369 | } 1370 | } 1371 | }, 1372 | "babel-jest": { 1373 | "version": "27.5.1", 1374 | "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-27.5.1.tgz", 1375 | "integrity": "sha512-cdQ5dXjGRd0IBRATiQ4mZGlGlRE8kJpjPOixdNRdT+m3UcNqmYWN6rK6nvtXYfY3D76cb8s/O1Ss8ea24PIwcg==", 1376 | "dev": true, 1377 | "requires": { 1378 | "@jest/transform": "^27.5.1", 1379 | "@jest/types": "^27.5.1", 1380 | "@types/babel__core": "^7.1.14", 1381 | "babel-plugin-istanbul": "^6.1.1", 1382 | "babel-preset-jest": "^27.5.1", 1383 | "chalk": "^4.0.0", 1384 | "graceful-fs": "^4.2.9", 1385 | "slash": "^3.0.0" 1386 | } 1387 | }, 1388 | "babel-plugin-istanbul": { 1389 | "version": "6.1.1", 1390 | "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", 1391 | "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", 1392 | "dev": true, 1393 | "requires": { 1394 | "@babel/helper-plugin-utils": "^7.0.0", 1395 | "@istanbuljs/load-nyc-config": "^1.0.0", 1396 | "@istanbuljs/schema": "^0.1.2", 1397 | "istanbul-lib-instrument": "^5.0.4", 1398 | "test-exclude": "^6.0.0" 1399 | } 1400 | }, 1401 | "babel-plugin-jest-hoist": { 1402 | "version": "27.5.1", 1403 | "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-27.5.1.tgz", 1404 | "integrity": "sha512-50wCwD5EMNW4aRpOwtqzyZHIewTYNxLA4nhB+09d8BIssfNfzBRhkBIHiaPv1Si226TQSvp8gxAJm2iY2qs2hQ==", 1405 | "dev": true, 1406 | "requires": { 1407 | "@babel/template": "^7.3.3", 1408 | "@babel/types": "^7.3.3", 1409 | "@types/babel__core": "^7.0.0", 1410 | "@types/babel__traverse": "^7.0.6" 1411 | } 1412 | }, 1413 | "babel-preset-current-node-syntax": { 1414 | "version": "1.0.1", 1415 | "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz", 1416 | "integrity": "sha512-M7LQ0bxarkxQoN+vz5aJPsLBn77n8QgTFmo8WK0/44auK2xlCXrYcUxHFxgU7qW5Yzw/CjmLRK2uJzaCd7LvqQ==", 1417 | "dev": true, 1418 | "requires": { 1419 | "@babel/plugin-syntax-async-generators": "^7.8.4", 1420 | "@babel/plugin-syntax-bigint": "^7.8.3", 1421 | "@babel/plugin-syntax-class-properties": "^7.8.3", 1422 | "@babel/plugin-syntax-import-meta": "^7.8.3", 1423 | "@babel/plugin-syntax-json-strings": "^7.8.3", 1424 | "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", 1425 | "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", 1426 | "@babel/plugin-syntax-numeric-separator": "^7.8.3", 1427 | "@babel/plugin-syntax-object-rest-spread": "^7.8.3", 1428 | "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", 1429 | "@babel/plugin-syntax-optional-chaining": "^7.8.3", 1430 | "@babel/plugin-syntax-top-level-await": "^7.8.3" 1431 | } 1432 | }, 1433 | "babel-preset-jest": { 1434 | "version": "27.5.1", 1435 | "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-27.5.1.tgz", 1436 | "integrity": "sha512-Nptf2FzlPCWYuJg41HBqXVT8ym6bXOevuCTbhxlUpjwtysGaIWFvDEjp4y+G7fl13FgOdjs7P/DmErqH7da0Ag==", 1437 | "dev": true, 1438 | "requires": { 1439 | "babel-plugin-jest-hoist": "^27.5.1", 1440 | "babel-preset-current-node-syntax": "^1.0.0" 1441 | } 1442 | }, 1443 | "balanced-match": { 1444 | "version": "1.0.2", 1445 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 1446 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 1447 | "dev": true 1448 | }, 1449 | "brace-expansion": { 1450 | "version": "1.1.11", 1451 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 1452 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 1453 | "dev": true, 1454 | "requires": { 1455 | "balanced-match": "^1.0.0", 1456 | "concat-map": "0.0.1" 1457 | } 1458 | }, 1459 | "braces": { 1460 | "version": "3.0.3", 1461 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", 1462 | "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", 1463 | "dev": true, 1464 | "requires": { 1465 | "fill-range": "^7.1.1" 1466 | }, 1467 | "dependencies": { 1468 | "fill-range": { 1469 | "version": "7.1.1", 1470 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", 1471 | "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", 1472 | "dev": true, 1473 | "requires": { 1474 | "to-regex-range": "^5.0.1" 1475 | } 1476 | } 1477 | } 1478 | }, 1479 | "browser-process-hrtime": { 1480 | "version": "1.0.0", 1481 | "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", 1482 | "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==", 1483 | "dev": true 1484 | }, 1485 | "browserslist": { 1486 | "version": "4.20.4", 1487 | "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.20.4.tgz", 1488 | "integrity": "sha512-ok1d+1WpnU24XYN7oC3QWgTyMhY/avPJ/r9T00xxvUOIparA/gc+UPUMaod3i+G6s+nI2nUb9xZ5k794uIwShw==", 1489 | "dev": true, 1490 | "requires": { 1491 | "caniuse-lite": "^1.0.30001349", 1492 | "electron-to-chromium": "^1.4.147", 1493 | "escalade": "^3.1.1", 1494 | "node-releases": "^2.0.5", 1495 | "picocolors": "^1.0.0" 1496 | } 1497 | }, 1498 | "bs-logger": { 1499 | "version": "0.2.6", 1500 | "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", 1501 | "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", 1502 | "dev": true, 1503 | "requires": { 1504 | "fast-json-stable-stringify": "2.x" 1505 | } 1506 | }, 1507 | "bser": { 1508 | "version": "2.1.1", 1509 | "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", 1510 | "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", 1511 | "dev": true, 1512 | "requires": { 1513 | "node-int64": "^0.4.0" 1514 | } 1515 | }, 1516 | "buffer-from": { 1517 | "version": "1.1.2", 1518 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", 1519 | "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" 1520 | }, 1521 | "callsites": { 1522 | "version": "3.1.0", 1523 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 1524 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 1525 | "dev": true 1526 | }, 1527 | "camelcase": { 1528 | "version": "5.3.1", 1529 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 1530 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 1531 | "dev": true 1532 | }, 1533 | "caniuse-lite": { 1534 | "version": "1.0.30001352", 1535 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001352.tgz", 1536 | "integrity": "sha512-GUgH8w6YergqPQDGWhJGt8GDRnY0L/iJVQcU3eJ46GYf52R8tk0Wxp0PymuFVZboJYXGiCqwozAYZNRjVj6IcA==", 1537 | "dev": true 1538 | }, 1539 | "chalk": { 1540 | "version": "4.1.2", 1541 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 1542 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 1543 | "dev": true, 1544 | "requires": { 1545 | "ansi-styles": "^4.1.0", 1546 | "supports-color": "^7.1.0" 1547 | } 1548 | }, 1549 | "char-regex": { 1550 | "version": "1.0.2", 1551 | "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", 1552 | "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", 1553 | "dev": true 1554 | }, 1555 | "ci-info": { 1556 | "version": "3.3.1", 1557 | "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.3.1.tgz", 1558 | "integrity": "sha512-SXgeMX9VwDe7iFFaEWkA5AstuER9YKqy4EhHqr4DVqkwmD9rpVimkMKWHdjn30Ja45txyjhSn63lVX69eVCckg==", 1559 | "dev": true 1560 | }, 1561 | "cjs-module-lexer": { 1562 | "version": "1.2.2", 1563 | "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.2.2.tgz", 1564 | "integrity": "sha512-cOU9usZw8/dXIXKtwa8pM0OTJQuJkxMN6w30csNRUerHfeQ5R6U3kkU/FtJeIf3M202OHfY2U8ccInBG7/xogA==", 1565 | "dev": true 1566 | }, 1567 | "cliui": { 1568 | "version": "7.0.4", 1569 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", 1570 | "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", 1571 | "dev": true, 1572 | "requires": { 1573 | "string-width": "^4.2.0", 1574 | "strip-ansi": "^6.0.0", 1575 | "wrap-ansi": "^7.0.0" 1576 | } 1577 | }, 1578 | "co": { 1579 | "version": "4.6.0", 1580 | "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", 1581 | "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", 1582 | "dev": true 1583 | }, 1584 | "collect-v8-coverage": { 1585 | "version": "1.0.1", 1586 | "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz", 1587 | "integrity": "sha512-iBPtljfCNcTKNAto0KEtDfZ3qzjJvqE3aTGZsbhjSBlorqpXJlaWWtPO35D+ZImoC3KWejX64o+yPGxhWSTzfg==", 1588 | "dev": true 1589 | }, 1590 | "color-convert": { 1591 | "version": "2.0.1", 1592 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 1593 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 1594 | "dev": true, 1595 | "requires": { 1596 | "color-name": "~1.1.4" 1597 | } 1598 | }, 1599 | "color-name": { 1600 | "version": "1.1.4", 1601 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1602 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1603 | "dev": true 1604 | }, 1605 | "combined-stream": { 1606 | "version": "1.0.8", 1607 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 1608 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 1609 | "dev": true, 1610 | "requires": { 1611 | "delayed-stream": "~1.0.0" 1612 | } 1613 | }, 1614 | "concat-map": { 1615 | "version": "0.0.1", 1616 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 1617 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 1618 | "dev": true 1619 | }, 1620 | "constructs": { 1621 | "version": "10.1.33", 1622 | "resolved": "https://registry.npmjs.org/constructs/-/constructs-10.1.33.tgz", 1623 | "integrity": "sha512-OgqPYjBLbMFcaQ9bsdvF59orh46GNwg3tBdyOlhuWm8ljRxaQ2jEJ3XAj06I0/+sTAcsF9sZ9GAVmJIiCTnBkA==" 1624 | }, 1625 | "convert-source-map": { 1626 | "version": "1.8.0", 1627 | "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.8.0.tgz", 1628 | "integrity": "sha512-+OQdjP49zViI/6i7nIJpA8rAl4sV/JdPfU9nZs3VqOwGIgizICvuN2ru6fMd+4llL0tar18UYJXfZ/TWtmhUjA==", 1629 | "dev": true, 1630 | "requires": { 1631 | "safe-buffer": "~5.1.1" 1632 | } 1633 | }, 1634 | "create-require": { 1635 | "version": "1.1.1", 1636 | "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", 1637 | "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", 1638 | "dev": true 1639 | }, 1640 | "cross-spawn": { 1641 | "version": "7.0.3", 1642 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", 1643 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", 1644 | "dev": true, 1645 | "requires": { 1646 | "path-key": "^3.1.0", 1647 | "shebang-command": "^2.0.0", 1648 | "which": "^2.0.1" 1649 | } 1650 | }, 1651 | "cssom": { 1652 | "version": "0.4.4", 1653 | "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz", 1654 | "integrity": "sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw==", 1655 | "dev": true 1656 | }, 1657 | "cssstyle": { 1658 | "version": "2.3.0", 1659 | "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", 1660 | "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", 1661 | "dev": true, 1662 | "requires": { 1663 | "cssom": "~0.3.6" 1664 | }, 1665 | "dependencies": { 1666 | "cssom": { 1667 | "version": "0.3.8", 1668 | "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", 1669 | "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", 1670 | "dev": true 1671 | } 1672 | } 1673 | }, 1674 | "data-urls": { 1675 | "version": "2.0.0", 1676 | "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", 1677 | "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", 1678 | "dev": true, 1679 | "requires": { 1680 | "abab": "^2.0.3", 1681 | "whatwg-mimetype": "^2.3.0", 1682 | "whatwg-url": "^8.0.0" 1683 | } 1684 | }, 1685 | "debug": { 1686 | "version": "4.3.4", 1687 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1688 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1689 | "dev": true, 1690 | "requires": { 1691 | "ms": "2.1.2" 1692 | } 1693 | }, 1694 | "decimal.js": { 1695 | "version": "10.3.1", 1696 | "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.3.1.tgz", 1697 | "integrity": "sha512-V0pfhfr8suzyPGOx3nmq4aHqabehUZn6Ch9kyFpV79TGDTWFmHqUqXdabR7QHqxzrYolF4+tVmJhUG4OURg5dQ==", 1698 | "dev": true 1699 | }, 1700 | "dedent": { 1701 | "version": "0.7.0", 1702 | "resolved": "https://registry.npmjs.org/dedent/-/dedent-0.7.0.tgz", 1703 | "integrity": "sha512-Q6fKUPqnAHAyhiUgFU7BUzLiv0kd8saH9al7tnu5Q/okj6dnupxyTgFIBjVzJATdfIAm9NAsvXNzjaKa+bxVyA==", 1704 | "dev": true 1705 | }, 1706 | "deep-is": { 1707 | "version": "0.1.4", 1708 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", 1709 | "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", 1710 | "dev": true 1711 | }, 1712 | "deepmerge": { 1713 | "version": "4.2.2", 1714 | "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", 1715 | "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==", 1716 | "dev": true 1717 | }, 1718 | "delayed-stream": { 1719 | "version": "1.0.0", 1720 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 1721 | "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", 1722 | "dev": true 1723 | }, 1724 | "detect-newline": { 1725 | "version": "3.1.0", 1726 | "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", 1727 | "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", 1728 | "dev": true 1729 | }, 1730 | "diff": { 1731 | "version": "4.0.2", 1732 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", 1733 | "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", 1734 | "dev": true 1735 | }, 1736 | "diff-sequences": { 1737 | "version": "27.5.1", 1738 | "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-27.5.1.tgz", 1739 | "integrity": "sha512-k1gCAXAsNgLwEL+Y8Wvl+M6oEFj5bgazfZULpS5CneoPPXRaCCW7dm+q21Ky2VEE5X+VeRDBVg1Pcvvsr4TtNQ==", 1740 | "dev": true 1741 | }, 1742 | "domexception": { 1743 | "version": "2.0.1", 1744 | "resolved": "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz", 1745 | "integrity": "sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg==", 1746 | "dev": true, 1747 | "requires": { 1748 | "webidl-conversions": "^5.0.0" 1749 | }, 1750 | "dependencies": { 1751 | "webidl-conversions": { 1752 | "version": "5.0.0", 1753 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz", 1754 | "integrity": "sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA==", 1755 | "dev": true 1756 | } 1757 | } 1758 | }, 1759 | "electron-to-chromium": { 1760 | "version": "1.4.150", 1761 | "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.150.tgz", 1762 | "integrity": "sha512-MP3oBer0X7ZeS9GJ0H6lmkn561UxiwOIY9TTkdxVY7lI9G6GVCKfgJaHaDcakwdKxBXA4T3ybeswH/WBIN/KTA==", 1763 | "dev": true 1764 | }, 1765 | "emittery": { 1766 | "version": "0.8.1", 1767 | "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.8.1.tgz", 1768 | "integrity": "sha512-uDfvUjVrfGJJhymx/kz6prltenw1u7WrCg1oa94zYY8xxVpLLUu045LAT0dhDZdXG58/EpPL/5kA180fQ/qudg==", 1769 | "dev": true 1770 | }, 1771 | "emoji-regex": { 1772 | "version": "8.0.0", 1773 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1774 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 1775 | "dev": true 1776 | }, 1777 | "error-ex": { 1778 | "version": "1.3.2", 1779 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", 1780 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", 1781 | "dev": true, 1782 | "requires": { 1783 | "is-arrayish": "^0.2.1" 1784 | } 1785 | }, 1786 | "escalade": { 1787 | "version": "3.1.1", 1788 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 1789 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 1790 | "dev": true 1791 | }, 1792 | "escape-string-regexp": { 1793 | "version": "1.0.5", 1794 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 1795 | "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", 1796 | "dev": true 1797 | }, 1798 | "escodegen": { 1799 | "version": "2.0.0", 1800 | "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.0.0.tgz", 1801 | "integrity": "sha512-mmHKys/C8BFUGI+MAWNcSYoORYLMdPzjrknd2Vc+bUsjN5bXcr8EhrNB+UTqfL1y3I9c4fw2ihgtMPQLBRiQxw==", 1802 | "dev": true, 1803 | "requires": { 1804 | "esprima": "^4.0.1", 1805 | "estraverse": "^5.2.0", 1806 | "esutils": "^2.0.2", 1807 | "optionator": "^0.8.1", 1808 | "source-map": "~0.6.1" 1809 | } 1810 | }, 1811 | "esprima": { 1812 | "version": "4.0.1", 1813 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 1814 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 1815 | "dev": true 1816 | }, 1817 | "estraverse": { 1818 | "version": "5.3.0", 1819 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", 1820 | "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", 1821 | "dev": true 1822 | }, 1823 | "esutils": { 1824 | "version": "2.0.3", 1825 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 1826 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 1827 | "dev": true 1828 | }, 1829 | "execa": { 1830 | "version": "5.1.1", 1831 | "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", 1832 | "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", 1833 | "dev": true, 1834 | "requires": { 1835 | "cross-spawn": "^7.0.3", 1836 | "get-stream": "^6.0.0", 1837 | "human-signals": "^2.1.0", 1838 | "is-stream": "^2.0.0", 1839 | "merge-stream": "^2.0.0", 1840 | "npm-run-path": "^4.0.1", 1841 | "onetime": "^5.1.2", 1842 | "signal-exit": "^3.0.3", 1843 | "strip-final-newline": "^2.0.0" 1844 | } 1845 | }, 1846 | "exit": { 1847 | "version": "0.1.2", 1848 | "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", 1849 | "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", 1850 | "dev": true 1851 | }, 1852 | "expect": { 1853 | "version": "27.5.1", 1854 | "resolved": "https://registry.npmjs.org/expect/-/expect-27.5.1.tgz", 1855 | "integrity": "sha512-E1q5hSUG2AmYQwQJ041nvgpkODHQvB+RKlB4IYdru6uJsyFTRyZAP463M+1lINorwbqAmUggi6+WwkD8lCS/Dw==", 1856 | "dev": true, 1857 | "requires": { 1858 | "@jest/types": "^27.5.1", 1859 | "jest-get-type": "^27.5.1", 1860 | "jest-matcher-utils": "^27.5.1", 1861 | "jest-message-util": "^27.5.1" 1862 | } 1863 | }, 1864 | "fast-json-stable-stringify": { 1865 | "version": "2.1.0", 1866 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 1867 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 1868 | "dev": true 1869 | }, 1870 | "fast-levenshtein": { 1871 | "version": "2.0.6", 1872 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 1873 | "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", 1874 | "dev": true 1875 | }, 1876 | "fb-watchman": { 1877 | "version": "2.0.1", 1878 | "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.1.tgz", 1879 | "integrity": "sha512-DkPJKQeY6kKwmuMretBhr7G6Vodr7bFwDYTXIkfG1gjvNpaxBTQV3PbXg6bR1c1UP4jPOX0jHUbbHANL9vRjVg==", 1880 | "dev": true, 1881 | "requires": { 1882 | "bser": "2.1.1" 1883 | } 1884 | }, 1885 | "find-up": { 1886 | "version": "4.1.0", 1887 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 1888 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 1889 | "dev": true, 1890 | "requires": { 1891 | "locate-path": "^5.0.0", 1892 | "path-exists": "^4.0.0" 1893 | } 1894 | }, 1895 | "form-data": { 1896 | "version": "3.0.1", 1897 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", 1898 | "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", 1899 | "dev": true, 1900 | "requires": { 1901 | "asynckit": "^0.4.0", 1902 | "combined-stream": "^1.0.8", 1903 | "mime-types": "^2.1.12" 1904 | } 1905 | }, 1906 | "fs.realpath": { 1907 | "version": "1.0.0", 1908 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1909 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 1910 | "dev": true 1911 | }, 1912 | "fsevents": { 1913 | "version": "2.3.2", 1914 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 1915 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 1916 | "dev": true, 1917 | "optional": true 1918 | }, 1919 | "function-bind": { 1920 | "version": "1.1.1", 1921 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", 1922 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", 1923 | "dev": true 1924 | }, 1925 | "gensync": { 1926 | "version": "1.0.0-beta.2", 1927 | "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", 1928 | "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", 1929 | "dev": true 1930 | }, 1931 | "get-caller-file": { 1932 | "version": "2.0.5", 1933 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 1934 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 1935 | "dev": true 1936 | }, 1937 | "get-package-type": { 1938 | "version": "0.1.0", 1939 | "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", 1940 | "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", 1941 | "dev": true 1942 | }, 1943 | "get-stream": { 1944 | "version": "6.0.1", 1945 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", 1946 | "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", 1947 | "dev": true 1948 | }, 1949 | "glob": { 1950 | "version": "7.2.3", 1951 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 1952 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 1953 | "dev": true, 1954 | "requires": { 1955 | "fs.realpath": "^1.0.0", 1956 | "inflight": "^1.0.4", 1957 | "inherits": "2", 1958 | "minimatch": "^3.1.1", 1959 | "once": "^1.3.0", 1960 | "path-is-absolute": "^1.0.0" 1961 | } 1962 | }, 1963 | "globals": { 1964 | "version": "11.12.0", 1965 | "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", 1966 | "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", 1967 | "dev": true 1968 | }, 1969 | "graceful-fs": { 1970 | "version": "4.2.10", 1971 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", 1972 | "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", 1973 | "dev": true 1974 | }, 1975 | "has": { 1976 | "version": "1.0.3", 1977 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", 1978 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", 1979 | "dev": true, 1980 | "requires": { 1981 | "function-bind": "^1.1.1" 1982 | } 1983 | }, 1984 | "has-flag": { 1985 | "version": "4.0.0", 1986 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1987 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1988 | "dev": true 1989 | }, 1990 | "html-encoding-sniffer": { 1991 | "version": "2.0.1", 1992 | "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz", 1993 | "integrity": "sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ==", 1994 | "dev": true, 1995 | "requires": { 1996 | "whatwg-encoding": "^1.0.5" 1997 | } 1998 | }, 1999 | "html-escaper": { 2000 | "version": "2.0.2", 2001 | "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", 2002 | "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", 2003 | "dev": true 2004 | }, 2005 | "http-proxy-agent": { 2006 | "version": "4.0.1", 2007 | "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", 2008 | "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", 2009 | "dev": true, 2010 | "requires": { 2011 | "@tootallnate/once": "1", 2012 | "agent-base": "6", 2013 | "debug": "4" 2014 | } 2015 | }, 2016 | "https-proxy-agent": { 2017 | "version": "5.0.1", 2018 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", 2019 | "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", 2020 | "dev": true, 2021 | "requires": { 2022 | "agent-base": "6", 2023 | "debug": "4" 2024 | } 2025 | }, 2026 | "human-signals": { 2027 | "version": "2.1.0", 2028 | "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", 2029 | "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", 2030 | "dev": true 2031 | }, 2032 | "iconv-lite": { 2033 | "version": "0.4.24", 2034 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 2035 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 2036 | "dev": true, 2037 | "requires": { 2038 | "safer-buffer": ">= 2.1.2 < 3" 2039 | } 2040 | }, 2041 | "import-local": { 2042 | "version": "3.1.0", 2043 | "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", 2044 | "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", 2045 | "dev": true, 2046 | "requires": { 2047 | "pkg-dir": "^4.2.0", 2048 | "resolve-cwd": "^3.0.0" 2049 | } 2050 | }, 2051 | "imurmurhash": { 2052 | "version": "0.1.4", 2053 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 2054 | "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", 2055 | "dev": true 2056 | }, 2057 | "inflight": { 2058 | "version": "1.0.6", 2059 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 2060 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 2061 | "dev": true, 2062 | "requires": { 2063 | "once": "^1.3.0", 2064 | "wrappy": "1" 2065 | } 2066 | }, 2067 | "inherits": { 2068 | "version": "2.0.4", 2069 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 2070 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 2071 | "dev": true 2072 | }, 2073 | "is-arrayish": { 2074 | "version": "0.2.1", 2075 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", 2076 | "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", 2077 | "dev": true 2078 | }, 2079 | "is-core-module": { 2080 | "version": "2.9.0", 2081 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.9.0.tgz", 2082 | "integrity": "sha512-+5FPy5PnwmO3lvfMb0AsoPaBG+5KHUI0wYFXOtYPnVVVspTFUuMZNfNaNVRt3FZadstu2c8x23vykRW/NBoU6A==", 2083 | "dev": true, 2084 | "requires": { 2085 | "has": "^1.0.3" 2086 | } 2087 | }, 2088 | "is-fullwidth-code-point": { 2089 | "version": "3.0.0", 2090 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 2091 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 2092 | "dev": true 2093 | }, 2094 | "is-generator-fn": { 2095 | "version": "2.1.0", 2096 | "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", 2097 | "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", 2098 | "dev": true 2099 | }, 2100 | "is-number": { 2101 | "version": "7.0.0", 2102 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 2103 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 2104 | "dev": true 2105 | }, 2106 | "is-potential-custom-element-name": { 2107 | "version": "1.0.1", 2108 | "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", 2109 | "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", 2110 | "dev": true 2111 | }, 2112 | "is-stream": { 2113 | "version": "2.0.1", 2114 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", 2115 | "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", 2116 | "dev": true 2117 | }, 2118 | "is-typedarray": { 2119 | "version": "1.0.0", 2120 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 2121 | "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==", 2122 | "dev": true 2123 | }, 2124 | "isexe": { 2125 | "version": "2.0.0", 2126 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 2127 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", 2128 | "dev": true 2129 | }, 2130 | "istanbul-lib-coverage": { 2131 | "version": "3.2.0", 2132 | "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz", 2133 | "integrity": "sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw==", 2134 | "dev": true 2135 | }, 2136 | "istanbul-lib-instrument": { 2137 | "version": "5.2.0", 2138 | "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.0.tgz", 2139 | "integrity": "sha512-6Lthe1hqXHBNsqvgDzGO6l03XNeu3CrG4RqQ1KM9+l5+jNGpEJfIELx1NS3SEHmJQA8np/u+E4EPRKRiu6m19A==", 2140 | "dev": true, 2141 | "requires": { 2142 | "@babel/core": "^7.12.3", 2143 | "@babel/parser": "^7.14.7", 2144 | "@istanbuljs/schema": "^0.1.2", 2145 | "istanbul-lib-coverage": "^3.2.0", 2146 | "semver": "^6.3.0" 2147 | } 2148 | }, 2149 | "istanbul-lib-report": { 2150 | "version": "3.0.0", 2151 | "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", 2152 | "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", 2153 | "dev": true, 2154 | "requires": { 2155 | "istanbul-lib-coverage": "^3.0.0", 2156 | "make-dir": "^3.0.0", 2157 | "supports-color": "^7.1.0" 2158 | } 2159 | }, 2160 | "istanbul-lib-source-maps": { 2161 | "version": "4.0.1", 2162 | "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", 2163 | "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", 2164 | "dev": true, 2165 | "requires": { 2166 | "debug": "^4.1.1", 2167 | "istanbul-lib-coverage": "^3.0.0", 2168 | "source-map": "^0.6.1" 2169 | } 2170 | }, 2171 | "istanbul-reports": { 2172 | "version": "3.1.4", 2173 | "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.4.tgz", 2174 | "integrity": "sha512-r1/DshN4KSE7xWEknZLLLLDn5CJybV3nw01VTkp6D5jzLuELlcbudfj/eSQFvrKsJuTVCGnePO7ho82Nw9zzfw==", 2175 | "dev": true, 2176 | "requires": { 2177 | "html-escaper": "^2.0.0", 2178 | "istanbul-lib-report": "^3.0.0" 2179 | } 2180 | }, 2181 | "jest": { 2182 | "version": "27.5.1", 2183 | "resolved": "https://registry.npmjs.org/jest/-/jest-27.5.1.tgz", 2184 | "integrity": "sha512-Yn0mADZB89zTtjkPJEXwrac3LHudkQMR+Paqa8uxJHCBr9agxztUifWCyiYrjhMPBoUVBjyny0I7XH6ozDr7QQ==", 2185 | "dev": true, 2186 | "requires": { 2187 | "@jest/core": "^27.5.1", 2188 | "import-local": "^3.0.2", 2189 | "jest-cli": "^27.5.1" 2190 | }, 2191 | "dependencies": { 2192 | "jest-cli": { 2193 | "version": "27.5.1", 2194 | "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-27.5.1.tgz", 2195 | "integrity": "sha512-Hc6HOOwYq4/74/c62dEE3r5elx8wjYqxY0r0G/nFrLDPMFRu6RA/u8qINOIkvhxG7mMQ5EJsOGfRpI8L6eFUVw==", 2196 | "dev": true, 2197 | "requires": { 2198 | "@jest/core": "^27.5.1", 2199 | "@jest/test-result": "^27.5.1", 2200 | "@jest/types": "^27.5.1", 2201 | "chalk": "^4.0.0", 2202 | "exit": "^0.1.2", 2203 | "graceful-fs": "^4.2.9", 2204 | "import-local": "^3.0.2", 2205 | "jest-config": "^27.5.1", 2206 | "jest-util": "^27.5.1", 2207 | "jest-validate": "^27.5.1", 2208 | "prompts": "^2.0.1", 2209 | "yargs": "^16.2.0" 2210 | } 2211 | } 2212 | } 2213 | }, 2214 | "jest-changed-files": { 2215 | "version": "27.5.1", 2216 | "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-27.5.1.tgz", 2217 | "integrity": "sha512-buBLMiByfWGCoMsLLzGUUSpAmIAGnbR2KJoMN10ziLhOLvP4e0SlypHnAel8iqQXTrcbmfEY9sSqae5sgUsTvw==", 2218 | "dev": true, 2219 | "requires": { 2220 | "@jest/types": "^27.5.1", 2221 | "execa": "^5.0.0", 2222 | "throat": "^6.0.1" 2223 | } 2224 | }, 2225 | "jest-circus": { 2226 | "version": "27.5.1", 2227 | "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-27.5.1.tgz", 2228 | "integrity": "sha512-D95R7x5UtlMA5iBYsOHFFbMD/GVA4R/Kdq15f7xYWUfWHBto9NYRsOvnSauTgdF+ogCpJ4tyKOXhUifxS65gdw==", 2229 | "dev": true, 2230 | "requires": { 2231 | "@jest/environment": "^27.5.1", 2232 | "@jest/test-result": "^27.5.1", 2233 | "@jest/types": "^27.5.1", 2234 | "@types/node": "*", 2235 | "chalk": "^4.0.0", 2236 | "co": "^4.6.0", 2237 | "dedent": "^0.7.0", 2238 | "expect": "^27.5.1", 2239 | "is-generator-fn": "^2.0.0", 2240 | "jest-each": "^27.5.1", 2241 | "jest-matcher-utils": "^27.5.1", 2242 | "jest-message-util": "^27.5.1", 2243 | "jest-runtime": "^27.5.1", 2244 | "jest-snapshot": "^27.5.1", 2245 | "jest-util": "^27.5.1", 2246 | "pretty-format": "^27.5.1", 2247 | "slash": "^3.0.0", 2248 | "stack-utils": "^2.0.3", 2249 | "throat": "^6.0.1" 2250 | } 2251 | }, 2252 | "jest-config": { 2253 | "version": "27.5.1", 2254 | "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-27.5.1.tgz", 2255 | "integrity": "sha512-5sAsjm6tGdsVbW9ahcChPAFCk4IlkQUknH5AvKjuLTSlcO/wCZKyFdn7Rg0EkC+OGgWODEy2hDpWB1PgzH0JNA==", 2256 | "dev": true, 2257 | "requires": { 2258 | "@babel/core": "^7.8.0", 2259 | "@jest/test-sequencer": "^27.5.1", 2260 | "@jest/types": "^27.5.1", 2261 | "babel-jest": "^27.5.1", 2262 | "chalk": "^4.0.0", 2263 | "ci-info": "^3.2.0", 2264 | "deepmerge": "^4.2.2", 2265 | "glob": "^7.1.1", 2266 | "graceful-fs": "^4.2.9", 2267 | "jest-circus": "^27.5.1", 2268 | "jest-environment-jsdom": "^27.5.1", 2269 | "jest-environment-node": "^27.5.1", 2270 | "jest-get-type": "^27.5.1", 2271 | "jest-jasmine2": "^27.5.1", 2272 | "jest-regex-util": "^27.5.1", 2273 | "jest-resolve": "^27.5.1", 2274 | "jest-runner": "^27.5.1", 2275 | "jest-util": "^27.5.1", 2276 | "jest-validate": "^27.5.1", 2277 | "micromatch": "^4.0.4", 2278 | "parse-json": "^5.2.0", 2279 | "pretty-format": "^27.5.1", 2280 | "slash": "^3.0.0", 2281 | "strip-json-comments": "^3.1.1" 2282 | } 2283 | }, 2284 | "jest-diff": { 2285 | "version": "27.5.1", 2286 | "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-27.5.1.tgz", 2287 | "integrity": "sha512-m0NvkX55LDt9T4mctTEgnZk3fmEg3NRYutvMPWM/0iPnkFj2wIeF45O1718cMSOFO1vINkqmxqD8vE37uTEbqw==", 2288 | "dev": true, 2289 | "requires": { 2290 | "chalk": "^4.0.0", 2291 | "diff-sequences": "^27.5.1", 2292 | "jest-get-type": "^27.5.1", 2293 | "pretty-format": "^27.5.1" 2294 | } 2295 | }, 2296 | "jest-docblock": { 2297 | "version": "27.5.1", 2298 | "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-27.5.1.tgz", 2299 | "integrity": "sha512-rl7hlABeTsRYxKiUfpHrQrG4e2obOiTQWfMEH3PxPjOtdsfLQO4ReWSZaQ7DETm4xu07rl4q/h4zcKXyU0/OzQ==", 2300 | "dev": true, 2301 | "requires": { 2302 | "detect-newline": "^3.0.0" 2303 | } 2304 | }, 2305 | "jest-each": { 2306 | "version": "27.5.1", 2307 | "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-27.5.1.tgz", 2308 | "integrity": "sha512-1Ff6p+FbhT/bXQnEouYy00bkNSY7OUpfIcmdl8vZ31A1UUaurOLPA8a8BbJOF2RDUElwJhmeaV7LnagI+5UwNQ==", 2309 | "dev": true, 2310 | "requires": { 2311 | "@jest/types": "^27.5.1", 2312 | "chalk": "^4.0.0", 2313 | "jest-get-type": "^27.5.1", 2314 | "jest-util": "^27.5.1", 2315 | "pretty-format": "^27.5.1" 2316 | } 2317 | }, 2318 | "jest-environment-jsdom": { 2319 | "version": "27.5.1", 2320 | "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-27.5.1.tgz", 2321 | "integrity": "sha512-TFBvkTC1Hnnnrka/fUb56atfDtJ9VMZ94JkjTbggl1PEpwrYtUBKMezB3inLmWqQsXYLcMwNoDQwoBTAvFfsfw==", 2322 | "dev": true, 2323 | "requires": { 2324 | "@jest/environment": "^27.5.1", 2325 | "@jest/fake-timers": "^27.5.1", 2326 | "@jest/types": "^27.5.1", 2327 | "@types/node": "*", 2328 | "jest-mock": "^27.5.1", 2329 | "jest-util": "^27.5.1", 2330 | "jsdom": "^16.6.0" 2331 | } 2332 | }, 2333 | "jest-environment-node": { 2334 | "version": "27.5.1", 2335 | "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-27.5.1.tgz", 2336 | "integrity": "sha512-Jt4ZUnxdOsTGwSRAfKEnE6BcwsSPNOijjwifq5sDFSA2kesnXTvNqKHYgM0hDq3549Uf/KzdXNYn4wMZJPlFLw==", 2337 | "dev": true, 2338 | "requires": { 2339 | "@jest/environment": "^27.5.1", 2340 | "@jest/fake-timers": "^27.5.1", 2341 | "@jest/types": "^27.5.1", 2342 | "@types/node": "*", 2343 | "jest-mock": "^27.5.1", 2344 | "jest-util": "^27.5.1" 2345 | } 2346 | }, 2347 | "jest-get-type": { 2348 | "version": "27.5.1", 2349 | "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-27.5.1.tgz", 2350 | "integrity": "sha512-2KY95ksYSaK7DMBWQn6dQz3kqAf3BB64y2udeG+hv4KfSOb9qwcYQstTJc1KCbsix+wLZWZYN8t7nwX3GOBLRw==", 2351 | "dev": true 2352 | }, 2353 | "jest-haste-map": { 2354 | "version": "27.5.1", 2355 | "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-27.5.1.tgz", 2356 | "integrity": "sha512-7GgkZ4Fw4NFbMSDSpZwXeBiIbx+t/46nJ2QitkOjvwPYyZmqttu2TDSimMHP1EkPOi4xUZAN1doE5Vd25H4Jng==", 2357 | "dev": true, 2358 | "requires": { 2359 | "@jest/types": "^27.5.1", 2360 | "@types/graceful-fs": "^4.1.2", 2361 | "@types/node": "*", 2362 | "anymatch": "^3.0.3", 2363 | "fb-watchman": "^2.0.0", 2364 | "fsevents": "^2.3.2", 2365 | "graceful-fs": "^4.2.9", 2366 | "jest-regex-util": "^27.5.1", 2367 | "jest-serializer": "^27.5.1", 2368 | "jest-util": "^27.5.1", 2369 | "jest-worker": "^27.5.1", 2370 | "micromatch": "^4.0.4", 2371 | "walker": "^1.0.7" 2372 | } 2373 | }, 2374 | "jest-jasmine2": { 2375 | "version": "27.5.1", 2376 | "resolved": "https://registry.npmjs.org/jest-jasmine2/-/jest-jasmine2-27.5.1.tgz", 2377 | "integrity": "sha512-jtq7VVyG8SqAorDpApwiJJImd0V2wv1xzdheGHRGyuT7gZm6gG47QEskOlzsN1PG/6WNaCo5pmwMHDf3AkG2pQ==", 2378 | "dev": true, 2379 | "requires": { 2380 | "@jest/environment": "^27.5.1", 2381 | "@jest/source-map": "^27.5.1", 2382 | "@jest/test-result": "^27.5.1", 2383 | "@jest/types": "^27.5.1", 2384 | "@types/node": "*", 2385 | "chalk": "^4.0.0", 2386 | "co": "^4.6.0", 2387 | "expect": "^27.5.1", 2388 | "is-generator-fn": "^2.0.0", 2389 | "jest-each": "^27.5.1", 2390 | "jest-matcher-utils": "^27.5.1", 2391 | "jest-message-util": "^27.5.1", 2392 | "jest-runtime": "^27.5.1", 2393 | "jest-snapshot": "^27.5.1", 2394 | "jest-util": "^27.5.1", 2395 | "pretty-format": "^27.5.1", 2396 | "throat": "^6.0.1" 2397 | } 2398 | }, 2399 | "jest-leak-detector": { 2400 | "version": "27.5.1", 2401 | "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-27.5.1.tgz", 2402 | "integrity": "sha512-POXfWAMvfU6WMUXftV4HolnJfnPOGEu10fscNCA76KBpRRhcMN2c8d3iT2pxQS3HLbA+5X4sOUPzYO2NUyIlHQ==", 2403 | "dev": true, 2404 | "requires": { 2405 | "jest-get-type": "^27.5.1", 2406 | "pretty-format": "^27.5.1" 2407 | } 2408 | }, 2409 | "jest-matcher-utils": { 2410 | "version": "27.5.1", 2411 | "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-27.5.1.tgz", 2412 | "integrity": "sha512-z2uTx/T6LBaCoNWNFWwChLBKYxTMcGBRjAt+2SbP929/Fflb9aa5LGma654Rz8z9HLxsrUaYzxE9T/EFIL/PAw==", 2413 | "dev": true, 2414 | "requires": { 2415 | "chalk": "^4.0.0", 2416 | "jest-diff": "^27.5.1", 2417 | "jest-get-type": "^27.5.1", 2418 | "pretty-format": "^27.5.1" 2419 | } 2420 | }, 2421 | "jest-message-util": { 2422 | "version": "27.5.1", 2423 | "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-27.5.1.tgz", 2424 | "integrity": "sha512-rMyFe1+jnyAAf+NHwTclDz0eAaLkVDdKVHHBFWsBWHnnh5YeJMNWWsv7AbFYXfK3oTqvL7VTWkhNLu1jX24D+g==", 2425 | "dev": true, 2426 | "requires": { 2427 | "@babel/code-frame": "^7.12.13", 2428 | "@jest/types": "^27.5.1", 2429 | "@types/stack-utils": "^2.0.0", 2430 | "chalk": "^4.0.0", 2431 | "graceful-fs": "^4.2.9", 2432 | "micromatch": "^4.0.4", 2433 | "pretty-format": "^27.5.1", 2434 | "slash": "^3.0.0", 2435 | "stack-utils": "^2.0.3" 2436 | } 2437 | }, 2438 | "jest-mock": { 2439 | "version": "27.5.1", 2440 | "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-27.5.1.tgz", 2441 | "integrity": "sha512-K4jKbY1d4ENhbrG2zuPWaQBvDly+iZ2yAW+T1fATN78hc0sInwn7wZB8XtlNnvHug5RMwV897Xm4LqmPM4e2Og==", 2442 | "dev": true, 2443 | "requires": { 2444 | "@jest/types": "^27.5.1", 2445 | "@types/node": "*" 2446 | } 2447 | }, 2448 | "jest-pnp-resolver": { 2449 | "version": "1.2.2", 2450 | "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz", 2451 | "integrity": "sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w==", 2452 | "dev": true 2453 | }, 2454 | "jest-regex-util": { 2455 | "version": "27.5.1", 2456 | "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-27.5.1.tgz", 2457 | "integrity": "sha512-4bfKq2zie+x16okqDXjXn9ql2B0dScQu+vcwe4TvFVhkVyuWLqpZrZtXxLLWoXYgn0E87I6r6GRYHF7wFZBUvg==", 2458 | "dev": true 2459 | }, 2460 | "jest-resolve": { 2461 | "version": "27.5.1", 2462 | "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-27.5.1.tgz", 2463 | "integrity": "sha512-FFDy8/9E6CV83IMbDpcjOhumAQPDyETnU2KZ1O98DwTnz8AOBsW/Xv3GySr1mOZdItLR+zDZ7I/UdTFbgSOVCw==", 2464 | "dev": true, 2465 | "requires": { 2466 | "@jest/types": "^27.5.1", 2467 | "chalk": "^4.0.0", 2468 | "graceful-fs": "^4.2.9", 2469 | "jest-haste-map": "^27.5.1", 2470 | "jest-pnp-resolver": "^1.2.2", 2471 | "jest-util": "^27.5.1", 2472 | "jest-validate": "^27.5.1", 2473 | "resolve": "^1.20.0", 2474 | "resolve.exports": "^1.1.0", 2475 | "slash": "^3.0.0" 2476 | } 2477 | }, 2478 | "jest-resolve-dependencies": { 2479 | "version": "27.5.1", 2480 | "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-27.5.1.tgz", 2481 | "integrity": "sha512-QQOOdY4PE39iawDn5rzbIePNigfe5B9Z91GDD1ae/xNDlu9kaat8QQ5EKnNmVWPV54hUdxCVwwj6YMgR2O7IOg==", 2482 | "dev": true, 2483 | "requires": { 2484 | "@jest/types": "^27.5.1", 2485 | "jest-regex-util": "^27.5.1", 2486 | "jest-snapshot": "^27.5.1" 2487 | } 2488 | }, 2489 | "jest-runner": { 2490 | "version": "27.5.1", 2491 | "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-27.5.1.tgz", 2492 | "integrity": "sha512-g4NPsM4mFCOwFKXO4p/H/kWGdJp9V8kURY2lX8Me2drgXqG7rrZAx5kv+5H7wtt/cdFIjhqYx1HrlqWHaOvDaQ==", 2493 | "dev": true, 2494 | "requires": { 2495 | "@jest/console": "^27.5.1", 2496 | "@jest/environment": "^27.5.1", 2497 | "@jest/test-result": "^27.5.1", 2498 | "@jest/transform": "^27.5.1", 2499 | "@jest/types": "^27.5.1", 2500 | "@types/node": "*", 2501 | "chalk": "^4.0.0", 2502 | "emittery": "^0.8.1", 2503 | "graceful-fs": "^4.2.9", 2504 | "jest-docblock": "^27.5.1", 2505 | "jest-environment-jsdom": "^27.5.1", 2506 | "jest-environment-node": "^27.5.1", 2507 | "jest-haste-map": "^27.5.1", 2508 | "jest-leak-detector": "^27.5.1", 2509 | "jest-message-util": "^27.5.1", 2510 | "jest-resolve": "^27.5.1", 2511 | "jest-runtime": "^27.5.1", 2512 | "jest-util": "^27.5.1", 2513 | "jest-worker": "^27.5.1", 2514 | "source-map-support": "^0.5.6", 2515 | "throat": "^6.0.1" 2516 | } 2517 | }, 2518 | "jest-runtime": { 2519 | "version": "27.5.1", 2520 | "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-27.5.1.tgz", 2521 | "integrity": "sha512-o7gxw3Gf+H2IGt8fv0RiyE1+r83FJBRruoA+FXrlHw6xEyBsU8ugA6IPfTdVyA0w8HClpbK+DGJxH59UrNMx8A==", 2522 | "dev": true, 2523 | "requires": { 2524 | "@jest/environment": "^27.5.1", 2525 | "@jest/fake-timers": "^27.5.1", 2526 | "@jest/globals": "^27.5.1", 2527 | "@jest/source-map": "^27.5.1", 2528 | "@jest/test-result": "^27.5.1", 2529 | "@jest/transform": "^27.5.1", 2530 | "@jest/types": "^27.5.1", 2531 | "chalk": "^4.0.0", 2532 | "cjs-module-lexer": "^1.0.0", 2533 | "collect-v8-coverage": "^1.0.0", 2534 | "execa": "^5.0.0", 2535 | "glob": "^7.1.3", 2536 | "graceful-fs": "^4.2.9", 2537 | "jest-haste-map": "^27.5.1", 2538 | "jest-message-util": "^27.5.1", 2539 | "jest-mock": "^27.5.1", 2540 | "jest-regex-util": "^27.5.1", 2541 | "jest-resolve": "^27.5.1", 2542 | "jest-snapshot": "^27.5.1", 2543 | "jest-util": "^27.5.1", 2544 | "slash": "^3.0.0", 2545 | "strip-bom": "^4.0.0" 2546 | } 2547 | }, 2548 | "jest-serializer": { 2549 | "version": "27.5.1", 2550 | "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-27.5.1.tgz", 2551 | "integrity": "sha512-jZCyo6iIxO1aqUxpuBlwTDMkzOAJS4a3eYz3YzgxxVQFwLeSA7Jfq5cbqCY+JLvTDrWirgusI/0KwxKMgrdf7w==", 2552 | "dev": true, 2553 | "requires": { 2554 | "@types/node": "*", 2555 | "graceful-fs": "^4.2.9" 2556 | } 2557 | }, 2558 | "jest-snapshot": { 2559 | "version": "27.5.1", 2560 | "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-27.5.1.tgz", 2561 | "integrity": "sha512-yYykXI5a0I31xX67mgeLw1DZ0bJB+gpq5IpSuCAoyDi0+BhgU/RIrL+RTzDmkNTchvDFWKP8lp+w/42Z3us5sA==", 2562 | "dev": true, 2563 | "requires": { 2564 | "@babel/core": "^7.7.2", 2565 | "@babel/generator": "^7.7.2", 2566 | "@babel/plugin-syntax-typescript": "^7.7.2", 2567 | "@babel/traverse": "^7.7.2", 2568 | "@babel/types": "^7.0.0", 2569 | "@jest/transform": "^27.5.1", 2570 | "@jest/types": "^27.5.1", 2571 | "@types/babel__traverse": "^7.0.4", 2572 | "@types/prettier": "^2.1.5", 2573 | "babel-preset-current-node-syntax": "^1.0.0", 2574 | "chalk": "^4.0.0", 2575 | "expect": "^27.5.1", 2576 | "graceful-fs": "^4.2.9", 2577 | "jest-diff": "^27.5.1", 2578 | "jest-get-type": "^27.5.1", 2579 | "jest-haste-map": "^27.5.1", 2580 | "jest-matcher-utils": "^27.5.1", 2581 | "jest-message-util": "^27.5.1", 2582 | "jest-util": "^27.5.1", 2583 | "natural-compare": "^1.4.0", 2584 | "pretty-format": "^27.5.1", 2585 | "semver": "^7.3.2" 2586 | }, 2587 | "dependencies": { 2588 | "semver": { 2589 | "version": "7.3.7", 2590 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", 2591 | "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", 2592 | "dev": true, 2593 | "requires": { 2594 | "lru-cache": "^6.0.0" 2595 | } 2596 | } 2597 | } 2598 | }, 2599 | "jest-util": { 2600 | "version": "27.5.1", 2601 | "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-27.5.1.tgz", 2602 | "integrity": "sha512-Kv2o/8jNvX1MQ0KGtw480E/w4fBCDOnH6+6DmeKi6LZUIlKA5kwY0YNdlzaWTiVgxqAqik11QyxDOKk543aKXw==", 2603 | "dev": true, 2604 | "requires": { 2605 | "@jest/types": "^27.5.1", 2606 | "@types/node": "*", 2607 | "chalk": "^4.0.0", 2608 | "ci-info": "^3.2.0", 2609 | "graceful-fs": "^4.2.9", 2610 | "picomatch": "^2.2.3" 2611 | } 2612 | }, 2613 | "jest-validate": { 2614 | "version": "27.5.1", 2615 | "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-27.5.1.tgz", 2616 | "integrity": "sha512-thkNli0LYTmOI1tDB3FI1S1RTp/Bqyd9pTarJwL87OIBFuqEb5Apv5EaApEudYg4g86e3CT6kM0RowkhtEnCBQ==", 2617 | "dev": true, 2618 | "requires": { 2619 | "@jest/types": "^27.5.1", 2620 | "camelcase": "^6.2.0", 2621 | "chalk": "^4.0.0", 2622 | "jest-get-type": "^27.5.1", 2623 | "leven": "^3.1.0", 2624 | "pretty-format": "^27.5.1" 2625 | }, 2626 | "dependencies": { 2627 | "camelcase": { 2628 | "version": "6.3.0", 2629 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", 2630 | "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", 2631 | "dev": true 2632 | } 2633 | } 2634 | }, 2635 | "jest-watcher": { 2636 | "version": "27.5.1", 2637 | "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-27.5.1.tgz", 2638 | "integrity": "sha512-z676SuD6Z8o8qbmEGhoEUFOM1+jfEiL3DXHK/xgEiG2EyNYfFG60jluWcupY6dATjfEsKQuibReS1djInQnoVw==", 2639 | "dev": true, 2640 | "requires": { 2641 | "@jest/test-result": "^27.5.1", 2642 | "@jest/types": "^27.5.1", 2643 | "@types/node": "*", 2644 | "ansi-escapes": "^4.2.1", 2645 | "chalk": "^4.0.0", 2646 | "jest-util": "^27.5.1", 2647 | "string-length": "^4.0.1" 2648 | } 2649 | }, 2650 | "jest-worker": { 2651 | "version": "27.5.1", 2652 | "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", 2653 | "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", 2654 | "dev": true, 2655 | "requires": { 2656 | "@types/node": "*", 2657 | "merge-stream": "^2.0.0", 2658 | "supports-color": "^8.0.0" 2659 | }, 2660 | "dependencies": { 2661 | "supports-color": { 2662 | "version": "8.1.1", 2663 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", 2664 | "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", 2665 | "dev": true, 2666 | "requires": { 2667 | "has-flag": "^4.0.0" 2668 | } 2669 | } 2670 | } 2671 | }, 2672 | "js-tokens": { 2673 | "version": "4.0.0", 2674 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 2675 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 2676 | "dev": true 2677 | }, 2678 | "js-yaml": { 2679 | "version": "3.14.1", 2680 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", 2681 | "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", 2682 | "dev": true, 2683 | "requires": { 2684 | "argparse": "^1.0.7", 2685 | "esprima": "^4.0.0" 2686 | } 2687 | }, 2688 | "jsdom": { 2689 | "version": "16.7.0", 2690 | "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-16.7.0.tgz", 2691 | "integrity": "sha512-u9Smc2G1USStM+s/x1ru5Sxrl6mPYCbByG1U/hUmqaVsm4tbNyS7CicOSRyuGQYZhTu0h84qkZZQ/I+dzizSVw==", 2692 | "dev": true, 2693 | "requires": { 2694 | "abab": "^2.0.5", 2695 | "acorn": "^8.2.4", 2696 | "acorn-globals": "^6.0.0", 2697 | "cssom": "^0.4.4", 2698 | "cssstyle": "^2.3.0", 2699 | "data-urls": "^2.0.0", 2700 | "decimal.js": "^10.2.1", 2701 | "domexception": "^2.0.1", 2702 | "escodegen": "^2.0.0", 2703 | "form-data": "^3.0.0", 2704 | "html-encoding-sniffer": "^2.0.1", 2705 | "http-proxy-agent": "^4.0.1", 2706 | "https-proxy-agent": "^5.0.0", 2707 | "is-potential-custom-element-name": "^1.0.1", 2708 | "nwsapi": "^2.2.0", 2709 | "parse5": "6.0.1", 2710 | "saxes": "^5.0.1", 2711 | "symbol-tree": "^3.2.4", 2712 | "tough-cookie": "^4.0.0", 2713 | "w3c-hr-time": "^1.0.2", 2714 | "w3c-xmlserializer": "^2.0.0", 2715 | "webidl-conversions": "^6.1.0", 2716 | "whatwg-encoding": "^1.0.5", 2717 | "whatwg-mimetype": "^2.3.0", 2718 | "whatwg-url": "^8.5.0", 2719 | "ws": "^7.4.6", 2720 | "xml-name-validator": "^3.0.0" 2721 | } 2722 | }, 2723 | "jsesc": { 2724 | "version": "2.5.2", 2725 | "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", 2726 | "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", 2727 | "dev": true 2728 | }, 2729 | "json-parse-even-better-errors": { 2730 | "version": "2.3.1", 2731 | "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", 2732 | "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", 2733 | "dev": true 2734 | }, 2735 | "json5": { 2736 | "version": "2.2.3", 2737 | "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", 2738 | "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", 2739 | "dev": true 2740 | }, 2741 | "kleur": { 2742 | "version": "3.0.3", 2743 | "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", 2744 | "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", 2745 | "dev": true 2746 | }, 2747 | "leven": { 2748 | "version": "3.1.0", 2749 | "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", 2750 | "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", 2751 | "dev": true 2752 | }, 2753 | "levn": { 2754 | "version": "0.3.0", 2755 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", 2756 | "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==", 2757 | "dev": true, 2758 | "requires": { 2759 | "prelude-ls": "~1.1.2", 2760 | "type-check": "~0.3.2" 2761 | } 2762 | }, 2763 | "lines-and-columns": { 2764 | "version": "1.2.4", 2765 | "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", 2766 | "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", 2767 | "dev": true 2768 | }, 2769 | "locate-path": { 2770 | "version": "5.0.0", 2771 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 2772 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 2773 | "dev": true, 2774 | "requires": { 2775 | "p-locate": "^4.1.0" 2776 | } 2777 | }, 2778 | "lodash": { 2779 | "version": "4.17.21", 2780 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 2781 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 2782 | "dev": true 2783 | }, 2784 | "lodash.memoize": { 2785 | "version": "4.1.2", 2786 | "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", 2787 | "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", 2788 | "dev": true 2789 | }, 2790 | "lru-cache": { 2791 | "version": "6.0.0", 2792 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 2793 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 2794 | "dev": true, 2795 | "requires": { 2796 | "yallist": "^4.0.0" 2797 | } 2798 | }, 2799 | "make-dir": { 2800 | "version": "3.1.0", 2801 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 2802 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 2803 | "dev": true, 2804 | "requires": { 2805 | "semver": "^6.0.0" 2806 | } 2807 | }, 2808 | "make-error": { 2809 | "version": "1.3.6", 2810 | "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", 2811 | "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", 2812 | "dev": true 2813 | }, 2814 | "makeerror": { 2815 | "version": "1.0.12", 2816 | "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", 2817 | "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", 2818 | "dev": true, 2819 | "requires": { 2820 | "tmpl": "1.0.5" 2821 | } 2822 | }, 2823 | "merge-stream": { 2824 | "version": "2.0.0", 2825 | "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", 2826 | "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", 2827 | "dev": true 2828 | }, 2829 | "micromatch": { 2830 | "version": "4.0.5", 2831 | "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", 2832 | "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", 2833 | "dev": true, 2834 | "requires": { 2835 | "braces": "^3.0.2", 2836 | "picomatch": "^2.3.1" 2837 | } 2838 | }, 2839 | "mime-db": { 2840 | "version": "1.52.0", 2841 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 2842 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 2843 | "dev": true 2844 | }, 2845 | "mime-types": { 2846 | "version": "2.1.35", 2847 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 2848 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 2849 | "dev": true, 2850 | "requires": { 2851 | "mime-db": "1.52.0" 2852 | } 2853 | }, 2854 | "mimic-fn": { 2855 | "version": "2.1.0", 2856 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", 2857 | "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", 2858 | "dev": true 2859 | }, 2860 | "minimatch": { 2861 | "version": "3.1.2", 2862 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 2863 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 2864 | "dev": true, 2865 | "requires": { 2866 | "brace-expansion": "^1.1.7" 2867 | } 2868 | }, 2869 | "ms": { 2870 | "version": "2.1.2", 2871 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2872 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 2873 | "dev": true 2874 | }, 2875 | "natural-compare": { 2876 | "version": "1.4.0", 2877 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 2878 | "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", 2879 | "dev": true 2880 | }, 2881 | "node-int64": { 2882 | "version": "0.4.0", 2883 | "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", 2884 | "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", 2885 | "dev": true 2886 | }, 2887 | "node-releases": { 2888 | "version": "2.0.5", 2889 | "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.5.tgz", 2890 | "integrity": "sha512-U9h1NLROZTq9uE1SNffn6WuPDg8icmi3ns4rEl/oTfIle4iLjTliCzgTsbaIFMq/Xn078/lfY/BL0GWZ+psK4Q==", 2891 | "dev": true 2892 | }, 2893 | "normalize-path": { 2894 | "version": "3.0.0", 2895 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 2896 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 2897 | "dev": true 2898 | }, 2899 | "npm-run-path": { 2900 | "version": "4.0.1", 2901 | "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", 2902 | "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", 2903 | "dev": true, 2904 | "requires": { 2905 | "path-key": "^3.0.0" 2906 | } 2907 | }, 2908 | "nwsapi": { 2909 | "version": "2.2.0", 2910 | "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.0.tgz", 2911 | "integrity": "sha512-h2AatdwYH+JHiZpv7pt/gSX1XoRGb7L/qSIeuqA6GwYoF9w1vP1cw42TO0aI2pNyshRK5893hNSl+1//vHK7hQ==", 2912 | "dev": true 2913 | }, 2914 | "once": { 2915 | "version": "1.4.0", 2916 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 2917 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 2918 | "dev": true, 2919 | "requires": { 2920 | "wrappy": "1" 2921 | } 2922 | }, 2923 | "onetime": { 2924 | "version": "5.1.2", 2925 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", 2926 | "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", 2927 | "dev": true, 2928 | "requires": { 2929 | "mimic-fn": "^2.1.0" 2930 | } 2931 | }, 2932 | "optionator": { 2933 | "version": "0.8.3", 2934 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", 2935 | "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", 2936 | "dev": true, 2937 | "requires": { 2938 | "deep-is": "~0.1.3", 2939 | "fast-levenshtein": "~2.0.6", 2940 | "levn": "~0.3.0", 2941 | "prelude-ls": "~1.1.2", 2942 | "type-check": "~0.3.2", 2943 | "word-wrap": "~1.2.3" 2944 | } 2945 | }, 2946 | "p-limit": { 2947 | "version": "2.3.0", 2948 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 2949 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 2950 | "dev": true, 2951 | "requires": { 2952 | "p-try": "^2.0.0" 2953 | } 2954 | }, 2955 | "p-locate": { 2956 | "version": "4.1.0", 2957 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 2958 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 2959 | "dev": true, 2960 | "requires": { 2961 | "p-limit": "^2.2.0" 2962 | } 2963 | }, 2964 | "p-try": { 2965 | "version": "2.2.0", 2966 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 2967 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 2968 | "dev": true 2969 | }, 2970 | "parse-json": { 2971 | "version": "5.2.0", 2972 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", 2973 | "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", 2974 | "dev": true, 2975 | "requires": { 2976 | "@babel/code-frame": "^7.0.0", 2977 | "error-ex": "^1.3.1", 2978 | "json-parse-even-better-errors": "^2.3.0", 2979 | "lines-and-columns": "^1.1.6" 2980 | } 2981 | }, 2982 | "parse5": { 2983 | "version": "6.0.1", 2984 | "resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz", 2985 | "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==", 2986 | "dev": true 2987 | }, 2988 | "path-exists": { 2989 | "version": "4.0.0", 2990 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 2991 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 2992 | "dev": true 2993 | }, 2994 | "path-is-absolute": { 2995 | "version": "1.0.1", 2996 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 2997 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 2998 | "dev": true 2999 | }, 3000 | "path-key": { 3001 | "version": "3.1.1", 3002 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 3003 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 3004 | "dev": true 3005 | }, 3006 | "path-parse": { 3007 | "version": "1.0.7", 3008 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 3009 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 3010 | "dev": true 3011 | }, 3012 | "picocolors": { 3013 | "version": "1.0.0", 3014 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", 3015 | "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", 3016 | "dev": true 3017 | }, 3018 | "picomatch": { 3019 | "version": "2.3.1", 3020 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 3021 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 3022 | "dev": true 3023 | }, 3024 | "pirates": { 3025 | "version": "4.0.5", 3026 | "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.5.tgz", 3027 | "integrity": "sha512-8V9+HQPupnaXMA23c5hvl69zXvTwTzyAYasnkb0Tts4XvO4CliqONMOnvlq26rkhLC3nWDFBJf73LU1e1VZLaQ==", 3028 | "dev": true 3029 | }, 3030 | "pkg-dir": { 3031 | "version": "4.2.0", 3032 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", 3033 | "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", 3034 | "dev": true, 3035 | "requires": { 3036 | "find-up": "^4.0.0" 3037 | } 3038 | }, 3039 | "prelude-ls": { 3040 | "version": "1.1.2", 3041 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", 3042 | "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==", 3043 | "dev": true 3044 | }, 3045 | "pretty-format": { 3046 | "version": "27.5.1", 3047 | "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", 3048 | "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", 3049 | "dev": true, 3050 | "requires": { 3051 | "ansi-regex": "^5.0.1", 3052 | "ansi-styles": "^5.0.0", 3053 | "react-is": "^17.0.1" 3054 | }, 3055 | "dependencies": { 3056 | "ansi-styles": { 3057 | "version": "5.2.0", 3058 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", 3059 | "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", 3060 | "dev": true 3061 | } 3062 | } 3063 | }, 3064 | "prompts": { 3065 | "version": "2.4.2", 3066 | "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", 3067 | "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", 3068 | "dev": true, 3069 | "requires": { 3070 | "kleur": "^3.0.3", 3071 | "sisteransi": "^1.0.5" 3072 | } 3073 | }, 3074 | "psl": { 3075 | "version": "1.8.0", 3076 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", 3077 | "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", 3078 | "dev": true 3079 | }, 3080 | "punycode": { 3081 | "version": "2.1.1", 3082 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 3083 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 3084 | "dev": true 3085 | }, 3086 | "querystringify": { 3087 | "version": "2.2.0", 3088 | "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz", 3089 | "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==", 3090 | "dev": true 3091 | }, 3092 | "react-is": { 3093 | "version": "17.0.2", 3094 | "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", 3095 | "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", 3096 | "dev": true 3097 | }, 3098 | "require-directory": { 3099 | "version": "2.1.1", 3100 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 3101 | "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", 3102 | "dev": true 3103 | }, 3104 | "requires-port": { 3105 | "version": "1.0.0", 3106 | "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", 3107 | "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==", 3108 | "dev": true 3109 | }, 3110 | "resolve": { 3111 | "version": "1.22.0", 3112 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.0.tgz", 3113 | "integrity": "sha512-Hhtrw0nLeSrFQ7phPp4OOcVjLPIeMnRlr5mcnVuMe7M/7eBn98A3hmFRLoFo3DLZkivSYwhRUJTyPyWAk56WLw==", 3114 | "dev": true, 3115 | "requires": { 3116 | "is-core-module": "^2.8.1", 3117 | "path-parse": "^1.0.7", 3118 | "supports-preserve-symlinks-flag": "^1.0.0" 3119 | } 3120 | }, 3121 | "resolve-cwd": { 3122 | "version": "3.0.0", 3123 | "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", 3124 | "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", 3125 | "dev": true, 3126 | "requires": { 3127 | "resolve-from": "^5.0.0" 3128 | } 3129 | }, 3130 | "resolve-from": { 3131 | "version": "5.0.0", 3132 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", 3133 | "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", 3134 | "dev": true 3135 | }, 3136 | "resolve.exports": { 3137 | "version": "1.1.0", 3138 | "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-1.1.0.tgz", 3139 | "integrity": "sha512-J1l+Zxxp4XK3LUDZ9m60LRJF/mAe4z6a4xyabPHk7pvK5t35dACV32iIjJDFeWZFfZlO29w6SZ67knR0tHzJtQ==", 3140 | "dev": true 3141 | }, 3142 | "rimraf": { 3143 | "version": "3.0.2", 3144 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 3145 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 3146 | "dev": true, 3147 | "requires": { 3148 | "glob": "^7.1.3" 3149 | } 3150 | }, 3151 | "safe-buffer": { 3152 | "version": "5.1.2", 3153 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 3154 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 3155 | "dev": true 3156 | }, 3157 | "safer-buffer": { 3158 | "version": "2.1.2", 3159 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 3160 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", 3161 | "dev": true 3162 | }, 3163 | "saxes": { 3164 | "version": "5.0.1", 3165 | "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", 3166 | "integrity": "sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw==", 3167 | "dev": true, 3168 | "requires": { 3169 | "xmlchars": "^2.2.0" 3170 | } 3171 | }, 3172 | "semver": { 3173 | "version": "6.3.0", 3174 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 3175 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 3176 | "dev": true 3177 | }, 3178 | "shebang-command": { 3179 | "version": "2.0.0", 3180 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 3181 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 3182 | "dev": true, 3183 | "requires": { 3184 | "shebang-regex": "^3.0.0" 3185 | } 3186 | }, 3187 | "shebang-regex": { 3188 | "version": "3.0.0", 3189 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 3190 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 3191 | "dev": true 3192 | }, 3193 | "signal-exit": { 3194 | "version": "3.0.7", 3195 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", 3196 | "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", 3197 | "dev": true 3198 | }, 3199 | "sisteransi": { 3200 | "version": "1.0.5", 3201 | "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", 3202 | "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", 3203 | "dev": true 3204 | }, 3205 | "slash": { 3206 | "version": "3.0.0", 3207 | "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", 3208 | "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", 3209 | "dev": true 3210 | }, 3211 | "source-map": { 3212 | "version": "0.6.1", 3213 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 3214 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" 3215 | }, 3216 | "source-map-support": { 3217 | "version": "0.5.21", 3218 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", 3219 | "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", 3220 | "requires": { 3221 | "buffer-from": "^1.0.0", 3222 | "source-map": "^0.6.0" 3223 | } 3224 | }, 3225 | "sprintf-js": { 3226 | "version": "1.0.3", 3227 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 3228 | "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", 3229 | "dev": true 3230 | }, 3231 | "stack-utils": { 3232 | "version": "2.0.5", 3233 | "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.5.tgz", 3234 | "integrity": "sha512-xrQcmYhOsn/1kX+Vraq+7j4oE2j/6BFscZ0etmYg81xuM8Gq0022Pxb8+IqgOFUIaxHs0KaSb7T1+OegiNrNFA==", 3235 | "dev": true, 3236 | "requires": { 3237 | "escape-string-regexp": "^2.0.0" 3238 | }, 3239 | "dependencies": { 3240 | "escape-string-regexp": { 3241 | "version": "2.0.0", 3242 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", 3243 | "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", 3244 | "dev": true 3245 | } 3246 | } 3247 | }, 3248 | "string-length": { 3249 | "version": "4.0.2", 3250 | "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", 3251 | "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", 3252 | "dev": true, 3253 | "requires": { 3254 | "char-regex": "^1.0.2", 3255 | "strip-ansi": "^6.0.0" 3256 | } 3257 | }, 3258 | "string-width": { 3259 | "version": "4.2.3", 3260 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 3261 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 3262 | "dev": true, 3263 | "requires": { 3264 | "emoji-regex": "^8.0.0", 3265 | "is-fullwidth-code-point": "^3.0.0", 3266 | "strip-ansi": "^6.0.1" 3267 | } 3268 | }, 3269 | "strip-ansi": { 3270 | "version": "6.0.1", 3271 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 3272 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 3273 | "dev": true, 3274 | "requires": { 3275 | "ansi-regex": "^5.0.1" 3276 | } 3277 | }, 3278 | "strip-bom": { 3279 | "version": "4.0.0", 3280 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", 3281 | "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", 3282 | "dev": true 3283 | }, 3284 | "strip-final-newline": { 3285 | "version": "2.0.0", 3286 | "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", 3287 | "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", 3288 | "dev": true 3289 | }, 3290 | "strip-json-comments": { 3291 | "version": "3.1.1", 3292 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 3293 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 3294 | "dev": true 3295 | }, 3296 | "supports-color": { 3297 | "version": "7.2.0", 3298 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 3299 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 3300 | "dev": true, 3301 | "requires": { 3302 | "has-flag": "^4.0.0" 3303 | } 3304 | }, 3305 | "supports-hyperlinks": { 3306 | "version": "2.2.0", 3307 | "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", 3308 | "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", 3309 | "dev": true, 3310 | "requires": { 3311 | "has-flag": "^4.0.0", 3312 | "supports-color": "^7.0.0" 3313 | } 3314 | }, 3315 | "supports-preserve-symlinks-flag": { 3316 | "version": "1.0.0", 3317 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 3318 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 3319 | "dev": true 3320 | }, 3321 | "symbol-tree": { 3322 | "version": "3.2.4", 3323 | "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", 3324 | "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", 3325 | "dev": true 3326 | }, 3327 | "terminal-link": { 3328 | "version": "2.1.1", 3329 | "resolved": "https://registry.npmjs.org/terminal-link/-/terminal-link-2.1.1.tgz", 3330 | "integrity": "sha512-un0FmiRUQNr5PJqy9kP7c40F5BOfpGlYTrxonDChEZB7pzZxRNp/bt+ymiy9/npwXya9KH99nJ/GXFIiUkYGFQ==", 3331 | "dev": true, 3332 | "requires": { 3333 | "ansi-escapes": "^4.2.1", 3334 | "supports-hyperlinks": "^2.0.0" 3335 | } 3336 | }, 3337 | "test-exclude": { 3338 | "version": "6.0.0", 3339 | "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", 3340 | "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", 3341 | "dev": true, 3342 | "requires": { 3343 | "@istanbuljs/schema": "^0.1.2", 3344 | "glob": "^7.1.4", 3345 | "minimatch": "^3.0.4" 3346 | } 3347 | }, 3348 | "throat": { 3349 | "version": "6.0.1", 3350 | "resolved": "https://registry.npmjs.org/throat/-/throat-6.0.1.tgz", 3351 | "integrity": "sha512-8hmiGIJMDlwjg7dlJ4yKGLK8EsYqKgPWbG3b4wjJddKNwc7N7Dpn08Df4szr/sZdMVeOstrdYSsqzX6BYbcB+w==", 3352 | "dev": true 3353 | }, 3354 | "tmpl": { 3355 | "version": "1.0.5", 3356 | "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", 3357 | "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==", 3358 | "dev": true 3359 | }, 3360 | "to-fast-properties": { 3361 | "version": "2.0.0", 3362 | "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", 3363 | "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", 3364 | "dev": true 3365 | }, 3366 | "to-regex-range": { 3367 | "version": "5.0.1", 3368 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 3369 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 3370 | "dev": true, 3371 | "requires": { 3372 | "is-number": "^7.0.0" 3373 | } 3374 | }, 3375 | "tough-cookie": { 3376 | "version": "4.1.3", 3377 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.1.3.tgz", 3378 | "integrity": "sha512-aX/y5pVRkfRnfmuX+OdbSdXvPe6ieKX/G2s7e98f4poJHnqH3281gDPm/metm6E/WRamfx7WC4HUqkWHfQHprw==", 3379 | "dev": true, 3380 | "requires": { 3381 | "psl": "^1.1.33", 3382 | "punycode": "^2.1.1", 3383 | "universalify": "^0.2.0", 3384 | "url-parse": "^1.5.3" 3385 | }, 3386 | "dependencies": { 3387 | "universalify": { 3388 | "version": "0.2.0", 3389 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.2.0.tgz", 3390 | "integrity": "sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==", 3391 | "dev": true 3392 | } 3393 | } 3394 | }, 3395 | "tr46": { 3396 | "version": "2.1.0", 3397 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-2.1.0.tgz", 3398 | "integrity": "sha512-15Ih7phfcdP5YxqiB+iDtLoaTz4Nd35+IiAv0kQ5FNKHzXgdWqPoTIqEDDJmXceQt4JZk6lVPT8lnDlPpGDppw==", 3399 | "dev": true, 3400 | "requires": { 3401 | "punycode": "^2.1.1" 3402 | } 3403 | }, 3404 | "ts-jest": { 3405 | "version": "27.1.5", 3406 | "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-27.1.5.tgz", 3407 | "integrity": "sha512-Xv6jBQPoBEvBq/5i2TeSG9tt/nqkbpcurrEG1b+2yfBrcJelOZF9Ml6dmyMh7bcW9JyFbRYpR5rxROSlBLTZHA==", 3408 | "dev": true, 3409 | "requires": { 3410 | "bs-logger": "0.x", 3411 | "fast-json-stable-stringify": "2.x", 3412 | "jest-util": "^27.0.0", 3413 | "json5": "2.x", 3414 | "lodash.memoize": "4.x", 3415 | "make-error": "1.x", 3416 | "semver": "7.x", 3417 | "yargs-parser": "20.x" 3418 | }, 3419 | "dependencies": { 3420 | "semver": { 3421 | "version": "7.3.7", 3422 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", 3423 | "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", 3424 | "dev": true, 3425 | "requires": { 3426 | "lru-cache": "^6.0.0" 3427 | } 3428 | } 3429 | } 3430 | }, 3431 | "ts-node": { 3432 | "version": "10.8.1", 3433 | "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.8.1.tgz", 3434 | "integrity": "sha512-Wwsnao4DQoJsN034wePSg5nZiw4YKXf56mPIAeD6wVmiv+RytNSWqc2f3fKvcUoV+Yn2+yocD71VOfQHbmVX4g==", 3435 | "dev": true, 3436 | "requires": { 3437 | "@cspotcode/source-map-support": "^0.8.0", 3438 | "@tsconfig/node10": "^1.0.7", 3439 | "@tsconfig/node12": "^1.0.7", 3440 | "@tsconfig/node14": "^1.0.0", 3441 | "@tsconfig/node16": "^1.0.2", 3442 | "acorn": "^8.4.1", 3443 | "acorn-walk": "^8.1.1", 3444 | "arg": "^4.1.0", 3445 | "create-require": "^1.1.0", 3446 | "diff": "^4.0.1", 3447 | "make-error": "^1.1.1", 3448 | "v8-compile-cache-lib": "^3.0.1", 3449 | "yn": "3.1.1" 3450 | }, 3451 | "dependencies": { 3452 | "acorn-walk": { 3453 | "version": "8.2.0", 3454 | "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", 3455 | "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", 3456 | "dev": true 3457 | } 3458 | } 3459 | }, 3460 | "type-check": { 3461 | "version": "0.3.2", 3462 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", 3463 | "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", 3464 | "dev": true, 3465 | "requires": { 3466 | "prelude-ls": "~1.1.2" 3467 | } 3468 | }, 3469 | "type-detect": { 3470 | "version": "4.0.8", 3471 | "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", 3472 | "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", 3473 | "dev": true 3474 | }, 3475 | "type-fest": { 3476 | "version": "0.21.3", 3477 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", 3478 | "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", 3479 | "dev": true 3480 | }, 3481 | "typedarray-to-buffer": { 3482 | "version": "3.1.5", 3483 | "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", 3484 | "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", 3485 | "dev": true, 3486 | "requires": { 3487 | "is-typedarray": "^1.0.0" 3488 | } 3489 | }, 3490 | "typescript": { 3491 | "version": "3.9.10", 3492 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.10.tgz", 3493 | "integrity": "sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q==", 3494 | "dev": true 3495 | }, 3496 | "url-parse": { 3497 | "version": "1.5.10", 3498 | "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz", 3499 | "integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==", 3500 | "dev": true, 3501 | "requires": { 3502 | "querystringify": "^2.1.1", 3503 | "requires-port": "^1.0.0" 3504 | } 3505 | }, 3506 | "v8-compile-cache-lib": { 3507 | "version": "3.0.1", 3508 | "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", 3509 | "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", 3510 | "dev": true 3511 | }, 3512 | "v8-to-istanbul": { 3513 | "version": "8.1.1", 3514 | "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-8.1.1.tgz", 3515 | "integrity": "sha512-FGtKtv3xIpR6BYhvgH8MI/y78oT7d8Au3ww4QIxymrCtZEh5b8gCw2siywE+puhEmuWKDtmfrvF5UlB298ut3w==", 3516 | "dev": true, 3517 | "requires": { 3518 | "@types/istanbul-lib-coverage": "^2.0.1", 3519 | "convert-source-map": "^1.6.0", 3520 | "source-map": "^0.7.3" 3521 | }, 3522 | "dependencies": { 3523 | "source-map": { 3524 | "version": "0.7.4", 3525 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", 3526 | "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==", 3527 | "dev": true 3528 | } 3529 | } 3530 | }, 3531 | "w3c-hr-time": { 3532 | "version": "1.0.2", 3533 | "resolved": "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz", 3534 | "integrity": "sha512-z8P5DvDNjKDoFIHK7q8r8lackT6l+jo/Ye3HOle7l9nICP9lf1Ci25fy9vHd0JOWewkIFzXIEig3TdKT7JQ5fQ==", 3535 | "dev": true, 3536 | "requires": { 3537 | "browser-process-hrtime": "^1.0.0" 3538 | } 3539 | }, 3540 | "w3c-xmlserializer": { 3541 | "version": "2.0.0", 3542 | "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-2.0.0.tgz", 3543 | "integrity": "sha512-4tzD0mF8iSiMiNs30BiLO3EpfGLZUT2MSX/G+o7ZywDzliWQ3OPtTZ0PTC3B3ca1UAf4cJMHB+2Bf56EriJuRA==", 3544 | "dev": true, 3545 | "requires": { 3546 | "xml-name-validator": "^3.0.0" 3547 | } 3548 | }, 3549 | "walker": { 3550 | "version": "1.0.8", 3551 | "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", 3552 | "integrity": "sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==", 3553 | "dev": true, 3554 | "requires": { 3555 | "makeerror": "1.0.12" 3556 | } 3557 | }, 3558 | "webidl-conversions": { 3559 | "version": "6.1.0", 3560 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz", 3561 | "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==", 3562 | "dev": true 3563 | }, 3564 | "whatwg-encoding": { 3565 | "version": "1.0.5", 3566 | "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz", 3567 | "integrity": "sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw==", 3568 | "dev": true, 3569 | "requires": { 3570 | "iconv-lite": "0.4.24" 3571 | } 3572 | }, 3573 | "whatwg-mimetype": { 3574 | "version": "2.3.0", 3575 | "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz", 3576 | "integrity": "sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==", 3577 | "dev": true 3578 | }, 3579 | "whatwg-url": { 3580 | "version": "8.7.0", 3581 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-8.7.0.tgz", 3582 | "integrity": "sha512-gAojqb/m9Q8a5IV96E3fHJM70AzCkgt4uXYX2O7EmuyOnLrViCQlsEBmF9UQIu3/aeAIp2U17rtbpZWNntQqdg==", 3583 | "dev": true, 3584 | "requires": { 3585 | "lodash": "^4.7.0", 3586 | "tr46": "^2.1.0", 3587 | "webidl-conversions": "^6.1.0" 3588 | } 3589 | }, 3590 | "which": { 3591 | "version": "2.0.2", 3592 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 3593 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 3594 | "dev": true, 3595 | "requires": { 3596 | "isexe": "^2.0.0" 3597 | } 3598 | }, 3599 | "word-wrap": { 3600 | "version": "1.2.4", 3601 | "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.4.tgz", 3602 | "integrity": "sha512-2V81OA4ugVo5pRo46hAoD2ivUJx8jXmWXfUkY4KFNw0hEptvN0QfH3K4nHiwzGeKl5rFKedV48QVoqYavy4YpA==", 3603 | "dev": true 3604 | }, 3605 | "wrap-ansi": { 3606 | "version": "7.0.0", 3607 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 3608 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 3609 | "dev": true, 3610 | "requires": { 3611 | "ansi-styles": "^4.0.0", 3612 | "string-width": "^4.1.0", 3613 | "strip-ansi": "^6.0.0" 3614 | } 3615 | }, 3616 | "wrappy": { 3617 | "version": "1.0.2", 3618 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 3619 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 3620 | "dev": true 3621 | }, 3622 | "write-file-atomic": { 3623 | "version": "3.0.3", 3624 | "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", 3625 | "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", 3626 | "dev": true, 3627 | "requires": { 3628 | "imurmurhash": "^0.1.4", 3629 | "is-typedarray": "^1.0.0", 3630 | "signal-exit": "^3.0.2", 3631 | "typedarray-to-buffer": "^3.1.5" 3632 | } 3633 | }, 3634 | "ws": { 3635 | "version": "7.5.10", 3636 | "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.10.tgz", 3637 | "integrity": "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ==", 3638 | "dev": true 3639 | }, 3640 | "xml-name-validator": { 3641 | "version": "3.0.0", 3642 | "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", 3643 | "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", 3644 | "dev": true 3645 | }, 3646 | "xmlchars": { 3647 | "version": "2.2.0", 3648 | "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", 3649 | "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", 3650 | "dev": true 3651 | }, 3652 | "y18n": { 3653 | "version": "5.0.8", 3654 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 3655 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 3656 | "dev": true 3657 | }, 3658 | "yallist": { 3659 | "version": "4.0.0", 3660 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 3661 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 3662 | "dev": true 3663 | }, 3664 | "yargs": { 3665 | "version": "16.2.0", 3666 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", 3667 | "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", 3668 | "dev": true, 3669 | "requires": { 3670 | "cliui": "^7.0.2", 3671 | "escalade": "^3.1.1", 3672 | "get-caller-file": "^2.0.5", 3673 | "require-directory": "^2.1.1", 3674 | "string-width": "^4.2.0", 3675 | "y18n": "^5.0.5", 3676 | "yargs-parser": "^20.2.2" 3677 | } 3678 | }, 3679 | "yargs-parser": { 3680 | "version": "20.2.9", 3681 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", 3682 | "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", 3683 | "dev": true 3684 | }, 3685 | "yn": { 3686 | "version": "3.1.1", 3687 | "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", 3688 | "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", 3689 | "dev": true 3690 | } 3691 | } 3692 | } 3693 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@aws-samples/forecast-mlops-pipeline", 3 | "version": "0.1.0", 4 | "bin": { 5 | "forecast-mlops-pipeline": "bin/forecast-mlops-pipeline.js" 6 | }, 7 | "scripts": { 8 | "build": "tsc", 9 | "watch": "tsc -w", 10 | "test": "jest", 11 | "cdk": "cdk" 12 | }, 13 | "devDependencies": { 14 | "@types/jest": "^27.5.0", 15 | "@types/node": "10.17.27", 16 | "@types/prettier": "2.6.0", 17 | "aws-cdk": "2.27.0", 18 | "jest": "^27.5.1", 19 | "ts-jest": "^27.1.4", 20 | "ts-node": "^10.7.0", 21 | "typescript": "^3.9.10" 22 | }, 23 | "dependencies": { 24 | "@aws-cdk/aws-glue-alpha": "^2.28.0-alpha.0", 25 | "aws-cdk-lib": "2.80.0", 26 | "constructs": "^10.0.0", 27 | "source-map-support": "^0.5.21" 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /sample-data/data.zip: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/aws-samples/amazon-forecast-mlops-pipeline-cdk/4c2bd451e6abc1809150d86f778da2516ef0a89a/sample-data/data.zip -------------------------------------------------------------------------------- /sample-data/forecast-config.json: -------------------------------------------------------------------------------- 1 | { 2 | "TimestampFormat": "yyyy-MM-dd HH:mm:ss", 3 | "DatasetGroup": { 4 | "DatasetGroupName":"ExampleDatatsetGroup", 5 | "Domain": "CUSTOM" 6 | }, 7 | "Dataset": { 8 | "TTS": { 9 | "DatasetName": "Example_target_ts", 10 | "Domain": "CUSTOM", 11 | "DatasetType": "TARGET_TIME_SERIES", 12 | "DataFrequency": "M", 13 | "Schema": { 14 | "Attributes": [ 15 | { 16 | "AttributeName": "timestamp", 17 | "AttributeType": "timestamp" 18 | }, 19 | { 20 | "AttributeName": "item_id", 21 | "AttributeType": "string" 22 | }, 23 | { 24 | "AttributeName": "target_value", 25 | "AttributeType": "float" 26 | } 27 | ] 28 | } 29 | }, 30 | "IM": { 31 | "DatasetName": "Example_meta", 32 | "Domain": "CUSTOM", 33 | "DatasetType": "ITEM_METADATA", 34 | "Schema": { 35 | "Attributes": [ 36 | { 37 | "AttributeName": "item_id", 38 | "AttributeType": "string" 39 | }, 40 | { 41 | "AttributeName": "Model_Type", 42 | "AttributeType": "string" 43 | } 44 | ] 45 | } 46 | } 47 | }, 48 | "Predictor": { 49 | "AlgorithmArn": "arn:aws:forecast:::algorithm/CNN-QR", 50 | "PredictorName": "ExamplePredictor", 51 | "ForecastHorizon": 12, 52 | "FeaturizationConfig":{ 53 | "ForecastFrequency":"M" 54 | }, 55 | "PerformAutoML": false, 56 | "PerformHPO": false, 57 | "InputDataConfig": { 58 | "SupplementaryFeatures": [{ 59 | "Name": "holiday", 60 | "Value": "US" 61 | }] 62 | }, 63 | "TrainingParameters": { 64 | "context_length": "24" 65 | } 66 | }, 67 | "Forecast": { 68 | "ForecastName": "ExampleForecast", 69 | "ForecastTypes":[ 70 | "0.10", 71 | "0.50", 72 | "0.90" 73 | ] 74 | }, 75 | "PerformDelete": true 76 | } -------------------------------------------------------------------------------- /test/forecast-mlops-pipeline.test.ts: -------------------------------------------------------------------------------- 1 | // import * as cdk from 'aws-cdk-lib'; 2 | // import { Template } from 'aws-cdk-lib/assertions'; 3 | // import * as ForecastMlopsPipeline from '../lib/forecast-mlops-pipeline-stack'; 4 | 5 | // example test. To run these tests, uncomment this file along with the 6 | // example resource in lib/forecast-mlops-pipeline-stack.ts 7 | test('SQS Queue Created', () => { 8 | // const app = new cdk.App(); 9 | // // WHEN 10 | // const stack = new ForecastMlopsPipeline.ForecastMlopsPipelineStack(app, 'MyTestStack'); 11 | // // THEN 12 | // const template = Template.fromStack(stack); 13 | 14 | // template.hasResourceProperties('AWS::SQS::Queue', { 15 | // VisibilityTimeout: 300 16 | // }); 17 | }); 18 | -------------------------------------------------------------------------------- /test/test.py: -------------------------------------------------------------------------------- 1 | import boto3 2 | import os 3 | import pandas as pd 4 | 5 | 6 | bucket = 'forecast-mlops-pipeline-resource-bucket-699690702171' 7 | prefix = 'export/20220628T173905' 8 | dst_dir = 'test' 9 | 10 | s3 = boto3.client('s3') 11 | obj_list = s3.list_objects(Bucket=bucket, Prefix=prefix)['Contents'] 12 | 13 | for obj in obj_list: 14 | if obj['Key'].endswith('.csv'): 15 | basename = os.path.basename(obj['Key']) 16 | s3.download_file(bucket, obj['Key'], os.path.join(dst_dir, basename)) 17 | 18 | csvs = os.listdir('test') 19 | 20 | df_list = [] 21 | for csv in csvs: 22 | if csv.endswith('.csv'): 23 | appendant = pd.read_csv(os.path.join('test', csv)) 24 | df_list.append(appendant) 25 | 26 | merged = pd.concat(df_list, axis=1, join='inner').sort_index() 27 | merged.to_csv('test/merged.csv') 28 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "ES2018", 4 | "module": "commonjs", 5 | "lib": [ 6 | "es2018" 7 | ], 8 | "declaration": true, 9 | "strict": true, 10 | "noImplicitAny": true, 11 | "strictNullChecks": true, 12 | "noImplicitThis": true, 13 | "alwaysStrict": true, 14 | "noUnusedLocals": false, 15 | "noUnusedParameters": false, 16 | "noImplicitReturns": true, 17 | "noFallthroughCasesInSwitch": false, 18 | "inlineSourceMap": true, 19 | "inlineSources": true, 20 | "experimentalDecorators": true, 21 | "strictPropertyInitialization": false, 22 | "typeRoots": [ 23 | "./node_modules/@types" 24 | ] 25 | }, 26 | "exclude": [ 27 | "node_modules", 28 | "cdk.out" 29 | ] 30 | } 31 | --------------------------------------------------------------------------------