diff --git a/.eslintrc.json b/.eslintrc.json index 3b9586024..af43ec760 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -16,6 +16,11 @@ "es6": true, "jest/globals": true }, + "globals": { + "Deno": true, + "log": true, + "window": true + }, "rules": { // Error out for code formatting errors "prettier/prettier": "error", @@ -50,11 +55,34 @@ "no-continue": "off", // From experience, named exports are almost always desired. I got tired of this rule "import/prefer-default-export": "off", + // Disable in favour of TS equivalent + "no-unused-vars": "off", // Unused vars are useful to keep method signatures consistent and documented "@typescript-eslint/no-unused-vars": "off", // For this project only use kebab-case "unicorn/filename-case": ["error", { "cases": { "kebabCase": true } }], + "filenames/match-regex": ["error", "^[a-z-]+(\\.(d|integration|test|setup))*$"], // Allow Array.from(set) mitigate TS2569 which would require '--downlevelIteration' - "unicorn/prefer-spread": "off" + "unicorn/prefer-spread": "off", + // Deno has dependencies file, this rule enforces it's named re-exports + "unicorn/prevent-abbreviations": "off", + // Allow disabling eslint per file + "eslint-comments/no-use": "off", + // Deno import style not supported (enable after upgrading) + "import/extensions": "off", + // Deno import style not supported (enable after upgrading) + "import/no-unresolved": "off", + // App it not yet internationalised + "i18n-text/no-en": "off", + // Showing false positives (enable after upgrading) + "no-shadow": "off", + // (enable to add improvements) + "unicorn/no-static-only-class": "off", + // Null is useful when explicit value is passed + "unicorn/no-null": "off", + // (enable to add improvements) + "unicorn/prefer-export-from": "off", + // Deno uses const a=new Buffer(new TextEncoder().encode('Hello World')); + "unicorn/no-new-buffer": "off" } } diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md index ab4364318..d996e8647 100644 --- a/.github/ISSUE_TEMPLATE/bug_report.md +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -14,9 +14,7 @@ assignees: '' -- - -**Expected behavior** +- **Expected behavior** diff --git a/.github/workflows/build-tests.yml b/.github/workflows/build-tests.yml index 239f2b0f2..69a284507 100644 --- a/.github/workflows/build-tests.yml +++ b/.github/workflows/build-tests.yml @@ -11,14 +11,14 @@ env: jobs: buildForAllPlatformsUbuntu: - name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} + name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.engineVersion }} runs-on: ubuntu-latest strategy: fail-fast: false matrix: projectPath: - test-project - unityVersion: + engineVersion: - 2019.2.11f1 - 2019.3.15f1 targetPlatform: @@ -59,7 +59,7 @@ jobs: - uses: ./ with: projectPath: ${{ matrix.projectPath }} - unityVersion: ${{ matrix.unityVersion }} + engineVersion: ${{ matrix.engineVersion }} targetPlatform: ${{ matrix.targetPlatform }} customParameters: -profile SomeProfile -someBoolean -someValue exampleValue @@ -68,6 +68,6 @@ jobs: ########################### - uses: actions/upload-artifact@v2 with: - name: Build Ubuntu (${{ matrix.unityVersion }}) + name: Build Ubuntu (${{ matrix.engineVersion }}) path: build retention-days: 14 diff --git a/.github/workflows/cloud-runner-pipeline.yml b/.github/workflows/cloud-runner-pipeline.yml index 1b7c22933..a0a44bd97 100644 --- a/.github/workflows/cloud-runner-pipeline.yml +++ b/.github/workflows/cloud-runner-pipeline.yml @@ -35,7 +35,7 @@ jobs: matrix: projectPath: - test-project - unityVersion: + engineVersion: # - 2019.2.11f1 - 2019.3.15f1 targetPlatform: @@ -82,7 +82,7 @@ jobs: cloudRunnerCluster: aws versioning: None projectPath: ${{ matrix.projectPath }} - unityVersion: ${{ matrix.unityVersion }} + engineVersion: ${{ matrix.engineVersion }} targetPlatform: ${{ matrix.targetPlatform }} githubToken: ${{ secrets.GITHUB_TOKEN }} postBuildSteps: | @@ -118,12 +118,12 @@ jobs: path: build-${{ steps.aws-fargate-unity-build.outputs.BUILD_GUID }}.tar retention-days: 14 k8sBuilds: - name: K8s (GKE Autopilot) build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} + name: K8s (GKE Autopilot) build for ${{ matrix.targetPlatform }} on version ${{ matrix.engineVersion }} runs-on: ubuntu-latest strategy: fail-fast: false matrix: - unityVersion: + engineVersion: # - 2019.2.11f1 - 2019.3.15f1 targetPlatform: @@ -166,7 +166,7 @@ jobs: TARGET_PLATFORM: ${{ matrix.targetPlatform }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} KUBE_CONFIG: ${{ steps.read-base64.outputs.base64 }} - unityVersion: ${{ matrix.unityVersion }} + engineVersion: ${{ matrix.engineVersion }} cloudRunnerTests: true versioning: None @@ -184,7 +184,7 @@ jobs: kubeConfig: ${{ steps.read-base64.outputs.base64 }} githubToken: ${{ secrets.GITHUB_TOKEN }} projectPath: test-project - unityVersion: ${{ matrix.unityVersion }} + engineVersion: ${{ matrix.engineVersion }} versioning: None postBuildSteps: | - name: upload diff --git a/.github/workflows/mac-build-tests.yml b/.github/workflows/mac-build-tests.yml index 72ffe4a47..5de7178b0 100644 --- a/.github/workflows/mac-build-tests.yml +++ b/.github/workflows/mac-build-tests.yml @@ -10,14 +10,14 @@ env: jobs: buildForAllPlatformsWindows: - name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} + name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.engineVersion }} runs-on: macos-latest strategy: fail-fast: false matrix: projectPath: - test-project - unityVersion: + engineVersion: - 2020.3.24f1 targetPlatform: - StandaloneOSX # Build a MacOS executable @@ -58,7 +58,7 @@ jobs: UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} with: projectPath: ${{ matrix.projectPath }} - unityVersion: ${{ matrix.unityVersion }} + engineVersion: ${{ matrix.engineVersion }} targetPlatform: ${{ matrix.targetPlatform }} customParameters: -profile SomeProfile -someBoolean -someValue exampleValue # We use dirty build because we are replacing the default project settings file above @@ -69,6 +69,6 @@ jobs: ########################### - uses: actions/upload-artifact@v2 with: - name: Build MacOS (${{ matrix.unityVersion }}) + name: Build MacOS (${{ matrix.engineVersion }}) path: build retention-days: 14 diff --git a/.github/workflows/windows-build-tests.yml b/.github/workflows/windows-build-tests.yml index 90998f3e9..1caa8ab29 100644 --- a/.github/workflows/windows-build-tests.yml +++ b/.github/workflows/windows-build-tests.yml @@ -10,14 +10,14 @@ env: jobs: buildForAllPlatformsWindows: - name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} + name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.engineVersion }} runs-on: windows-2019 strategy: fail-fast: false matrix: projectPath: - test-project - unityVersion: + engineVersion: - 2020.3.24f1 targetPlatform: - StandaloneWindows64 # Build a Windows 64-bit standalone. @@ -61,7 +61,7 @@ jobs: UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} with: projectPath: ${{ matrix.projectPath }} - unityVersion: ${{ matrix.unityVersion }} + engineVersion: ${{ matrix.engineVersion }} targetPlatform: ${{ matrix.targetPlatform }} customParameters: -profile SomeProfile -someBoolean -someValue exampleValue allowDirtyBuild: true @@ -72,6 +72,6 @@ jobs: ########################### - uses: actions/upload-artifact@v2 with: - name: Build Windows (${{ matrix.unityVersion }}) + name: Build Windows (${{ matrix.engineVersion }}) path: build retention-days: 14 diff --git a/.gitignore b/.gitignore index bc20e9148..7d35185d6 100644 --- a/.gitignore +++ b/.gitignore @@ -1,7 +1,13 @@ .idea node_modules coverage/ +.coverage/ lib/ .vsconfig yarn-error.log .orig +*.log +logs/* +!**/.gitkeep +.env* +!.env*.dist diff --git a/.run/GameCI build ._test-project.run.xml b/.run/GameCI build ._test-project.run.xml new file mode 100644 index 000000000..aebc35acb --- /dev/null +++ b/.run/GameCI build ._test-project.run.xml @@ -0,0 +1,6 @@ + + + + \ No newline at end of file diff --git a/.run/Reamde.md b/.run/Reamde.md new file mode 100644 index 000000000..7e7794084 --- /dev/null +++ b/.run/Reamde.md @@ -0,0 +1,7 @@ +# .run + +This folder contains development script for the project. + +WebStorm allows running it directly from the IDE. + +Webstorm interface example with play button diff --git a/.run/example-run-interface.png b/.run/example-run-interface.png new file mode 100644 index 000000000..0dbb7cc94 Binary files /dev/null and b/.run/example-run-interface.png differ diff --git a/.tools/lcov/genhtml.perl b/.tools/lcov/genhtml.perl new file mode 100644 index 000000000..a8f16a0e6 --- /dev/null +++ b/.tools/lcov/genhtml.perl @@ -0,0 +1,6102 @@ +#!/usr/bin/env perl +# +# Copyright (c) International Business Machines Corp., 2002,2012 +# +# This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or (at +# your option) any later version. +# +# This program is distributed in the hope that it will be useful, but +# WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +# General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, see +# . +# +# +# genhtml +# +# This script generates HTML output from .info files as created by the +# geninfo script. Call it with --help and refer to the genhtml man page +# to get information on usage and available options. +# +# +# History: +# 2002-08-23 created by Peter Oberparleiter +# IBM Lab Boeblingen +# based on code by Manoj Iyer and +# Megan Bock +# IBM Austin +# 2002-08-27 / Peter Oberparleiter: implemented frame view +# 2002-08-29 / Peter Oberparleiter: implemented test description filtering +# so that by default only descriptions for test cases which +# actually hit some source lines are kept +# 2002-09-05 / Peter Oberparleiter: implemented --no-sourceview +# 2002-09-05 / Mike Kobler: One of my source file paths includes a "+" in +# the directory name. I found that genhtml.pl died when it +# encountered it. I was able to fix the problem by modifying +# the string with the escape character before parsing it. +# 2002-10-26 / Peter Oberparleiter: implemented --num-spaces +# 2003-04-07 / Peter Oberparleiter: fixed bug which resulted in an error +# when trying to combine .info files containing data without +# a test name +# 2003-04-10 / Peter Oberparleiter: extended fix by Mike to also cover +# other special characters +# 2003-04-30 / Peter Oberparleiter: made info write to STDERR, not STDOUT +# 2003-07-10 / Peter Oberparleiter: added line checksum support +# 2004-08-09 / Peter Oberparleiter: added configuration file support +# 2005-03-04 / Cal Pierog: added legend to HTML output, fixed coloring of +# "good coverage" background +# 2006-03-18 / Marcus Boerger: added --custom-intro, --custom-outro and +# overwrite --no-prefix if --prefix is present +# 2006-03-20 / Peter Oberparleiter: changes to custom_* function (rename +# to html_prolog/_epilog, minor modifications to implementation), +# changed prefix/noprefix handling to be consistent with current +# logic +# 2006-03-20 / Peter Oberparleiter: added --html-extension option +# 2008-07-14 / Tom Zoerner: added --function-coverage command line option; +# added function table to source file page +# 2008-08-13 / Peter Oberparleiter: modified function coverage +# implementation (now enabled per default), +# introduced sorting option (enabled per default) +# + +use strict; +use warnings; +use File::Basename; +use File::Temp qw(tempfile); +use Getopt::Long; +use Digest::MD5 qw(md5_base64); +use Cwd qw/abs_path cwd/; + +use Perl::OSType ':all'; + +# Global constants +our $title = "LCOV - code coverage report"; +our $tool_dir = abs_path(dirname($0)); +our $lcov_version = "LCOV version 1.15.alpha0w"; +our $lcov_url = "http://ltp.sourceforge.net/coverage/lcov.php"; +our $tool_name = basename($0); + +# Specify coverage rate default precision +our $default_precision = 1; + +# Specify coverage rate limits (in %) for classifying file entries +# HI: $hi_limit <= rate <= 100 graph color: green +# MED: $med_limit <= rate < $hi_limit graph color: orange +# LO: 0 <= rate < $med_limit graph color: red + +# For line coverage/all coverage types if not specified +our $hi_limit = 90; +our $med_limit = 75; + +# For function coverage +our $fn_hi_limit; +our $fn_med_limit; + +# For branch coverage +our $br_hi_limit; +our $br_med_limit; + +# Width of overview image +our $overview_width = 80; + +# Resolution of overview navigation: this number specifies the maximum +# difference in lines between the position a user selected from the overview +# and the position the source code window is scrolled to. +our $nav_resolution = 4; + +# Clicking a line in the overview image should show the source code view at +# a position a bit further up so that the requested line is not the first +# line in the window. This number specifies that offset in lines. +our $nav_offset = 10; + +# Clicking on a function name should show the source code at a position a +# few lines before the first line of code of that function. This number +# specifies that offset in lines. +our $func_offset = 2; + +our $overview_title = "top level"; + +# Width for line coverage information in the source code view +our $line_field_width = 12; + +# Width for branch coverage information in the source code view +our $br_field_width = 16; + +# Internal Constants + +# Header types +our $HDR_DIR = 0; +our $HDR_FILE = 1; +our $HDR_SOURCE = 2; +our $HDR_TESTDESC = 3; +our $HDR_FUNC = 4; + +# Sort types +our $SORT_FILE = 0; +our $SORT_LINE = 1; +our $SORT_FUNC = 2; +our $SORT_BRANCH = 3; + +# Fileview heading types +our $HEAD_NO_DETAIL = 1; +our $HEAD_DETAIL_HIDDEN = 2; +our $HEAD_DETAIL_SHOWN = 3; + +# Additional offsets used when converting branch coverage data to HTML +our $BR_LEN = 3; +our $BR_OPEN = 4; +our $BR_CLOSE = 5; + +# Branch data combination types +our $BR_SUB = 0; +our $BR_ADD = 1; + +# Block value used for unnamed blocks +our $UNNAMED_BLOCK = vec(pack('b*', 1 x 32), 0, 32); + +# Error classes which users may specify to ignore during processing +our $ERROR_SOURCE = 0; +our %ERROR_ID = ( + "source" => $ERROR_SOURCE, +); + +# Data related prototypes +sub resolve_absolute_path($$); +sub print_usage(*); +sub gen_html(); +sub html_create($$); +sub process_dir($); +sub process_file($$$); +sub info(@); +sub read_info_file($); +sub get_info_entry($); +sub set_info_entry($$$$$$$$$;$$$$$$); +sub get_prefix($@); +sub shorten_prefix($); +sub get_dir_list(@); +sub get_relative_base_path($); +sub read_testfile($); +sub get_date_string(); +sub create_sub_dir($); +sub subtract_counts($$); +sub add_counts($$); +sub apply_baseline($$); +sub remove_unused_descriptions(); +sub get_found_and_hit($); +sub get_affecting_tests($$$); +sub combine_info_files($$); +sub merge_checksums($$$); +sub combine_info_entries($$$); +sub apply_prefix($@); +sub system_no_output($@); +sub read_config($); +sub apply_config($); +sub get_html_prolog($); +sub get_html_epilog($); +sub write_dir_page($$$$$$$$$$$$$$$$$); +sub classify_rate($$$$); +sub combine_brcount($$$;$); +sub get_br_found_and_hit($); +sub warn_handler($); +sub die_handler($); +sub parse_ignore_errors(@); +sub parse_dir_prefix(@); +sub rate($$;$$$); + + +# HTML related prototypes +sub escape_html($); +sub get_bar_graph_code($$$); + +sub write_png_files(); +sub write_htaccess_file(); +sub write_css_file(); +sub write_description_file($$$$$$$); +sub write_function_table(*$$$$$$$$$$); + +sub write_html(*$); +sub write_html_prolog(*$$); +sub write_html_epilog(*$;$); + +sub write_header(*$$$$$$$$$$); +sub write_header_prolog(*$); +sub write_header_line(*@); +sub write_header_epilog(*$); + +sub write_file_table(*$$$$$$$); +sub write_file_table_prolog(*$@); +sub write_file_table_entry(*$$$@); +sub write_file_table_detail_entry(*$@); +sub write_file_table_epilog(*); + +sub write_test_table_prolog(*$); +sub write_test_table_entry(*$$); +sub write_test_table_epilog(*); + +sub write_source($$$$$$$); +sub write_source_prolog(*); +sub write_source_line(*$$$$$); +sub write_source_epilog(*); + +sub write_frameset(*$$$); +sub write_overview_line(*$$$); +sub write_overview(*$$$$); + +# External prototype (defined in genpng) +sub gen_png($$$@); + + +# Global variables & initialization +our %info_data; # Hash containing all data from .info file +our @opt_dir_prefix; # Array of prefixes to remove from all sub directories +our @dir_prefix; +our %test_description; # Hash containing test descriptions if available +our $date; + +our @info_filenames; # List of .info files to use as data source +our $test_title; # Title for output as written to each page header +our $output_directory; # Name of directory in which to store output +our $base_filename; # Optional name of file containing baseline data +our $desc_filename; # Name of file containing test descriptions +our $css_filename; # Optional name of external stylesheet file to use +our $quiet; # If set, suppress information messages +our $help; # Help option flag +our $version; # Version option flag +our $show_details; # If set, generate detailed directory view +our $no_prefix; # If set, do not remove filename prefix +our $func_coverage; # If set, generate function coverage statistics +our $no_func_coverage; # Disable func_coverage +our $br_coverage; # If set, generate branch coverage statistics +our $no_br_coverage; # Disable br_coverage +our $sort = 1; # If set, provide directory listings with sorted entries +our $no_sort; # Disable sort +our $frames; # If set, use frames for source code view +our $keep_descriptions; # If set, do not remove unused test case descriptions +our $no_sourceview; # If set, do not create a source code view for each file +our $highlight; # If set, highlight lines covered by converted data only +our $legend; # If set, include legend in output +our $tab_size = 8; # Number of spaces to use in place of tab +our $config; # Configuration file contents +our $html_prolog_file; # Custom HTML prolog file (up to and including ) +our $html_epilog_file; # Custom HTML epilog file (from onwards) +our $html_prolog; # Actual HTML prolog +our $html_epilog; # Actual HTML epilog +our $html_ext = "html"; # Extension for generated HTML files +our $html_gzip = 0; # Compress with gzip +our $demangle_cpp = 0; # Demangle C++ function names +our @opt_ignore_errors; # Ignore certain error classes during processing +our @ignore; +our $opt_config_file; # User-specified configuration file location +our %opt_rc; +our $opt_missed; # List/sort lines by missed counts +our $charset = "UTF-8"; # Default charset for HTML pages +our @fileview_sortlist; +our @fileview_sortname = ("", "-sort-l", "-sort-f", "-sort-b"); +our @funcview_sortlist; +our @rate_name = ("Lo", "Med", "Hi"); +our @rate_png = ("ruby.png", "amber.png", "emerald.png"); +our $lcov_func_coverage = 1; +our $lcov_branch_coverage = 0; +our $rc_desc_html = 0; # lcovrc: genhtml_desc_html +our $os_is_windows; +our $dir_sep; # Directory separator character + +our $cwd = cwd(); # Current working directory + +# +# Code entry point +# + +$SIG{__WARN__} = \&warn_handler; +$SIG{__DIE__} = \&die_handler; + +# Are we running on Windows? +$os_is_windows = is_os_type('Windows'); + +if ($os_is_windows) { + $dir_sep = "\\"; + $date = localtime; +} +else +{ + $dir_sep = "/"; + $date = get_date_string(); +} + +# Check command line for a configuration file name +Getopt::Long::Configure("pass_through", "no_auto_abbrev"); +GetOptions("config-file=s" => \$opt_config_file, + "rc=s%" => \%opt_rc); +Getopt::Long::Configure("default"); + +{ + # Remove spaces around rc options + my %new_opt_rc; + + while (my ($key, $value) = each(%opt_rc)) { + $key =~ s/^\s+|\s+$//g; + $value =~ s/^\s+|\s+$//g; + + $new_opt_rc{$key} = $value; + } + %opt_rc = %new_opt_rc; +} + +# Read configuration file if available +if (defined($opt_config_file)) +{ + $config = read_config($opt_config_file); +} +elsif (defined($ENV{"HOME"}) && (-r $ENV{"HOME"}."/.lcovrc")) +{ + $config = read_config($ENV{"HOME"}."/.lcovrc"); +} +elsif ((!$os_is_windows) and (-r "/etc/lcovrc")) +{ + $config = read_config("/etc/lcovrc"); +} elsif ((!$os_is_windows) and (-r "/usr/local/etc/lcovrc")) +{ + $config = read_config("/usr/local/etc/lcovrc"); +} +elsif (($os_is_windows) and (-r "$tool_dir/../etc/lcovrc")) +{ + $config = read_config("$tool_dir/../etc/lcovrc"); +} +elsif (($os_is_windows) and (-r "$tool_dir/../lcovrc")) +{ + $config = read_config("$tool_dir/../lcovrc"); +} + +if ($config || %opt_rc) +{ + # Copy configuration file and --rc values to variables + apply_config({ + "genhtml_css_file" => \$css_filename, + "genhtml_hi_limit" => \$hi_limit, + "genhtml_med_limit" => \$med_limit, + "genhtml_line_field_width" => \$line_field_width, + "genhtml_overview_width" => \$overview_width, + "genhtml_nav_resolution" => \$nav_resolution, + "genhtml_nav_offset" => \$nav_offset, + "genhtml_keep_descriptions" => \$keep_descriptions, + "genhtml_no_prefix" => \$no_prefix, + "genhtml_no_source" => \$no_sourceview, + "genhtml_num_spaces" => \$tab_size, + "genhtml_highlight" => \$highlight, + "genhtml_legend" => \$legend, + "genhtml_html_prolog" => \$html_prolog_file, + "genhtml_html_epilog" => \$html_epilog_file, + "genhtml_html_extension" => \$html_ext, + "genhtml_html_gzip" => \$html_gzip, + "genhtml_precision" => \$default_precision, + "genhtml_function_hi_limit" => \$fn_hi_limit, + "genhtml_function_med_limit" => \$fn_med_limit, + "genhtml_function_coverage" => \$func_coverage, + "genhtml_branch_hi_limit" => \$br_hi_limit, + "genhtml_branch_med_limit" => \$br_med_limit, + "genhtml_branch_coverage" => \$br_coverage, + "genhtml_branch_field_width" => \$br_field_width, + "genhtml_sort" => \$sort, + "genhtml_charset" => \$charset, + "genhtml_desc_html" => \$rc_desc_html, + "genhtml_demangle_cpp" => \$demangle_cpp, + "genhtml_missed" => \$opt_missed, + "lcov_function_coverage" => \$lcov_func_coverage, + "lcov_branch_coverage" => \$lcov_branch_coverage, + }); +} + +# Copy related values if not specified +$fn_hi_limit = $hi_limit if (!defined($fn_hi_limit)); +$fn_med_limit = $med_limit if (!defined($fn_med_limit)); +$br_hi_limit = $hi_limit if (!defined($br_hi_limit)); +$br_med_limit = $med_limit if (!defined($br_med_limit)); +$func_coverage = $lcov_func_coverage if (!defined($func_coverage)); +$br_coverage = $lcov_branch_coverage if (!defined($br_coverage)); + +# Parse command line options +if (!GetOptions("output-directory|o=s" => \$output_directory, + "title|t=s" => \$test_title, + "description-file|d=s" => \$desc_filename, + "keep-descriptions|k" => \$keep_descriptions, + "css-file|c=s" => \$css_filename, + "baseline-file|b=s" => \$base_filename, + "prefix|p=s" => \@opt_dir_prefix, + "num-spaces=i" => \$tab_size, + "no-prefix" => \$no_prefix, + "no-sourceview" => \$no_sourceview, + "show-details|s" => \$show_details, + "frames|f" => \$frames, + "highlight" => \$highlight, + "legend" => \$legend, + "quiet|q" => \$quiet, + "help|h|?" => \$help, + "version|v" => \$version, + "html-prolog=s" => \$html_prolog_file, + "html-epilog=s" => \$html_epilog_file, + "html-extension=s" => \$html_ext, + "html-gzip" => \$html_gzip, + "function-coverage" => \$func_coverage, + "no-function-coverage" => \$no_func_coverage, + "branch-coverage" => \$br_coverage, + "no-branch-coverage" => \$no_br_coverage, + "sort" => \$sort, + "no-sort" => \$no_sort, + "demangle-cpp" => \$demangle_cpp, + "ignore-errors=s" => \@opt_ignore_errors, + "config-file=s" => \$opt_config_file, + "rc=s%" => \%opt_rc, + "precision=i" => \$default_precision, + "missed" => \$opt_missed, + )) +{ + print(STDERR "Use $tool_name --help to get usage information\n"); + exit(1); +} else { + # Merge options + if ($no_func_coverage) { + $func_coverage = 0; + } + if ($no_br_coverage) { + $br_coverage = 0; + } + + # Merge sort options + if ($no_sort) { + $sort = 0; + } +} + +@info_filenames = @ARGV; + +# Check for help option +if ($help) +{ + print_usage(*STDOUT); + exit(0); +} + +# Check for version option +if ($version) +{ + print("$tool_name: $lcov_version\n"); + exit(0); +} + +if( $os_is_windows ) +{ + # Replace forward slashes by backslashes + if( defined($desc_filename) ) { $desc_filename =~ s/\//\\/g; } + if( defined($css_filename) ) { $css_filename =~ s/\//\\/g; } + if( defined($base_filename) ) { $base_filename =~ s/\//\\/g; } + for( @opt_dir_prefix ) + { + s/\//\\/g; + } + if( defined($opt_config_file) ) { $opt_config_file =~ s/\//\\/g; } + for( @info_filenames ) + { + s/\//\\/g; + } +} + +# Determine which errors the user wants us to ignore +parse_ignore_errors(@opt_ignore_errors); + +# Split the list of prefixes if needed +parse_dir_prefix(@opt_dir_prefix); + +# Check for info filename +if (!@info_filenames) +{ + die("No filename specified\n". + "Use $tool_name --help to get usage information\n"); +} + +# Generate a title if none is specified +if (!$test_title) +{ + if (scalar(@info_filenames) == 1) + { + # Only one filename specified, use it as title + $test_title = basename($info_filenames[0]); + } + else + { + # More than one filename specified, used default title + $test_title = "unnamed"; + } +} + +# Make sure css_filename is an absolute path (in case we're changing +# directories) +if ($css_filename) +{ + $css_filename = resolve_absolute_path($css_filename, $cwd); +} + +# Make sure tab_size is within valid range +if ($tab_size < 1) +{ + print(STDERR "ERROR: invalid number of spaces specified: ". + "$tab_size!\n"); + exit(1); +} + +# Get HTML prolog and epilog +$html_prolog = get_html_prolog($html_prolog_file); +$html_epilog = get_html_epilog($html_epilog_file); + +# Issue a warning if --no-sourceview is enabled together with --frames +if ($no_sourceview && defined($frames)) +{ + warn("WARNING: option --frames disabled because --no-sourceview ". + "was specified!\n"); + $frames = undef; +} + +# Issue a warning if --no-prefix is enabled together with --prefix +if ($no_prefix && @dir_prefix) +{ + warn("WARNING: option --prefix disabled because --no-prefix was ". + "specified!\n"); + @dir_prefix = undef; +} + +@fileview_sortlist = ($SORT_FILE); +@funcview_sortlist = ($SORT_FILE); + +if ($sort) { + push(@fileview_sortlist, $SORT_LINE); + push(@fileview_sortlist, $SORT_FUNC) if ($func_coverage); + push(@fileview_sortlist, $SORT_BRANCH) if ($br_coverage); + push(@funcview_sortlist, $SORT_LINE); +} + +if ($frames) +{ + # Include genpng code needed for overview image generation + do("$tool_dir/genpng"); +} + +# Ensure that the c++filt tool is available when using --demangle-cpp +if ($demangle_cpp) +{ + if (system_no_output(3, "c++filt", "--version")) { + die("ERROR: could not find c++filt tool needed for ". + "--demangle-cpp\n"); + } +} + +# Make sure precision is within valid range +if ($default_precision < 1 || $default_precision > 4) +{ + die("ERROR: specified precision is out of range (1 to 4)\n"); +} + +# Make sure output_directory exists, create it if necessary +if ($output_directory) +{ + stat($output_directory); + + if (! -e _) + { + create_sub_dir($output_directory); + } +} + +# Do something +gen_html(); + +exit(0); + + +# +# print_usage(handle) +# +# Print usage information. +# + +sub print_usage(*) +{ + local *HANDLE = $_[0]; + + print(HANDLE <{$filename}; + my $funcdata = $data->{"func"}; + my $sumfnccount = $data->{"sumfnc"}; + + if (defined($funcdata)) { + foreach my $func_name (keys(%{$funcdata})) { + $fns{$func_name} = 1; + } + } + + if (defined($sumfnccount)) { + foreach my $func_name (keys(%{$sumfnccount})) { + $fns{$func_name} = 1; + } + } + } + + @result = keys(%fns); + + return \@result; +} + +# +# rename_functions(info, conv) +# +# Rename all function names in INFO according to CONV: OLD_NAME -> NEW_NAME. +# In case two functions demangle to the same name, assume that they are +# different object code implementations for the same source function. +# + +sub rename_functions($$) +{ + my ($info, $conv) = @_; + + foreach my $filename (keys(%{$info})) { + my $data = $info->{$filename}; + my $funcdata; + my $testfncdata; + my $sumfnccount; + my %newfuncdata; + my %newsumfnccount; + my $f_found; + my $f_hit; + + # funcdata: function name -> line number + $funcdata = $data->{"func"}; + foreach my $fn (keys(%{$funcdata})) { + my $cn = $conv->{$fn}; + + # Abort if two functions on different lines map to the + # same demangled name. + if (defined($newfuncdata{$cn}) && + $newfuncdata{$cn} != $funcdata->{$fn}) { + die("ERROR: Demangled function name $cn ". + "maps to different lines (". + $newfuncdata{$cn}." vs ". + $funcdata->{$fn}.") in $filename\n"); + } + $newfuncdata{$cn} = $funcdata->{$fn}; + } + $data->{"func"} = \%newfuncdata; + + # testfncdata: test name -> testfnccount + # testfnccount: function name -> execution count + $testfncdata = $data->{"testfnc"}; + foreach my $tn (keys(%{$testfncdata})) { + my $testfnccount = $testfncdata->{$tn}; + my %newtestfnccount; + + foreach my $fn (keys(%{$testfnccount})) { + my $cn = $conv->{$fn}; + + # Add counts for different functions that map + # to the same name. + $newtestfnccount{$cn} += + $testfnccount->{$fn}; + } + $testfncdata->{$tn} = \%newtestfnccount; + } + + # sumfnccount: function name -> execution count + $sumfnccount = $data->{"sumfnc"}; + foreach my $fn (keys(%{$sumfnccount})) { + my $cn = $conv->{$fn}; + + # Add counts for different functions that map + # to the same name. + $newsumfnccount{$cn} += $sumfnccount->{$fn}; + } + $data->{"sumfnc"} = \%newsumfnccount; + + # Update function found and hit counts since they may have + # changed + $f_found = 0; + $f_hit = 0; + foreach my $fn (keys(%newsumfnccount)) { + $f_found++; + $f_hit++ if ($newsumfnccount{$fn} > 0); + } + $data->{"f_found"} = $f_found; + $data->{"f_hit"} = $f_hit; + } +} + +# +# gen_html() +# +# Generate a set of HTML pages from contents of .info file INFO_FILENAME. +# Files will be written to the current directory. If provided, test case +# descriptions will be read from .tests file TEST_FILENAME and included +# in ouput. +# +# Die on error. +# + +sub gen_html() +{ + local *HTML_HANDLE; + my %overview; + my %base_data; + my $lines_found; + my $lines_hit; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + my $overall_found = 0; + my $overall_hit = 0; + my $total_fn_found = 0; + my $total_fn_hit = 0; + my $total_br_found = 0; + my $total_br_hit = 0; + my $dir_name; + my $link_name; + my @dir_list; + my %new_info; + + # Read in all specified .info files + foreach (@info_filenames) + { + %new_info = %{read_info_file($_)}; + + # Combine %new_info with %info_data + %info_data = %{combine_info_files(\%info_data, \%new_info)}; + } + + info("Found %d entries.\n", scalar(keys(%info_data))); + + # Read and apply baseline data if specified + if ($base_filename) + { + # Read baseline file + info("Reading baseline file $base_filename\n"); + %base_data = %{read_info_file($base_filename)}; + info("Found %d entries.\n", scalar(keys(%base_data))); + + # Apply baseline + info("Subtracting baseline data.\n"); + %info_data = %{apply_baseline(\%info_data, \%base_data)}; + } + + @dir_list = get_dir_list(keys(%info_data)); + + if ($no_prefix) + { + # User requested that we leave filenames alone + info("User asked not to remove filename prefix\n"); + } + elsif (! @dir_prefix) + { + # Get prefix common to most directories in list + my $prefix = get_prefix(1, keys(%info_data)); + + if ($prefix) + { + info("Found common filename prefix \"$prefix\"\n"); + $dir_prefix[0] = $prefix; + + } + else + { + info("No common filename prefix found!\n"); + $no_prefix=1; + } + } + else + { + my $msg = "Using user-specified filename prefix "; + for my $i (0 .. $#dir_prefix) + { + $dir_prefix[$i] = resolve_absolute_path( $dir_prefix[$i], $cwd ); + $msg .= ", " unless 0 == $i; + $msg .= "\"" . $dir_prefix[$i] . "\""; + } + info($msg . "\n"); + } + + + # Read in test description file if specified + if ($desc_filename) + { + info("Reading test description file $desc_filename\n"); + %test_description = %{read_testfile($desc_filename)}; + + # Remove test descriptions which are not referenced + # from %info_data if user didn't tell us otherwise + if (!$keep_descriptions) + { + remove_unused_descriptions(); + } + } + + # Change to output directory if specified + if ($output_directory) + { + chdir($output_directory) + or die("ERROR: cannot change to directory ". + "$output_directory!\n"); + } + + info("Writing .css and .png files.\n"); + write_css_file(); + write_png_files(); + + if ($html_gzip) + { + info("Writing .htaccess file.\n"); + write_htaccess_file(); + } + + info("Generating output.\n"); + + # Process each subdirectory and collect overview information + foreach $dir_name (@dir_list) + { + ($lines_found, $lines_hit, $fn_found, $fn_hit, + $br_found, $br_hit) + = process_dir($dir_name); + + # Remove prefix if applicable + if (!$no_prefix && @dir_prefix) + { + # Match directory names beginning with one of @dir_prefix + $dir_name = apply_prefix($dir_name,@dir_prefix); + } + + # Remove leading directory separator + $dir_name =~ s/^\Q$dir_sep\E+//; + + # Handle files in root directory gracefully + $dir_name = "root" if ($dir_name eq ""); + + # Generate name for directory overview HTML page + $link_name = $dir_name."/index.$html_ext"; + + if( $os_is_windows ) + { + $link_name =~ s/^([a-zA-Z]):/$1/; + } + + $overview{$dir_name} = [$lines_found, $lines_hit, $fn_found, + $fn_hit, $br_found, $br_hit, $link_name, + get_rate($lines_found, $lines_hit), + get_rate($fn_found, $fn_hit), + get_rate($br_found, $br_hit)]; + $overall_found += $lines_found; + $overall_hit += $lines_hit; + $total_fn_found += $fn_found; + $total_fn_hit += $fn_hit; + $total_br_found += $br_found; + $total_br_hit += $br_hit; + } + + # Generate overview page + info("Writing directory view page.\n"); + + # Create sorted pages + foreach (@fileview_sortlist) { + write_dir_page($fileview_sortname[$_], ".", "", $test_title, + undef, $overall_found, $overall_hit, + $total_fn_found, $total_fn_hit, $total_br_found, + $total_br_hit, \%overview, {}, {}, {}, 0, $_); + } + + # Check if there are any test case descriptions to write out + if (%test_description) + { + info("Writing test case description file.\n"); + write_description_file( \%test_description, + $overall_found, $overall_hit, + $total_fn_found, $total_fn_hit, + $total_br_found, $total_br_hit); + } + + print_overall_rate(1, $overall_found, $overall_hit, + $func_coverage, $total_fn_found, $total_fn_hit, + $br_coverage, $total_br_found, $total_br_hit); + + chdir($cwd); +} + +# +# html_create(handle, filename) +# + +sub html_create($$) +{ + my $handle = $_[0]; + my $filename = $_[1]; + + if ($html_gzip) + { + open($handle, "|-", "gzip -c >'$filename'") + or die("ERROR: cannot open $filename for writing ". + "(gzip)!\n"); + } + else + { + open($handle, ">", $filename) + or die("ERROR: cannot open $filename for writing!\n"); + } +} + +sub write_dir_page($$$$$$$$$$$$$$$$$) +{ + my ($name, $rel_dir, $base_dir, $title, $trunc_dir, $overall_found, + $overall_hit, $total_fn_found, $total_fn_hit, $total_br_found, + $total_br_hit, $overview, $testhash, $testfnchash, $testbrhash, + $view_type, $sort_type) = @_; + + # Generate directory overview page including details + html_create(*HTML_HANDLE, $rel_dir.$dir_sep."index$name.$html_ext"); + if (!defined($trunc_dir)) { + $trunc_dir = ""; + } + $title .= " - " if ($trunc_dir ne ""); + write_html_prolog(*HTML_HANDLE, $base_dir, "LCOV - $title$trunc_dir"); + write_header(*HTML_HANDLE, $view_type, $trunc_dir, $rel_dir, + $overall_found, $overall_hit, $total_fn_found, + $total_fn_hit, $total_br_found, $total_br_hit, $sort_type); + write_file_table(*HTML_HANDLE, $base_dir, $overview, $testhash, + $testfnchash, $testbrhash, $view_type, $sort_type); + write_html_epilog(*HTML_HANDLE, $base_dir); + close(*HTML_HANDLE); +} + + +# +# process_dir(dir_name) +# + +sub process_dir($) +{ + my $abs_dir = $_[0]; + my $trunc_dir; + my $rel_dir = $abs_dir; + my $base_dir; + my $filename; + my %overview; + my $lines_found; + my $lines_hit; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + my $overall_found=0; + my $overall_hit=0; + my $total_fn_found=0; + my $total_fn_hit=0; + my $total_br_found = 0; + my $total_br_hit = 0; + my $base_name; + my $extension; + my $testdata; + my %testhash; + my $testfncdata; + my %testfnchash; + my $testbrdata; + my %testbrhash; + my @sort_list; + local *HTML_HANDLE; + + # Remove prefix if applicable + if (!$no_prefix) + { + # Match directory name beginning with one of @dir_prefix + $rel_dir = apply_prefix($rel_dir,@dir_prefix); + } + + $trunc_dir = $rel_dir; + + if( $os_is_windows ) + { + # Change volume name to be "directory friendly" + $rel_dir =~ s/^([a-zA-Z]):/$1/; + } + else + { + # Remove leading / + $rel_dir =~ s/^\/+//; + } + + # Handle files in root directory gracefully + $rel_dir = "root" if ($rel_dir eq ""); + $trunc_dir = "root" if ($trunc_dir eq ""); + + $base_dir = get_relative_base_path($rel_dir); + + create_sub_dir($rel_dir); + + # Match filenames which specify files in this directory, not including + # sub-directories + foreach $filename ( grep( /^\Q$abs_dir$dir_sep\E[^\Q$dir_sep\E]*$/, keys(%info_data) ) ) + { + my $page_link; + my $func_link; + + ($lines_found, $lines_hit, $fn_found, $fn_hit, $br_found, + $br_hit, $testdata, $testfncdata, $testbrdata) = + process_file($trunc_dir, $rel_dir, $filename); + + $base_name = basename($filename); + + if ($no_sourceview) { + $page_link = ""; + } elsif ($frames) { + # Link to frameset page + $page_link = "$base_name.gcov.frameset.$html_ext"; + } else { + # Link directory to source code view page + $page_link = "$base_name.gcov.$html_ext"; + } + $overview{$base_name} = [$lines_found, $lines_hit, $fn_found, + $fn_hit, $br_found, $br_hit, + $page_link, + get_rate($lines_found, $lines_hit), + get_rate($fn_found, $fn_hit), + get_rate($br_found, $br_hit)]; + + $testhash{$base_name} = $testdata; + $testfnchash{$base_name} = $testfncdata; + $testbrhash{$base_name} = $testbrdata; + + $overall_found += $lines_found; + $overall_hit += $lines_hit; + + $total_fn_found += $fn_found; + $total_fn_hit += $fn_hit; + + $total_br_found += $br_found; + $total_br_hit += $br_hit; + } + + # Create sorted pages + foreach (@fileview_sortlist) { + # Generate directory overview page (without details) + write_dir_page($fileview_sortname[$_], $rel_dir, $base_dir, + $test_title, $trunc_dir, $overall_found, + $overall_hit, $total_fn_found, $total_fn_hit, + $total_br_found, $total_br_hit, \%overview, {}, + {}, {}, 1, $_); + if (!$show_details) { + next; + } + # Generate directory overview page including details + write_dir_page("-detail".$fileview_sortname[$_], $rel_dir, + $base_dir, $test_title, $trunc_dir, + $overall_found, $overall_hit, $total_fn_found, + $total_fn_hit, $total_br_found, $total_br_hit, + \%overview, \%testhash, \%testfnchash, + \%testbrhash, 1, $_); + } + + # Calculate resulting line counts + return ($overall_found, $overall_hit, $total_fn_found, $total_fn_hit, + $total_br_found, $total_br_hit); +} + + +# +# get_converted_lines(testdata) +# +# Return hash of line numbers of those lines which were only covered in +# converted data sets. +# + +sub get_converted_lines($) +{ + my $testdata = $_[0]; + my $testcount; + my %converted; + my %nonconverted; + my $hash; + my $testcase; + my $line; + my %result; + + + # Get a hash containing line numbers with positive counts both for + # converted and original data sets + foreach $testcase (keys(%{$testdata})) + { + # Check to see if this is a converted data set + if ($testcase =~ /,diff$/) + { + $hash = \%converted; + } + else + { + $hash = \%nonconverted; + } + + $testcount = $testdata->{$testcase}; + # Add lines with a positive count to hash + foreach $line (keys%{$testcount}) + { + if ($testcount->{$line} > 0) + { + $hash->{$line} = 1; + } + } + } + + # Combine both hashes to resulting list + foreach $line (keys(%converted)) + { + if (!defined($nonconverted{$line})) + { + $result{$line} = 1; + } + } + + return \%result; +} + + +sub write_function_page($$$$$$$$$$$$$$$$$$) +{ + my ($base_dir, $rel_dir, $trunc_dir, $base_name, $title, + $lines_found, $lines_hit, $fn_found, $fn_hit, $br_found, $br_hit, + $sumcount, $funcdata, $sumfnccount, $testfncdata, $sumbrcount, + $testbrdata, $sort_type) = @_; + my $pagetitle; + my $filename; + + # Generate function table for this file + if ($sort_type == 0) { + $filename = "$rel_dir$dir_sep$base_name.func.$html_ext"; + } else { + $filename = "$rel_dir$dir_sep$base_name.func-sort-c.$html_ext"; + } + html_create(*HTML_HANDLE, $filename); + $pagetitle = "LCOV - $title - $trunc_dir$dir_sep$base_name - functions"; + write_html_prolog(*HTML_HANDLE, $base_dir, $pagetitle); + write_header(*HTML_HANDLE, 4, "$trunc_dir$dir_sep$base_name", + "$rel_dir$dir_sep$base_name", $lines_found, $lines_hit, + $fn_found, $fn_hit, $br_found, $br_hit, $sort_type); + write_function_table(*HTML_HANDLE, "$base_name.gcov.$html_ext", + $sumcount, $funcdata, + $sumfnccount, $testfncdata, $sumbrcount, + $testbrdata, $base_name, + $base_dir, $sort_type); + write_html_epilog(*HTML_HANDLE, $base_dir, 1); + close(*HTML_HANDLE); +} + + +# +# process_file(trunc_dir, rel_dir, filename) +# + +sub process_file($$$) +{ + info("Processing file ".apply_prefix($_[2], @dir_prefix)."\n"); + + my $trunc_dir = $_[0]; + my $rel_dir = $_[1]; + my $filename = $_[2]; + my $base_name = basename($filename); + my $base_dir = get_relative_base_path($rel_dir); + my $testdata; + my $testcount; + my $sumcount; + my $funcdata; + my $checkdata; + my $testfncdata; + my $sumfnccount; + my $testbrdata; + my $sumbrcount; + my $lines_found; + my $lines_hit; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + my $converted; + my @source; + my $pagetitle; + local *HTML_HANDLE; + + ($testdata, $sumcount, $funcdata, $checkdata, $testfncdata, + $sumfnccount, $testbrdata, $sumbrcount, $lines_found, $lines_hit, + $fn_found, $fn_hit, $br_found, $br_hit) + = get_info_entry($info_data{$filename}); + + # Return after this point in case user asked us not to generate + # source code view + if ($no_sourceview) + { + return ($lines_found, $lines_hit, $fn_found, $fn_hit, + $br_found, $br_hit, $testdata, $testfncdata, + $testbrdata); + } + + $converted = get_converted_lines($testdata); + # Generate source code view for this file + html_create(*HTML_HANDLE, "$rel_dir$dir_sep$base_name.gcov.$html_ext"); + $pagetitle = "LCOV - $test_title - $trunc_dir$dir_sep$base_name"; + write_html_prolog(*HTML_HANDLE, $base_dir, $pagetitle); + write_header(*HTML_HANDLE, 2, "$trunc_dir$dir_sep$base_name", + "$rel_dir$dir_sep$base_name", $lines_found, $lines_hit, + $fn_found, $fn_hit, $br_found, $br_hit, 0); + @source = write_source(*HTML_HANDLE, $filename, $sumcount, $checkdata, + $converted, $funcdata, $sumbrcount); + + write_html_epilog(*HTML_HANDLE, $base_dir, 1); + close(*HTML_HANDLE); + + if ($func_coverage) { + # Create function tables + foreach (@funcview_sortlist) { + write_function_page($base_dir, $rel_dir, $trunc_dir, + $base_name, $test_title, + $lines_found, $lines_hit, + $fn_found, $fn_hit, $br_found, + $br_hit, $sumcount, + $funcdata, $sumfnccount, + $testfncdata, $sumbrcount, + $testbrdata, $_); + } + } + + # Additional files are needed in case of frame output + if (!$frames) + { + return ($lines_found, $lines_hit, $fn_found, $fn_hit, + $br_found, $br_hit, $testdata, $testfncdata, + $testbrdata); + } + + # Create overview png file + gen_png("$rel_dir$dir_sep$base_name.gcov.png", $overview_width, $tab_size, + @source); + + # Create frameset page + html_create(*HTML_HANDLE, + "$rel_dir$dir_sep$base_name.gcov.frameset.$html_ext"); + write_frameset(*HTML_HANDLE, $base_dir, $base_name, $pagetitle); + close(*HTML_HANDLE); + + # Write overview frame + html_create(*HTML_HANDLE, + "$rel_dir$dir_sep$base_name.gcov.overview.$html_ext"); + write_overview(*HTML_HANDLE, $base_dir, $base_name, $pagetitle, + scalar(@source)); + close(*HTML_HANDLE); + + return ($lines_found, $lines_hit, $fn_found, $fn_hit, $br_found, + $br_hit, $testdata, $testfncdata, $testbrdata); +} + + +sub compress_brcount($) +{ + my ($brcount) = @_; + my $db; + + $db = brcount_to_db($brcount); + return db_to_brcount($db, $brcount); +} + + +# +# read_info_file(info_filename) +# +# Read in the contents of the .info file specified by INFO_FILENAME. Data will +# be returned as a reference to a hash containing the following mappings: +# +# %result: for each filename found in file -> \%data +# +# %data: "test" -> \%testdata +# "sum" -> \%sumcount +# "func" -> \%funcdata +# "found" -> $lines_found (number of instrumented lines found in file) +# "hit" -> $lines_hit (number of executed lines in file) +# "f_found" -> $fn_found (number of instrumented functions found in file) +# "f_hit" -> $fn_hit (number of executed functions in file) +# "b_found" -> $br_found (number of instrumented branches found in file) +# "b_hit" -> $br_hit (number of executed branches in file) +# "check" -> \%checkdata +# "testfnc" -> \%testfncdata +# "sumfnc" -> \%sumfnccount +# "testbr" -> \%testbrdata +# "sumbr" -> \%sumbrcount +# +# %testdata : name of test affecting this file -> \%testcount +# %testfncdata: name of test affecting this file -> \%testfnccount +# %testbrdata: name of test affecting this file -> \%testbrcount +# +# %testcount : line number -> execution count for a single test +# %testfnccount: function name -> execution count for a single test +# %testbrcount : line number -> branch coverage data for a single test +# %sumcount : line number -> execution count for all tests +# %sumfnccount : function name -> execution count for all tests +# %sumbrcount : line number -> branch coverage data for all tests +# %funcdata : function name -> line number +# %checkdata : line number -> checksum of source code line +# $brdata : vector of items: block, branch, taken +# +# Note that .info file sections referring to the same file and test name +# will automatically be combined by adding all execution counts. +# +# Note that if INFO_FILENAME ends with ".gz", it is assumed that the file +# is compressed using GZIP. If available, GUNZIP will be used to decompress +# this file. +# +# Die on error. +# + +sub read_info_file($) +{ + my $tracefile = $_[0]; # Name of tracefile + my %result; # Resulting hash: file -> data + my $data; # Data handle for current entry + my $testdata; # " " + my $testcount; # " " + my $sumcount; # " " + my $funcdata; # " " + my $checkdata; # " " + my $testfncdata; + my $testfnccount; + my $sumfnccount; + my $testbrdata; + my $testbrcount; + my $sumbrcount; + my $line; # Current line read from .info file + my $testname; # Current test name + my $filename; # Current filename + my $hitcount; # Count for lines hit + my $count; # Execution count of current line + my $negative; # If set, warn about negative counts + my $changed_testname; # If set, warn about changed testname + my $line_checksum; # Checksum of current line + my $notified_about_relative_paths; + local *INFO_HANDLE; # Filehandle for .info file + + info("Reading data file $tracefile\n"); + + # Check if file exists and is readable + stat($_[0]); + if (!(-r _)) + { + die("ERROR: cannot read file $_[0]!\n"); + } + + # Check if this is really a plain file + if (!(-f _)) + { + die("ERROR: not a plain file: $_[0]!\n"); + } + + # Check for .gz extension + if ($_[0] =~ /\.gz$/) + { + # Check for availability of GZIP tool + system_no_output(1, "gunzip" ,"-h") + and die("ERROR: gunzip command not available!\n"); + + # Check integrity of compressed file + system_no_output(1, "gunzip", "-t", $_[0]) + and die("ERROR: integrity check failed for ". + "compressed file $_[0]!\n"); + + # Open compressed file + open(INFO_HANDLE, "-|", "gunzip -c '$_[0]'") + or die("ERROR: cannot start gunzip to decompress ". + "file $_[0]!\n"); + } + else + { + # Open decompressed file + open(INFO_HANDLE, "<", $_[0]) + or die("ERROR: cannot read file $_[0]!\n"); + } + + $testname = ""; + while () + { + chomp($_); + $line = $_; + + # Switch statement + foreach ($line) + { + /^TN:([^,]*)(,diff)?/ && do + { + # Test name information found + $testname = defined($1) ? $1 : ""; + if ($testname =~ s/\W/_/g) + { + $changed_testname = 1; + } + $testname .= $2 if (defined($2)); + last; + }; + + /^[SK]F:(.*)/ && do + { + # Filename information found + # Retrieve data for new entry + $filename = File::Spec->rel2abs($1, $cwd); + + if (!File::Spec->file_name_is_absolute($1) && + !$notified_about_relative_paths) + { + info("Resolved relative source file ". + "path \"$1\" with CWD to ". + "\"$filename\".\n"); + $notified_about_relative_paths = 1; + } + + if($os_is_windows) + { + # Replace "forward" slashes with backslashes + $filename =~ s/\//\\/g; + } + + $data = $result{$filename}; + ($testdata, $sumcount, $funcdata, $checkdata, + $testfncdata, $sumfnccount, $testbrdata, + $sumbrcount) = + get_info_entry($data); + + if (defined($testname)) + { + $testcount = $testdata->{$testname}; + $testfnccount = $testfncdata->{$testname}; + $testbrcount = $testbrdata->{$testname}; + } + else + { + $testcount = {}; + $testfnccount = {}; + $testbrcount = {}; + } + last; + }; + + /^DA:(\d+),(-?\d+)(,[^,\s]+)?/ && do + { + # Fix negative counts + $count = $2 < 0 ? 0 : $2; + if ($2 < 0) + { + $negative = 1; + } + # Execution count found, add to structure + # Add summary counts + $sumcount->{$1} += $count; + + # Add test-specific counts + if (defined($testname)) + { + $testcount->{$1} += $count; + } + + # Store line checksum if available + if (defined($3)) + { + $line_checksum = substr($3, 1); + + # Does it match a previous definition + if (defined($checkdata->{$1}) && + ($checkdata->{$1} ne + $line_checksum)) + { + die("ERROR: checksum mismatch ". + "at $filename:$1\n"); + } + + $checkdata->{$1} = $line_checksum; + } + last; + }; + + /^FN:(\d+),([^,]+)/ && do + { + last if (!$func_coverage); + + # Function data found, add to structure + $funcdata->{$2} = $1; + + # Also initialize function call data + if (!defined($sumfnccount->{$2})) { + $sumfnccount->{$2} = 0; + } + if (defined($testname)) + { + if (!defined($testfnccount->{$2})) { + $testfnccount->{$2} = 0; + } + } + last; + }; + + /^FNDA:(\d+),([^,]+)/ && do + { + last if (!$func_coverage); + # Function call count found, add to structure + # Add summary counts + $sumfnccount->{$2} += $1; + + # Add test-specific counts + if (defined($testname)) + { + $testfnccount->{$2} += $1; + } + last; + }; + + /^BRDA:(\d+),(\d+),(\d+),(\d+|-)/ && do { + # Branch coverage data found + my ($line, $block, $branch, $taken) = + ($1, $2, $3, $4); + + last if (!$br_coverage); + $block = -1 if ($block == $UNNAMED_BLOCK); + $sumbrcount->{$line} .= + "$block,$branch,$taken:"; + + # Add test-specific counts + if (defined($testname)) { + $testbrcount->{$line} .= + "$block,$branch,$taken:"; + } + last; + }; + + /^end_of_record/ && do + { + # Found end of section marker + if ($filename) + { + # Store current section data + if (defined($testname)) + { + $testdata->{$testname} = + $testcount; + $testfncdata->{$testname} = + $testfnccount; + $testbrdata->{$testname} = + $testbrcount; + } + + set_info_entry($data, $testdata, + $sumcount, $funcdata, + $checkdata, $testfncdata, + $sumfnccount, + $testbrdata, + $sumbrcount); + $result{$filename} = $data; + last; + } + }; + + # default + last; + } + } + close(INFO_HANDLE); + + # Calculate lines_found and lines_hit for each file + foreach $filename (keys(%result)) + { + $data = $result{$filename}; + + ($testdata, $sumcount, undef, undef, $testfncdata, + $sumfnccount, $testbrdata, $sumbrcount) = + get_info_entry($data); + + # Filter out empty files + if (scalar(keys(%{$sumcount})) == 0) + { + delete($result{$filename}); + next; + } + # Filter out empty test cases + foreach $testname (keys(%{$testdata})) + { + if (!defined($testdata->{$testname}) || + scalar(keys(%{$testdata->{$testname}})) == 0) + { + delete($testdata->{$testname}); + delete($testfncdata->{$testname}); + } + } + + $data->{"found"} = scalar(keys(%{$sumcount})); + $hitcount = 0; + + foreach (keys(%{$sumcount})) + { + if ($sumcount->{$_} > 0) { $hitcount++; } + } + + $data->{"hit"} = $hitcount; + + # Get found/hit values for function call data + $data->{"f_found"} = scalar(keys(%{$sumfnccount})); + $hitcount = 0; + + foreach (keys(%{$sumfnccount})) { + if ($sumfnccount->{$_} > 0) { + $hitcount++; + } + } + $data->{"f_hit"} = $hitcount; + + # Combine branch data for the same branches + (undef, $data->{"b_found"}, $data->{"b_hit"}) = + compress_brcount($sumbrcount); + foreach $testname (keys(%{$testbrdata})) { + compress_brcount($testbrdata->{$testname}); + } + } + + if (scalar(keys(%result)) == 0) + { + die("ERROR: no valid records found in tracefile $tracefile\n"); + } + if ($negative) + { + warn("WARNING: negative counts found in tracefile ". + "$tracefile\n"); + } + if ($changed_testname) + { + warn("WARNING: invalid characters removed from testname in ". + "tracefile $tracefile\n"); + } + + return(\%result); +} + + +# +# get_info_entry(hash_ref) +# +# Retrieve data from an entry of the structure generated by read_info_file(). +# Return a list of references to hashes: +# (test data hash ref, sum count hash ref, funcdata hash ref, checkdata hash +# ref, testfncdata hash ref, sumfnccount hash ref, lines found, lines hit, +# functions found, functions hit) +# + +sub get_info_entry($) +{ + my $testdata_ref = $_[0]->{"test"}; + my $sumcount_ref = $_[0]->{"sum"}; + my $funcdata_ref = $_[0]->{"func"}; + my $checkdata_ref = $_[0]->{"check"}; + my $testfncdata = $_[0]->{"testfnc"}; + my $sumfnccount = $_[0]->{"sumfnc"}; + my $testbrdata = $_[0]->{"testbr"}; + my $sumbrcount = $_[0]->{"sumbr"}; + my $lines_found = $_[0]->{"found"}; + my $lines_hit = $_[0]->{"hit"}; + my $fn_found = $_[0]->{"f_found"}; + my $fn_hit = $_[0]->{"f_hit"}; + my $br_found = $_[0]->{"b_found"}; + my $br_hit = $_[0]->{"b_hit"}; + + return ($testdata_ref, $sumcount_ref, $funcdata_ref, $checkdata_ref, + $testfncdata, $sumfnccount, $testbrdata, $sumbrcount, + $lines_found, $lines_hit, $fn_found, $fn_hit, + $br_found, $br_hit); +} + + +# +# set_info_entry(hash_ref, testdata_ref, sumcount_ref, funcdata_ref, +# checkdata_ref, testfncdata_ref, sumfcncount_ref, +# testbrdata_ref, sumbrcount_ref[,lines_found, +# lines_hit, f_found, f_hit, $b_found, $b_hit]) +# +# Update the hash referenced by HASH_REF with the provided data references. +# + +sub set_info_entry($$$$$$$$$;$$$$$$) +{ + my $data_ref = $_[0]; + + $data_ref->{"test"} = $_[1]; + $data_ref->{"sum"} = $_[2]; + $data_ref->{"func"} = $_[3]; + $data_ref->{"check"} = $_[4]; + $data_ref->{"testfnc"} = $_[5]; + $data_ref->{"sumfnc"} = $_[6]; + $data_ref->{"testbr"} = $_[7]; + $data_ref->{"sumbr"} = $_[8]; + + if (defined($_[9])) { $data_ref->{"found"} = $_[9]; } + if (defined($_[10])) { $data_ref->{"hit"} = $_[10]; } + if (defined($_[11])) { $data_ref->{"f_found"} = $_[11]; } + if (defined($_[12])) { $data_ref->{"f_hit"} = $_[12]; } + if (defined($_[13])) { $data_ref->{"b_found"} = $_[13]; } + if (defined($_[14])) { $data_ref->{"b_hit"} = $_[14]; } +} + + +# +# add_counts(data1_ref, data2_ref) +# +# DATA1_REF and DATA2_REF are references to hashes containing a mapping +# +# line number -> execution count +# +# Return a list (RESULT_REF, LINES_FOUND, LINES_HIT) where RESULT_REF +# is a reference to a hash containing the combined mapping in which +# execution counts are added. +# + +sub add_counts($$) +{ + my $data1_ref = $_[0]; # Hash 1 + my $data2_ref = $_[1]; # Hash 2 + my %result; # Resulting hash + my $line; # Current line iteration scalar + my $data1_count; # Count of line in hash1 + my $data2_count; # Count of line in hash2 + my $found = 0; # Total number of lines found + my $hit = 0; # Number of lines with a count > 0 + + foreach $line (keys(%$data1_ref)) + { + $data1_count = $data1_ref->{$line}; + $data2_count = $data2_ref->{$line}; + + # Add counts if present in both hashes + if (defined($data2_count)) { $data1_count += $data2_count; } + + # Store sum in %result + $result{$line} = $data1_count; + + $found++; + if ($data1_count > 0) { $hit++; } + } + + # Add lines unique to data2_ref + foreach $line (keys(%$data2_ref)) + { + # Skip lines already in data1_ref + if (defined($data1_ref->{$line})) { next; } + + # Copy count from data2_ref + $result{$line} = $data2_ref->{$line}; + + $found++; + if ($result{$line} > 0) { $hit++; } + } + + return (\%result, $found, $hit); +} + + +# +# merge_checksums(ref1, ref2, filename) +# +# REF1 and REF2 are references to hashes containing a mapping +# +# line number -> checksum +# +# Merge checksum lists defined in REF1 and REF2 and return reference to +# resulting hash. Die if a checksum for a line is defined in both hashes +# but does not match. +# + +sub merge_checksums($$$) +{ + my $ref1 = $_[0]; + my $ref2 = $_[1]; + my $filename = $_[2]; + my %result; + my $line; + + foreach $line (keys(%{$ref1})) + { + if (defined($ref2->{$line}) && + ($ref1->{$line} ne $ref2->{$line})) + { + die("ERROR: checksum mismatch at $filename:$line\n"); + } + $result{$line} = $ref1->{$line}; + } + + foreach $line (keys(%{$ref2})) + { + $result{$line} = $ref2->{$line}; + } + + return \%result; +} + + +# +# merge_func_data(funcdata1, funcdata2, filename) +# + +sub merge_func_data($$$) +{ + my ($funcdata1, $funcdata2, $filename) = @_; + my %result; + my $func; + + if (defined($funcdata1)) { + %result = %{$funcdata1}; + } + + foreach $func (keys(%{$funcdata2})) { + my $line1 = $result{$func}; + my $line2 = $funcdata2->{$func}; + + if (defined($line1) && ($line1 != $line2)) { + warn("WARNING: function data mismatch at ". + "$filename:$line2\n"); + next; + } + $result{$func} = $line2; + } + + return \%result; +} + + +# +# add_fnccount(fnccount1, fnccount2) +# +# Add function call count data. Return list (fnccount_added, f_found, f_hit) +# + +sub add_fnccount($$) +{ + my ($fnccount1, $fnccount2) = @_; + my %result; + my $fn_found; + my $fn_hit; + my $function; + + if (defined($fnccount1)) { + %result = %{$fnccount1}; + } + foreach $function (keys(%{$fnccount2})) { + $result{$function} += $fnccount2->{$function}; + } + $fn_found = scalar(keys(%result)); + $fn_hit = 0; + foreach $function (keys(%result)) { + if ($result{$function} > 0) { + $fn_hit++; + } + } + + return (\%result, $fn_found, $fn_hit); +} + +# +# add_testfncdata(testfncdata1, testfncdata2) +# +# Add function call count data for several tests. Return reference to +# added_testfncdata. +# + +sub add_testfncdata($$) +{ + my ($testfncdata1, $testfncdata2) = @_; + my %result; + my $testname; + + foreach $testname (keys(%{$testfncdata1})) { + if (defined($testfncdata2->{$testname})) { + my $fnccount; + + # Function call count data for this testname exists + # in both data sets: add + ($fnccount) = add_fnccount( + $testfncdata1->{$testname}, + $testfncdata2->{$testname}); + $result{$testname} = $fnccount; + next; + } + # Function call count data for this testname is unique to + # data set 1: copy + $result{$testname} = $testfncdata1->{$testname}; + } + + # Add count data for testnames unique to data set 2 + foreach $testname (keys(%{$testfncdata2})) { + if (!defined($result{$testname})) { + $result{$testname} = $testfncdata2->{$testname}; + } + } + return \%result; +} + + +# +# brcount_to_db(brcount) +# +# Convert brcount data to the following format: +# +# db: line number -> block hash +# block hash: block number -> branch hash +# branch hash: branch number -> taken value +# + +sub brcount_to_db($) +{ + my ($brcount) = @_; + my $line; + my $db; + + # Add branches to database + foreach $line (keys(%{$brcount})) { + my $brdata = $brcount->{$line}; + + foreach my $entry (split(/:/, $brdata)) { + my ($block, $branch, $taken) = split(/,/, $entry); + my $old = $db->{$line}->{$block}->{$branch}; + + if (!defined($old) || $old eq "-") { + $old = $taken; + } elsif ($taken ne "-") { + $old += $taken; + } + + $db->{$line}->{$block}->{$branch} = $old; + } + } + + return $db; +} + + +# +# db_to_brcount(db[, brcount]) +# +# Convert branch coverage data back to brcount format. If brcount is specified, +# the converted data is directly inserted in brcount. +# + +sub db_to_brcount($;$) +{ + my ($db, $brcount) = @_; + my $line; + my $br_found = 0; + my $br_hit = 0; + + # Convert database back to brcount format + foreach $line (sort({$a <=> $b} keys(%{$db}))) { + my $ldata = $db->{$line}; + my $brdata; + my $block; + + foreach $block (sort({$a <=> $b} keys(%{$ldata}))) { + my $bdata = $ldata->{$block}; + my $branch; + + foreach $branch (sort({$a <=> $b} keys(%{$bdata}))) { + my $taken = $bdata->{$branch}; + + $br_found++; + $br_hit++ if ($taken ne "-" && $taken > 0); + $brdata .= "$block,$branch,$taken:"; + } + } + $brcount->{$line} = $brdata; + } + + return ($brcount, $br_found, $br_hit); +} + + +# +# brcount_db_combine(db1, db2, op) +# +# db1 := db1 op db2, where +# db1, db2: brcount data as returned by brcount_to_db +# op: one of $BR_ADD and BR_SUB +# +sub brcount_db_combine($$$) +{ + my ($db1, $db2, $op) = @_; + + foreach my $line (keys(%{$db2})) { + my $ldata = $db2->{$line}; + + foreach my $block (keys(%{$ldata})) { + my $bdata = $ldata->{$block}; + + foreach my $branch (keys(%{$bdata})) { + my $taken = $bdata->{$branch}; + my $new = $db1->{$line}->{$block}->{$branch}; + + if (!defined($new) || $new eq "-") { + $new = $taken; + } elsif ($taken ne "-") { + if ($op == $BR_ADD) { + $new += $taken; + } elsif ($op == $BR_SUB) { + $new -= $taken; + $new = 0 if ($new < 0); + } + } + + $db1->{$line}->{$block}->{$branch} = $new; + } + } + } +} + + +# +# brcount_db_get_found_and_hit(db) +# +# Return (br_found, br_hit) for db. +# + +sub brcount_db_get_found_and_hit($) +{ + my ($db) = @_; + my ($br_found , $br_hit) = (0, 0); + + foreach my $line (keys(%{$db})) { + my $ldata = $db->{$line}; + + foreach my $block (keys(%{$ldata})) { + my $bdata = $ldata->{$block}; + + foreach my $branch (keys(%{$bdata})) { + my $taken = $bdata->{$branch}; + + $br_found++; + $br_hit++ if ($taken ne "-" && $taken > 0); + } + } + } + + return ($br_found, $br_hit); +} + + +# combine_brcount(brcount1, brcount2, type, inplace) +# +# If add is BR_ADD, add branch coverage data and return list brcount_added. +# If add is BR_SUB, subtract the taken values of brcount2 from brcount1 and +# return brcount_sub. If inplace is set, the result is inserted into brcount1. +# + +sub combine_brcount($$$;$) +{ + my ($brcount1, $brcount2, $type, $inplace) = @_; + my ($db1, $db2); + + $db1 = brcount_to_db($brcount1); + $db2 = brcount_to_db($brcount2); + brcount_db_combine($db1, $db2, $type); + + return db_to_brcount($db1, $inplace ? $brcount1 : undef); +} + + +# +# add_testbrdata(testbrdata1, testbrdata2) +# +# Add branch coverage data for several tests. Return reference to +# added_testbrdata. +# + +sub add_testbrdata($$) +{ + my ($testbrdata1, $testbrdata2) = @_; + my %result; + my $testname; + + foreach $testname (keys(%{$testbrdata1})) { + if (defined($testbrdata2->{$testname})) { + my $brcount; + + # Branch coverage data for this testname exists + # in both data sets: add + ($brcount) = combine_brcount($testbrdata1->{$testname}, + $testbrdata2->{$testname}, $BR_ADD); + $result{$testname} = $brcount; + next; + } + # Branch coverage data for this testname is unique to + # data set 1: copy + $result{$testname} = $testbrdata1->{$testname}; + } + + # Add count data for testnames unique to data set 2 + foreach $testname (keys(%{$testbrdata2})) { + if (!defined($result{$testname})) { + $result{$testname} = $testbrdata2->{$testname}; + } + } + return \%result; +} + + +# +# combine_info_entries(entry_ref1, entry_ref2, filename) +# +# Combine .info data entry hashes referenced by ENTRY_REF1 and ENTRY_REF2. +# Return reference to resulting hash. +# + +sub combine_info_entries($$$) +{ + my $entry1 = $_[0]; # Reference to hash containing first entry + my $testdata1; + my $sumcount1; + my $funcdata1; + my $checkdata1; + my $testfncdata1; + my $sumfnccount1; + my $testbrdata1; + my $sumbrcount1; + + my $entry2 = $_[1]; # Reference to hash containing second entry + my $testdata2; + my $sumcount2; + my $funcdata2; + my $checkdata2; + my $testfncdata2; + my $sumfnccount2; + my $testbrdata2; + my $sumbrcount2; + + my %result; # Hash containing combined entry + my %result_testdata; + my $result_sumcount = {}; + my $result_funcdata; + my $result_testfncdata; + my $result_sumfnccount; + my $result_testbrdata; + my $result_sumbrcount; + my $lines_found; + my $lines_hit; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + + my $testname; + my $filename = $_[2]; + + # Retrieve data + ($testdata1, $sumcount1, $funcdata1, $checkdata1, $testfncdata1, + $sumfnccount1, $testbrdata1, $sumbrcount1) = get_info_entry($entry1); + ($testdata2, $sumcount2, $funcdata2, $checkdata2, $testfncdata2, + $sumfnccount2, $testbrdata2, $sumbrcount2) = get_info_entry($entry2); + + # Merge checksums + $checkdata1 = merge_checksums($checkdata1, $checkdata2, $filename); + + # Combine funcdata + $result_funcdata = merge_func_data($funcdata1, $funcdata2, $filename); + + # Combine function call count data + $result_testfncdata = add_testfncdata($testfncdata1, $testfncdata2); + ($result_sumfnccount, $fn_found, $fn_hit) = + add_fnccount($sumfnccount1, $sumfnccount2); + + # Combine branch coverage data + $result_testbrdata = add_testbrdata($testbrdata1, $testbrdata2); + ($result_sumbrcount, $br_found, $br_hit) = + combine_brcount($sumbrcount1, $sumbrcount2, $BR_ADD); + + # Combine testdata + foreach $testname (keys(%{$testdata1})) + { + if (defined($testdata2->{$testname})) + { + # testname is present in both entries, requires + # combination + ($result_testdata{$testname}) = + add_counts($testdata1->{$testname}, + $testdata2->{$testname}); + } + else + { + # testname only present in entry1, add to result + $result_testdata{$testname} = $testdata1->{$testname}; + } + + # update sum count hash + ($result_sumcount, $lines_found, $lines_hit) = + add_counts($result_sumcount, + $result_testdata{$testname}); + } + + foreach $testname (keys(%{$testdata2})) + { + # Skip testnames already covered by previous iteration + if (defined($testdata1->{$testname})) { next; } + + # testname only present in entry2, add to result hash + $result_testdata{$testname} = $testdata2->{$testname}; + + # update sum count hash + ($result_sumcount, $lines_found, $lines_hit) = + add_counts($result_sumcount, + $result_testdata{$testname}); + } + + # Calculate resulting sumcount + + # Store result + set_info_entry(\%result, \%result_testdata, $result_sumcount, + $result_funcdata, $checkdata1, $result_testfncdata, + $result_sumfnccount, $result_testbrdata, + $result_sumbrcount, $lines_found, $lines_hit, + $fn_found, $fn_hit, $br_found, $br_hit); + + return(\%result); +} + + +# +# combine_info_files(info_ref1, info_ref2) +# +# Combine .info data in hashes referenced by INFO_REF1 and INFO_REF2. Return +# reference to resulting hash. +# + +sub combine_info_files($$) +{ + my %hash1 = %{$_[0]}; + my %hash2 = %{$_[1]}; + my $filename; + + foreach $filename (keys(%hash2)) + { + if ($hash1{$filename}) + { + # Entry already exists in hash1, combine them + $hash1{$filename} = + combine_info_entries($hash1{$filename}, + $hash2{$filename}, + $filename); + } + else + { + # Entry is unique in both hashes, simply add to + # resulting hash + $hash1{$filename} = $hash2{$filename}; + } + } + + return(\%hash1); +} + + +# +# get_prefix(min_dir, filename_list) +# +# Search FILENAME_LIST for a directory prefix which is common to as many +# list entries as possible, so that removing this prefix will minimize the +# sum of the lengths of all resulting shortened filenames while observing +# that no filename has less than MIN_DIR parent directories. +# + +sub get_prefix($@) +{ + my ($min_dir, @filename_list) = @_; + my %prefix; # mapping: prefix -> sum of lengths + my $current; # Temporary iteration variable + + # Find list of prefixes + foreach (@filename_list) + { + # Need explicit assignment to get a copy of $_ so that + # shortening the contained prefix does not affect the list + $current = $_; + while ($current = shorten_prefix($current)) + { + $current .= $dir_sep; + + # Skip rest if the remaining prefix has already been + # added to hash + if (exists($prefix{$current})) { last; } + + # Initialize with 0 + $prefix{$current}="0"; + } + + } + + # Remove all prefixes that would cause filenames to have less than + # the minimum number of parent directories + foreach my $filename (@filename_list) { + my $dir = dirname($filename); + + for (my $i = 0; $i < $min_dir; $i++) { + delete($prefix{$dir.$dir_sep}); + $dir = shorten_prefix($dir); + } + } + + # Check if any prefix remains + return undef if (!%prefix); + + # Calculate sum of lengths for all prefixes + foreach $current (keys(%prefix)) + { + foreach (@filename_list) + { + # Add original length + $prefix{$current} += length($_); + + # Check whether prefix matches + if (substr($_, 0, length($current)) eq $current) + { + # Subtract prefix length for this filename + $prefix{$current} -= length($current); + } + } + } + + # Find and return prefix with minimal sum + $current = (keys(%prefix))[0]; + + foreach (keys(%prefix)) + { + if ($prefix{$_} < $prefix{$current}) + { + $current = $_; + } + } + + $current =~ s/\Q$dir_sep\E$//; + + return($current); +} + + +# +# shorten_prefix(prefix) +# +# Return PREFIX shortened by last directory component. +# + +sub shorten_prefix($) +{ + my @list = split("\\".$dir_sep, $_[0]); + + pop(@list); + return join($dir_sep, @list); +} + + + +# +# get_dir_list(filename_list) +# +# Return sorted list of directories for each entry in given FILENAME_LIST. +# + +sub get_dir_list(@) +{ + my %result; + + foreach (@_) + { + $result{shorten_prefix($_)} = ""; + } + + return(sort(keys(%result))); +} + + +# +# get_relative_base_path(subdirectory) +# +# Return a relative path string which references the base path when applied +# in SUBDIRECTORY. +# +# Example: get_relative_base_path("fs/mm") -> "../../" +# + +sub get_relative_base_path($) +{ + my $result = ""; + my $index; + + # Make an empty directory path a special case + if (!$_[0]) { return(""); } + + # Count number of /s in path + $index = () = $_[0] =~ /\Q$dir_sep\E/g; + + # Add a ../ to $result for each / in the directory path + 1 + for (; $index>=0; $index--) + { + $result .= "../"; + } + + return $result; +} + + +# +# read_testfile(test_filename) +# +# Read in file TEST_FILENAME which contains test descriptions in the format: +# +# TN: +# TD: +# +# for each test case. Return a reference to a hash containing a mapping +# +# test name -> test description. +# +# Die on error. +# + +sub read_testfile($) +{ + my %result; + my $test_name; + my $changed_testname; + local *TEST_HANDLE; + + open(TEST_HANDLE, "<", $_[0]) + or die("ERROR: cannot open $_[0]!\n"); + + while () + { + chomp($_); + + # Match lines beginning with TN: + if (/^TN:\s+(.*?)\s*$/) + { + # Store name for later use + $test_name = $1; + if ($test_name =~ s/\W/_/g) + { + $changed_testname = 1; + } + } + + # Match lines beginning with TD: + if (/^TD:\s+(.*?)\s*$/) + { + if (!defined($test_name)) { + die("ERROR: Found test description without prior test name in $_[0]:$.\n"); + } + # Check for empty line + if ($1) + { + # Add description to hash + $result{$test_name} .= " $1"; + } + else + { + # Add empty line + $result{$test_name} .= "\n\n"; + } + } + } + + close(TEST_HANDLE); + + if ($changed_testname) + { + warn("WARNING: invalid characters removed from testname in ". + "descriptions file $_[0]\n"); + } + + return \%result; +} + + +# +# escape_html(STRING) +# +# Return a copy of STRING in which all occurrences of HTML special characters +# are escaped. +# + +sub escape_html($) +{ + my $string = $_[0]; + + if (!$string) { return ""; } + + $string =~ s/&/&/g; # & -> & + $string =~ s/ < + $string =~ s/>/>/g; # > -> > + $string =~ s/\"/"/g; # " -> " + + while ($string =~ /^([^\t]*)(\t)/) + { + my $replacement = " "x($tab_size - (length($1) % $tab_size)); + $string =~ s/^([^\t]*)(\t)/$1$replacement/; + } + + $string =~ s/\n/
/g; # \n ->
+ + return $string; +} + + +# +# get_date_string() +# +# Return the current date in the form: yyyy-mm-dd +# + +sub get_date_string() +{ + my $year; + my $month; + my $day; + my $hour; + my $min; + my $sec; + my @timeresult; + + if (defined $ENV{'SOURCE_DATE_EPOCH'}) + { + @timeresult = gmtime($ENV{'SOURCE_DATE_EPOCH'}); + } + else + { + @timeresult = localtime(); + } + ($year, $month, $day, $hour, $min, $sec) = + @timeresult[5, 4, 3, 2, 1, 0]; + + return sprintf("%d-%02d-%02d %02d:%02d:%02d", $year+1900, $month+1, + $day, $hour, $min, $sec); +} + + +# +# create_sub_dir(dir_name) +# +# Create subdirectory DIR_NAME if it does not already exist, including all its +# parent directories. +# +# Die on error. +# + +sub create_sub_dir($) +{ + my ($dir) = @_; + stat($dir); + + if ( !-e _ ) + { + if ($os_is_windows) + { + system("mkdir", $dir) + and die("ERROR: cannot create directory '$dir'!\n"); + } + else + { + system("mkdir", "-p", $dir) + and die("ERROR: cannot create directory $dir!\n"); + } + } +} + + +# +# write_description_file(descriptions, overall_found, overall_hit, +# total_fn_found, total_fn_hit, total_br_found, +# total_br_hit) +# +# Write HTML file containing all test case descriptions. DESCRIPTIONS is a +# reference to a hash containing a mapping +# +# test case name -> test case description +# +# Die on error. +# + +sub write_description_file($$$$$$$) +{ + my %description = %{$_[0]}; + my $found = $_[1]; + my $hit = $_[2]; + my $fn_found = $_[3]; + my $fn_hit = $_[4]; + my $br_found = $_[5]; + my $br_hit = $_[6]; + my $test_name; + local *HTML_HANDLE; + + html_create(*HTML_HANDLE,"descriptions.$html_ext"); + write_html_prolog(*HTML_HANDLE, "", "LCOV - test case descriptions"); + write_header(*HTML_HANDLE, 3, "", "", $found, $hit, $fn_found, + $fn_hit, $br_found, $br_hit, 0); + + write_test_table_prolog(*HTML_HANDLE, + "Test case descriptions - alphabetical list"); + + foreach $test_name (sort(keys(%description))) + { + my $desc = $description{$test_name}; + + $desc = escape_html($desc) if (!$rc_desc_html); + write_test_table_entry(*HTML_HANDLE, $test_name, $desc); + } + + write_test_table_epilog(*HTML_HANDLE); + write_html_epilog(*HTML_HANDLE, ""); + + close(*HTML_HANDLE); +} + + + +# +# write_png_files() +# +# Create all necessary .png files for the HTML-output in the current +# directory. .png-files are used as bar graphs. +# +# Die on error. +# + +sub write_png_files() +{ + my %data; + local *PNG_HANDLE; + + $data{"ruby.png"} = + [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, + 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x01, 0x01, 0x03, 0x00, 0x00, 0x00, 0x25, + 0xdb, 0x56, 0xca, 0x00, 0x00, 0x00, 0x07, 0x74, 0x49, 0x4d, + 0x45, 0x07, 0xd2, 0x07, 0x11, 0x0f, 0x18, 0x10, 0x5d, 0x57, + 0x34, 0x6e, 0x00, 0x00, 0x00, 0x09, 0x70, 0x48, 0x59, 0x73, + 0x00, 0x00, 0x0b, 0x12, 0x00, 0x00, 0x0b, 0x12, 0x01, 0xd2, + 0xdd, 0x7e, 0xfc, 0x00, 0x00, 0x00, 0x04, 0x67, 0x41, 0x4d, + 0x41, 0x00, 0x00, 0xb1, 0x8f, 0x0b, 0xfc, 0x61, 0x05, 0x00, + 0x00, 0x00, 0x06, 0x50, 0x4c, 0x54, 0x45, 0xff, 0x35, 0x2f, + 0x00, 0x00, 0x00, 0xd0, 0x33, 0x9a, 0x9d, 0x00, 0x00, 0x00, + 0x0a, 0x49, 0x44, 0x41, 0x54, 0x78, 0xda, 0x63, 0x60, 0x00, + 0x00, 0x00, 0x02, 0x00, 0x01, 0xe5, 0x27, 0xde, 0xfc, 0x00, + 0x00, 0x00, 0x00, 0x49, 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, + 0x82]; + $data{"amber.png"} = + [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, + 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x01, 0x01, 0x03, 0x00, 0x00, 0x00, 0x25, + 0xdb, 0x56, 0xca, 0x00, 0x00, 0x00, 0x07, 0x74, 0x49, 0x4d, + 0x45, 0x07, 0xd2, 0x07, 0x11, 0x0f, 0x28, 0x04, 0x98, 0xcb, + 0xd6, 0xe0, 0x00, 0x00, 0x00, 0x09, 0x70, 0x48, 0x59, 0x73, + 0x00, 0x00, 0x0b, 0x12, 0x00, 0x00, 0x0b, 0x12, 0x01, 0xd2, + 0xdd, 0x7e, 0xfc, 0x00, 0x00, 0x00, 0x04, 0x67, 0x41, 0x4d, + 0x41, 0x00, 0x00, 0xb1, 0x8f, 0x0b, 0xfc, 0x61, 0x05, 0x00, + 0x00, 0x00, 0x06, 0x50, 0x4c, 0x54, 0x45, 0xff, 0xe0, 0x50, + 0x00, 0x00, 0x00, 0xa2, 0x7a, 0xda, 0x7e, 0x00, 0x00, 0x00, + 0x0a, 0x49, 0x44, 0x41, 0x54, 0x78, 0xda, 0x63, 0x60, 0x00, + 0x00, 0x00, 0x02, 0x00, 0x01, 0xe5, 0x27, 0xde, 0xfc, 0x00, + 0x00, 0x00, 0x00, 0x49, 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, + 0x82]; + $data{"emerald.png"} = + [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, + 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x01, 0x01, 0x03, 0x00, 0x00, 0x00, 0x25, + 0xdb, 0x56, 0xca, 0x00, 0x00, 0x00, 0x07, 0x74, 0x49, 0x4d, + 0x45, 0x07, 0xd2, 0x07, 0x11, 0x0f, 0x22, 0x2b, 0xc9, 0xf5, + 0x03, 0x33, 0x00, 0x00, 0x00, 0x09, 0x70, 0x48, 0x59, 0x73, + 0x00, 0x00, 0x0b, 0x12, 0x00, 0x00, 0x0b, 0x12, 0x01, 0xd2, + 0xdd, 0x7e, 0xfc, 0x00, 0x00, 0x00, 0x04, 0x67, 0x41, 0x4d, + 0x41, 0x00, 0x00, 0xb1, 0x8f, 0x0b, 0xfc, 0x61, 0x05, 0x00, + 0x00, 0x00, 0x06, 0x50, 0x4c, 0x54, 0x45, 0x1b, 0xea, 0x59, + 0x0a, 0x0a, 0x0a, 0x0f, 0xba, 0x50, 0x83, 0x00, 0x00, 0x00, + 0x0a, 0x49, 0x44, 0x41, 0x54, 0x78, 0xda, 0x63, 0x60, 0x00, + 0x00, 0x00, 0x02, 0x00, 0x01, 0xe5, 0x27, 0xde, 0xfc, 0x00, + 0x00, 0x00, 0x00, 0x49, 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, + 0x82]; + $data{"snow.png"} = + [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, + 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x01, 0x01, 0x03, 0x00, 0x00, 0x00, 0x25, + 0xdb, 0x56, 0xca, 0x00, 0x00, 0x00, 0x07, 0x74, 0x49, 0x4d, + 0x45, 0x07, 0xd2, 0x07, 0x11, 0x0f, 0x1e, 0x1d, 0x75, 0xbc, + 0xef, 0x55, 0x00, 0x00, 0x00, 0x09, 0x70, 0x48, 0x59, 0x73, + 0x00, 0x00, 0x0b, 0x12, 0x00, 0x00, 0x0b, 0x12, 0x01, 0xd2, + 0xdd, 0x7e, 0xfc, 0x00, 0x00, 0x00, 0x04, 0x67, 0x41, 0x4d, + 0x41, 0x00, 0x00, 0xb1, 0x8f, 0x0b, 0xfc, 0x61, 0x05, 0x00, + 0x00, 0x00, 0x06, 0x50, 0x4c, 0x54, 0x45, 0xff, 0xff, 0xff, + 0x00, 0x00, 0x00, 0x55, 0xc2, 0xd3, 0x7e, 0x00, 0x00, 0x00, + 0x0a, 0x49, 0x44, 0x41, 0x54, 0x78, 0xda, 0x63, 0x60, 0x00, + 0x00, 0x00, 0x02, 0x00, 0x01, 0xe5, 0x27, 0xde, 0xfc, 0x00, + 0x00, 0x00, 0x00, 0x49, 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, + 0x82]; + $data{"glass.png"} = + [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, + 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x01, 0x01, 0x03, 0x00, 0x00, 0x00, 0x25, + 0xdb, 0x56, 0xca, 0x00, 0x00, 0x00, 0x04, 0x67, 0x41, 0x4d, + 0x41, 0x00, 0x00, 0xb1, 0x8f, 0x0b, 0xfc, 0x61, 0x05, 0x00, + 0x00, 0x00, 0x06, 0x50, 0x4c, 0x54, 0x45, 0xff, 0xff, 0xff, + 0x00, 0x00, 0x00, 0x55, 0xc2, 0xd3, 0x7e, 0x00, 0x00, 0x00, + 0x01, 0x74, 0x52, 0x4e, 0x53, 0x00, 0x40, 0xe6, 0xd8, 0x66, + 0x00, 0x00, 0x00, 0x01, 0x62, 0x4b, 0x47, 0x44, 0x00, 0x88, + 0x05, 0x1d, 0x48, 0x00, 0x00, 0x00, 0x09, 0x70, 0x48, 0x59, + 0x73, 0x00, 0x00, 0x0b, 0x12, 0x00, 0x00, 0x0b, 0x12, 0x01, + 0xd2, 0xdd, 0x7e, 0xfc, 0x00, 0x00, 0x00, 0x07, 0x74, 0x49, + 0x4d, 0x45, 0x07, 0xd2, 0x07, 0x13, 0x0f, 0x08, 0x19, 0xc4, + 0x40, 0x56, 0x10, 0x00, 0x00, 0x00, 0x0a, 0x49, 0x44, 0x41, + 0x54, 0x78, 0x9c, 0x63, 0x60, 0x00, 0x00, 0x00, 0x02, 0x00, + 0x01, 0x48, 0xaf, 0xa4, 0x71, 0x00, 0x00, 0x00, 0x00, 0x49, + 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82]; + $data{"updown.png"} = + [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, + 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x0a, + 0x00, 0x00, 0x00, 0x0e, 0x08, 0x06, 0x00, 0x00, 0x00, 0x16, + 0xa3, 0x8d, 0xab, 0x00, 0x00, 0x00, 0x3c, 0x49, 0x44, 0x41, + 0x54, 0x28, 0xcf, 0x63, 0x60, 0x40, 0x03, 0xff, 0xa1, 0x00, + 0x5d, 0x9c, 0x11, 0x5d, 0x11, 0x8a, 0x24, 0x23, 0x23, 0x23, + 0x86, 0x42, 0x6c, 0xa6, 0x20, 0x2b, 0x66, 0xc4, 0xa7, 0x08, + 0x59, 0x31, 0x23, 0x21, 0x45, 0x30, 0xc0, 0xc4, 0x30, 0x60, + 0x80, 0xfa, 0x6e, 0x24, 0x3e, 0x78, 0x48, 0x0a, 0x70, 0x62, + 0xa2, 0x90, 0x81, 0xd8, 0x44, 0x01, 0x00, 0xe9, 0x5c, 0x2f, + 0xf5, 0xe2, 0x9d, 0x0f, 0xf9, 0x00, 0x00, 0x00, 0x00, 0x49, + 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82] if ($sort); + foreach (keys(%data)) + { + open(PNG_HANDLE, ">", $_) + or die("ERROR: cannot create $_!\n"); + binmode(PNG_HANDLE); + print(PNG_HANDLE map(chr,@{$data{$_}})); + close(PNG_HANDLE); + } +} + + +# +# write_htaccess_file() +# + +sub write_htaccess_file() +{ + local *HTACCESS_HANDLE; + my $htaccess_data; + + open(*HTACCESS_HANDLE, ">", ".htaccess") + or die("ERROR: cannot open .htaccess for writing!\n"); + + $htaccess_data = (<<"END_OF_HTACCESS") +AddEncoding x-gzip .html +END_OF_HTACCESS + ; + + print(HTACCESS_HANDLE $htaccess_data); + close(*HTACCESS_HANDLE); +} + + +# +# write_css_file() +# +# Write the cascading style sheet file gcov.css to the current directory. +# This file defines basic layout attributes of all generated HTML pages. +# + +sub write_css_file() +{ + local *CSS_HANDLE; + + # Check for a specified external style sheet file + if ($css_filename) + { + # Simply copy that file + system("cp", $css_filename, "gcov.css") + and die("ERROR: cannot copy file $css_filename!\n"); + return; + } + + open(CSS_HANDLE, ">", "gcov.css") + or die ("ERROR: cannot open gcov.css for writing!\n"); + + + # ************************************************************* + + my $css_data = ($_=<<"END_OF_CSS") + /* All views: initial background and text color */ + body + { + color: #000000; + background-color: #FFFFFF; + } + + /* All views: standard link format*/ + a:link + { + color: #284FA8; + text-decoration: underline; + } + + /* All views: standard link - visited format */ + a:visited + { + color: #00CB40; + text-decoration: underline; + } + + /* All views: standard link - activated format */ + a:active + { + color: #FF0040; + text-decoration: underline; + } + + /* All views: main title format */ + td.title + { + text-align: center; + padding-bottom: 10px; + font-family: sans-serif; + font-size: 20pt; + font-style: italic; + font-weight: bold; + } + + /* All views: header item format */ + td.headerItem + { + text-align: right; + padding-right: 6px; + font-family: sans-serif; + font-weight: bold; + vertical-align: top; + white-space: nowrap; + } + + /* All views: header item value format */ + td.headerValue + { + text-align: left; + color: #284FA8; + font-family: sans-serif; + font-weight: bold; + white-space: nowrap; + } + + /* All views: header item coverage table heading */ + td.headerCovTableHead + { + text-align: center; + padding-right: 6px; + padding-left: 6px; + padding-bottom: 0px; + font-family: sans-serif; + font-size: 80%; + white-space: nowrap; + } + + /* All views: header item coverage table entry */ + td.headerCovTableEntry + { + text-align: right; + color: #284FA8; + font-family: sans-serif; + font-weight: bold; + white-space: nowrap; + padding-left: 12px; + padding-right: 4px; + background-color: #DAE7FE; + } + + /* All views: header item coverage table entry for high coverage rate */ + td.headerCovTableEntryHi + { + text-align: right; + color: #000000; + font-family: sans-serif; + font-weight: bold; + white-space: nowrap; + padding-left: 12px; + padding-right: 4px; + background-color: #A7FC9D; + } + + /* All views: header item coverage table entry for medium coverage rate */ + td.headerCovTableEntryMed + { + text-align: right; + color: #000000; + font-family: sans-serif; + font-weight: bold; + white-space: nowrap; + padding-left: 12px; + padding-right: 4px; + background-color: #FFEA20; + } + + /* All views: header item coverage table entry for ow coverage rate */ + td.headerCovTableEntryLo + { + text-align: right; + color: #000000; + font-family: sans-serif; + font-weight: bold; + white-space: nowrap; + padding-left: 12px; + padding-right: 4px; + background-color: #FF0000; + } + + /* All views: header legend value for legend entry */ + td.headerValueLeg + { + text-align: left; + color: #000000; + font-family: sans-serif; + font-size: 80%; + white-space: nowrap; + padding-top: 4px; + } + + /* All views: color of horizontal ruler */ + td.ruler + { + background-color: #6688D4; + } + + /* All views: version string format */ + td.versionInfo + { + text-align: center; + padding-top: 2px; + font-family: sans-serif; + font-style: italic; + } + + /* Directory view/File view (all)/Test case descriptions: + table headline format */ + td.tableHead + { + text-align: center; + color: #FFFFFF; + background-color: #6688D4; + font-family: sans-serif; + font-size: 120%; + font-weight: bold; + white-space: nowrap; + padding-left: 4px; + padding-right: 4px; + } + + span.tableHeadSort + { + padding-right: 4px; + } + + /* Directory view/File view (all): filename entry format */ + td.coverFile + { + text-align: left; + padding-left: 10px; + padding-right: 20px; + color: #284FA8; + background-color: #DAE7FE; + font-family: monospace; + } + + /* Directory view/File view (all): bar-graph entry format*/ + td.coverBar + { + padding-left: 10px; + padding-right: 10px; + background-color: #DAE7FE; + } + + /* Directory view/File view (all): bar-graph outline color */ + td.coverBarOutline + { + background-color: #000000; + } + + /* Directory view/File view (all): percentage entry for files with + high coverage rate */ + td.coverPerHi + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #A7FC9D; + font-weight: bold; + font-family: sans-serif; + } + + /* Directory view/File view (all): line count entry for files with + high coverage rate */ + td.coverNumHi + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #A7FC9D; + white-space: nowrap; + font-family: sans-serif; + } + + /* Directory view/File view (all): percentage entry for files with + medium coverage rate */ + td.coverPerMed + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #FFEA20; + font-weight: bold; + font-family: sans-serif; + } + + /* Directory view/File view (all): line count entry for files with + medium coverage rate */ + td.coverNumMed + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #FFEA20; + white-space: nowrap; + font-family: sans-serif; + } + + /* Directory view/File view (all): percentage entry for files with + low coverage rate */ + td.coverPerLo + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #FF0000; + font-weight: bold; + font-family: sans-serif; + } + + /* Directory view/File view (all): line count entry for files with + low coverage rate */ + td.coverNumLo + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #FF0000; + white-space: nowrap; + font-family: sans-serif; + } + + /* File view (all): "show/hide details" link format */ + a.detail:link + { + color: #B8D0FF; + font-size:80%; + } + + /* File view (all): "show/hide details" link - visited format */ + a.detail:visited + { + color: #B8D0FF; + font-size:80%; + } + + /* File view (all): "show/hide details" link - activated format */ + a.detail:active + { + color: #FFFFFF; + font-size:80%; + } + + /* File view (detail): test name entry */ + td.testName + { + text-align: right; + padding-right: 10px; + background-color: #DAE7FE; + font-family: sans-serif; + } + + /* File view (detail): test percentage entry */ + td.testPer + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #DAE7FE; + font-family: sans-serif; + } + + /* File view (detail): test lines count entry */ + td.testNum + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #DAE7FE; + font-family: sans-serif; + } + + /* Test case descriptions: test name format*/ + dt + { + font-family: sans-serif; + font-weight: bold; + } + + /* Test case descriptions: description table body */ + td.testDescription + { + padding-top: 10px; + padding-left: 30px; + padding-bottom: 10px; + padding-right: 30px; + background-color: #DAE7FE; + } + + /* Source code view: function entry */ + td.coverFn + { + text-align: left; + padding-left: 10px; + padding-right: 20px; + color: #284FA8; + background-color: #DAE7FE; + font-family: monospace; + } + + /* Source code view: function entry zero count*/ + td.coverFnLo + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #FF0000; + font-weight: bold; + font-family: sans-serif; + } + + /* Source code view: function entry nonzero count*/ + td.coverFnHi + { + text-align: right; + padding-left: 10px; + padding-right: 10px; + background-color: #DAE7FE; + font-weight: bold; + font-family: sans-serif; + } + + /* Source code view: source code format */ + pre.source + { + font-family: monospace; + white-space: pre; + margin-top: 2px; + } + + /* Source code view: line number format */ + span.lineNum + { + background-color: #EFE383; + } + + /* Source code view: format for lines which were executed */ + td.lineCov, + span.lineCov + { + background-color: #CAD7FE; + } + + /* Source code view: format for Cov legend */ + span.coverLegendCov + { + padding-left: 10px; + padding-right: 10px; + padding-bottom: 2px; + background-color: #CAD7FE; + } + + /* Source code view: format for lines which were not executed */ + td.lineNoCov, + span.lineNoCov + { + background-color: #FF6230; + } + + /* Source code view: format for NoCov legend */ + span.coverLegendNoCov + { + padding-left: 10px; + padding-right: 10px; + padding-bottom: 2px; + background-color: #FF6230; + } + + /* Source code view (function table): standard link - visited format */ + td.lineNoCov > a:visited, + td.lineCov > a:visited + { + color: black; + text-decoration: underline; + } + + /* Source code view: format for lines which were executed only in a + previous version */ + span.lineDiffCov + { + background-color: #B5F7AF; + } + + /* Source code view: format for branches which were executed + * and taken */ + span.branchCov + { + background-color: #CAD7FE; + } + + /* Source code view: format for branches which were executed + * but not taken */ + span.branchNoCov + { + background-color: #FF6230; + } + + /* Source code view: format for branches which were not executed */ + span.branchNoExec + { + background-color: #FF6230; + } + + /* Source code view: format for the source code heading line */ + pre.sourceHeading + { + white-space: pre; + font-family: monospace; + font-weight: bold; + margin: 0px; + } + + /* All views: header legend value for low rate */ + td.headerValueLegL + { + font-family: sans-serif; + text-align: center; + white-space: nowrap; + padding-left: 4px; + padding-right: 2px; + background-color: #FF0000; + font-size: 80%; + } + + /* All views: header legend value for med rate */ + td.headerValueLegM + { + font-family: sans-serif; + text-align: center; + white-space: nowrap; + padding-left: 2px; + padding-right: 2px; + background-color: #FFEA20; + font-size: 80%; + } + + /* All views: header legend value for hi rate */ + td.headerValueLegH + { + font-family: sans-serif; + text-align: center; + white-space: nowrap; + padding-left: 2px; + padding-right: 4px; + background-color: #A7FC9D; + font-size: 80%; + } + + /* All views except source code view: legend format for low coverage */ + span.coverLegendCovLo + { + padding-left: 10px; + padding-right: 10px; + padding-top: 2px; + background-color: #FF0000; + } + + /* All views except source code view: legend format for med coverage */ + span.coverLegendCovMed + { + padding-left: 10px; + padding-right: 10px; + padding-top: 2px; + background-color: #FFEA20; + } + + /* All views except source code view: legend format for hi coverage */ + span.coverLegendCovHi + { + padding-left: 10px; + padding-right: 10px; + padding-top: 2px; + background-color: #A7FC9D; + } +END_OF_CSS + ; + + # ************************************************************* + + + # Remove leading tab from all lines + $css_data =~ s/^\t//gm; + + print(CSS_HANDLE $css_data); + + close(CSS_HANDLE); +} + + +# +# get_bar_graph_code(base_dir, cover_found, cover_hit) +# +# Return a string containing HTML code which implements a bar graph display +# for a coverage rate of cover_hit * 100 / cover_found. +# + +sub get_bar_graph_code($$$) +{ + my ($base_dir, $found, $hit) = @_; + my $rate; + my $alt; + my $width; + my $remainder; + my $png_name; + my $graph_code; + + # Check number of instrumented lines + if ($_[1] == 0) { return ""; } + + $alt = rate($hit, $found, "%"); + $width = rate($hit, $found, undef, 0); + $remainder = 100 - $width; + + # Decide which .png file to use + $png_name = $rate_png[classify_rate($found, $hit, $med_limit, + $hi_limit)]; + + if ($width == 0) + { + # Zero coverage + $graph_code = (<$alt +END_OF_HTML + ; + } + elsif ($width == 100) + { + # Full coverage + $graph_code = (<$alt +END_OF_HTML + ; + } + else + { + # Positive coverage + $graph_code = (<$alt$alt +END_OF_HTML + ; + } + + # Remove leading tabs from all lines + $graph_code =~ s/^\t+//gm; + chomp($graph_code); + + return($graph_code); +} + +# +# sub classify_rate(found, hit, med_limit, high_limit) +# +# Return 0 for low rate, 1 for medium rate and 2 for hi rate. +# + +sub classify_rate($$$$) +{ + my ($found, $hit, $med, $hi) = @_; + my $rate; + + if ($found == 0) { + return 2; + } + $rate = rate($hit, $found); + if ($rate < $med) { + return 0; + } elsif ($rate < $hi) { + return 1; + } + return 2; +} + + +# +# write_html(filehandle, html_code) +# +# Write out HTML_CODE to FILEHANDLE while removing a leading tabulator mark +# in each line of HTML_CODE. +# + +sub write_html(*$) +{ + local *HTML_HANDLE = $_[0]; + my $html_code = $_[1]; + + # Remove leading tab from all lines + $html_code =~ s/^\t//gm; + + print(HTML_HANDLE $html_code) + or die("ERROR: cannot write HTML data ($!)\n"); +} + + +# +# write_html_prolog(filehandle, base_dir, pagetitle) +# +# Write an HTML prolog common to all HTML files to FILEHANDLE. PAGETITLE will +# be used as HTML page title. BASE_DIR contains a relative path which points +# to the base directory. +# + +sub write_html_prolog(*$$) +{ + my $basedir = $_[1]; + my $pagetitle = $_[2]; + my $prolog; + + $prolog = $html_prolog; + $prolog =~ s/\@pagetitle\@/$pagetitle/g; + $prolog =~ s/\@basedir\@/$basedir/g; + + write_html($_[0], $prolog); +} + + +# +# write_header_prolog(filehandle, base_dir) +# +# Write beginning of page header HTML code. +# + +sub write_header_prolog(*$) +{ + # ************************************************************* + + write_html($_[0], < + $title + + + + + +END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_header_line(handle, content) +# +# Write a header line with the specified table contents. +# + +sub write_header_line(*@) +{ + my ($handle, @content) = @_; + my $entry; + + write_html($handle, " \n"); + foreach $entry (@content) { + my ($width, $class, $text, $colspan) = @{$entry}; + + if (defined($width)) { + $width = " width=\"$width\""; + } else { + $width = ""; + } + if (defined($class)) { + $class = " class=\"$class\""; + } else { + $class = ""; + } + if (defined($colspan)) { + $colspan = " colspan=\"$colspan\""; + } else { + $colspan = ""; + } + $text = "" if (!defined($text)); + write_html($handle, + " $text\n"); + } + write_html($handle, " \n"); +} + + +# +# write_header_epilog(filehandle, base_dir) +# +# Write end of page header HTML code. +# + +sub write_header_epilog(*$) +{ + # ************************************************************* + + write_html($_[0], < +
+ + + + + + +END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_file_table_prolog(handle, file_heading, ([heading, num_cols], ...)) +# +# Write heading for file table. +# + +sub write_file_table_prolog(*$@) +{ + my ($handle, $file_heading, @columns) = @_; + my $num_columns = 0; + my $file_width; + my $col; + my $width; + + $width = 20 if (scalar(@columns) == 1); + $width = 10 if (scalar(@columns) == 2); + $width = 8 if (scalar(@columns) > 2); + + foreach $col (@columns) { + my ($heading, $cols) = @{$col}; + + $num_columns += $cols; + } + $file_width = 100 - $num_columns * $width; + + # Table definition + write_html($handle, < + + + + +END_OF_HTML + # Empty first row + foreach $col (@columns) { + my ($heading, $cols) = @{$col}; + + while ($cols-- > 0) { + write_html($handle, < +END_OF_HTML + } + } + # Next row + write_html($handle, < + + + +END_OF_HTML + # Heading row + foreach $col (@columns) { + my ($heading, $cols) = @{$col}; + my $colspan = ""; + + $colspan = " colspan=$cols" if ($cols > 1); + write_html($handle, <$heading +END_OF_HTML + } + write_html($handle, < +END_OF_HTML +} + + +# write_file_table_entry(handle, base_dir, filename, page_link, +# ([ found, hit, med_limit, hi_limit, graph ], ..) +# +# Write an entry of the file table. +# + +sub write_file_table_entry(*$$$@) +{ + my ($handle, $base_dir, $filename, $page_link, @entries) = @_; + my $file_code; + my $entry; + my $esc_filename = escape_html($filename); + + # Add link to source if provided + if (defined($page_link) && $page_link ne "") { + $file_code = "$esc_filename"; + } else { + $file_code = $esc_filename; + } + + # First column: filename + write_html($handle, < + +END_OF_HTML + # Columns as defined + foreach $entry (@entries) { + my ($found, $hit, $med, $hi, $graph) = @{$entry}; + my $bar_graph; + my $class; + my $rate; + + # Generate bar graph if requested + if ($graph) { + $bar_graph = get_bar_graph_code($base_dir, $found, + $hit); + write_html($handle, < + $bar_graph + +END_OF_HTML + } + # Get rate color and text + if ($found == 0) { + $rate = "-"; + $class = "Hi"; + } else { + $rate = rate($hit, $found, " %"); + $class = $rate_name[classify_rate($found, $hit, + $med, $hi)]; + } + if ($opt_missed) { + # Show negative number of items without coverage + $hit = -($found - $hit); + } + write_html($handle, <$rate + +END_OF_HTML + } + # End of row + write_html($handle, < +END_OF_HTML +} + + +# +# write_file_table_detail_entry(filehandle, test_name, ([found, hit], ...)) +# +# Write entry for detail section in file table. +# + +sub write_file_table_detail_entry(*$@) +{ + my ($handle, $test, @entries) = @_; + my $entry; + + if ($test eq "") { + $test = "<unnamed>"; + } elsif ($test =~ /^(.*),diff$/) { + $test = $1." (converted)"; + } + # Testname + write_html($handle, < + +END_OF_HTML + # Test data + foreach $entry (@entries) { + my ($found, $hit) = @{$entry}; + my $rate = rate($hit, $found, " %"); + + write_html($handle, <$rate + +END_OF_HTML + } + + write_html($handle, < + +END_OF_HTML + + # ************************************************************* +} + + +# +# write_file_table_epilog(filehandle) +# +# Write end of file table HTML code. +# + +sub write_file_table_epilog(*) +{ + # ************************************************************* + + write_html($_[0], < + +
+ +END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_test_table_prolog(filehandle, table_heading) +# +# Write heading for test case description table. +# + +sub write_test_table_prolog(*$) +{ + # ************************************************************* + + write_html($_[0], < +

$file_heading$file_code$hit / $found$test$hit / $found
+ + + + + + + + + + + + +

$_[1]
+
+END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_test_table_entry(filehandle, test_name, test_description) +# +# Write entry for the test table. +# + +sub write_test_table_entry(*$$) +{ + # ************************************************************* + + write_html($_[0], <$_[1]  +
$_[2]

+END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_test_table_epilog(filehandle) +# +# Write end of test description table HTML code. +# + +sub write_test_table_epilog(*) +{ + # ************************************************************* + + write_html($_[0], < +
+ +
+ +END_OF_HTML + ; + + # ************************************************************* +} + + +sub fmt_centered($$) +{ + my ($width, $text) = @_; + my $w0 = length($text); + my $w1 = $width > $w0 ? int(($width - $w0) / 2) : 0; + my $w2 = $width > $w0 ? $width - $w0 - $w1 : 0; + + return (" "x$w1).$text.(" "x$w2); +} + + +# +# write_source_prolog(filehandle) +# +# Write start of source code table. +# + +sub write_source_prolog(*) +{ + my $lineno_heading = " "; + my $branch_heading = ""; + my $line_heading = fmt_centered($line_field_width, "Line data"); + my $source_heading = " Source code"; + + if ($br_coverage) { + $branch_heading = fmt_centered($br_field_width, "Branch data"). + " "; + } + # ************************************************************* + + write_html($_[0], < + +
+ + + +
${lineno_heading}${branch_heading}${line_heading} ${source_heading}
+
+END_OF_HTML
+	;
+
+	# *************************************************************
+}
+
+sub cmp_blocks($$)
+{
+	my ($a, $b) = @_;
+	my ($fa, $fb) = ($a->[0], $b->[0]);
+
+	return $fa->[0] <=> $fb->[0] if ($fa->[0] != $fb->[0]);
+	return $fa->[1] <=> $fb->[1];
+}
+
+#
+# get_branch_blocks(brdata)
+#
+# Group branches that belong to the same basic block.
+#
+# Returns: [block1, block2, ...]
+# block:   [branch1, branch2, ...]
+# branch:  [block_num, branch_num, taken_count, text_length, open, close]
+#
+
+sub get_branch_blocks($)
+{
+	my ($brdata) = @_;
+	my $last_block_num;
+	my $block = [];
+	my @blocks;
+
+	return () if (!defined($brdata));
+
+	# Group branches
+	foreach my $entry (split(/:/, $brdata)) {
+		my ($block_num, $branch, $taken) = split(/,/, $entry);
+		my $br;
+
+		if (defined($last_block_num) && $block_num != $last_block_num) {
+			push(@blocks, $block);
+			$block = [];
+		}
+		$br = [$block_num, $branch, $taken, 3, 0, 0];
+		push(@{$block}, $br);
+		$last_block_num = $block_num;
+	}
+	push(@blocks, $block) if (scalar(@{$block}) > 0);
+
+	# Add braces to first and last branch in group
+	foreach $block (@blocks) {
+		$block->[0]->[$BR_OPEN] = 1;
+		$block->[0]->[$BR_LEN]++;
+		$block->[scalar(@{$block}) - 1]->[$BR_CLOSE] = 1;
+		$block->[scalar(@{$block}) - 1]->[$BR_LEN]++;
+	}
+
+	return sort(cmp_blocks @blocks);
+}
+
+#
+# get_block_len(block)
+#
+# Calculate total text length of all branches in a block of branches.
+#
+
+sub get_block_len($)
+{
+	my ($block) = @_;
+	my $len = 0;
+	my $branch;
+
+	foreach $branch (@{$block}) {
+		$len += $branch->[$BR_LEN];
+	}
+
+	return $len;
+}
+
+
+#
+# get_branch_html(brdata)
+#
+# Return a list of HTML lines which represent the specified branch coverage
+# data in source code view.
+#
+
+sub get_branch_html($)
+{
+	my ($brdata) = @_;
+	my @blocks = get_branch_blocks($brdata);
+	my $block;
+	my $branch;
+	my $line_len = 0;
+	my $line = [];	# [branch2|" ", branch|" ", ...]
+	my @lines;	# [line1, line2, ...]
+	my @result;
+
+	# Distribute blocks to lines
+	foreach $block (@blocks) {
+		my $block_len = get_block_len($block);
+
+		# Does this block fit into the current line?
+		if ($line_len + $block_len <= $br_field_width) {
+			# Add it
+			$line_len += $block_len;
+			push(@{$line}, @{$block});
+			next;
+		} elsif ($block_len <= $br_field_width) {
+			# It would fit if the line was empty - add it to new
+			# line
+			push(@lines, $line);
+			$line_len = $block_len;
+			$line = [ @{$block} ];
+			next;
+		}
+		# Split the block into several lines
+		foreach $branch (@{$block}) {
+			if ($line_len + $branch->[$BR_LEN] >= $br_field_width) {
+				# Start a new line
+				if (($line_len + 1 <= $br_field_width) &&
+				    scalar(@{$line}) > 0 &&
+				    !$line->[scalar(@$line) - 1]->[$BR_CLOSE]) {
+					# Try to align branch symbols to be in
+					# one # row
+					push(@{$line}, " ");
+				}
+				push(@lines, $line);
+				$line_len = 0;
+				$line = [];
+			}
+			push(@{$line}, $branch);
+			$line_len += $branch->[$BR_LEN];
+		}
+	}
+	push(@lines, $line);
+
+	# Convert to HTML
+	foreach $line (@lines) {
+		my $current = "";
+		my $current_len = 0;
+
+		foreach $branch (@$line) {
+			# Skip alignment space
+			if ($branch eq " ") {
+				$current .= " ";
+				$current_len++;
+				next;
+			}
+
+			my ($block_num, $br_num, $taken, $len, $open, $close) =
+			   @{$branch};
+			my $class;
+			my $title;
+			my $text;
+
+			if ($taken eq '-') {
+				$class	= "branchNoExec";
+				$text	= " # ";
+				$title	= "Branch $br_num was not executed";
+			} elsif ($taken == 0) {
+				$class	= "branchNoCov";
+				$text	= " - ";
+				$title	= "Branch $br_num was not taken";
+			} else {
+				$class	= "branchCov";
+				$text	= " + ";
+				$title	= "Branch $br_num was taken $taken ".
+					  "time";
+				$title .= "s" if ($taken > 1);
+			}
+			$current .= "[" if ($open);
+			$current .= "";
+			$current .= $text."";
+			$current .= "]" if ($close);
+			$current_len += $len;
+		}
+
+		# Right-align result text
+		if ($current_len < $br_field_width) {
+			$current = (" "x($br_field_width - $current_len)).
+				   $current;
+		}
+		push(@result, $current);
+	}
+
+	return @result;
+}
+
+
+#
+# format_count(count, width)
+#
+# Return a right-aligned representation of count that fits in width characters.
+#
+
+sub format_count($$)
+{
+	my ($count, $width) = @_;
+	my $result;
+	my $exp;
+
+	$result = sprintf("%*.0f", $width, $count);
+	while (length($result) > $width) {
+		last if ($count < 10);
+		$exp++;
+		$count = int($count/10);
+		$result = sprintf("%*s", $width, ">$count*10^$exp");
+	}
+	return $result;
+}
+
+#
+# write_source_line(filehandle, line_num, source, hit_count, converted,
+#                   brdata)
+#
+# Write formatted source code line. Return a line in a format as needed
+# by gen_png()
+#
+
+sub write_source_line(*$$$$$)
+{
+	my ($handle, $line, $source, $count, $converted, $brdata) = @_;
+	my $source_format;
+	my $count_format;
+	my $result;
+	my $anchor_start = "";
+	my $anchor_end = "";
+	my $count_field_width = $line_field_width - 1;
+	my @br_html;
+	my $html;
+
+	# Get branch HTML data for this line
+	@br_html = get_branch_html($brdata) if ($br_coverage);
+
+	if (!defined($count)) {
+		$result		= "";
+		$source_format	= "";
+		$count_format	= " "x$count_field_width;
+	}
+	elsif ($count == 0) {
+		$result		= $count;
+		$source_format	= '';
+		$count_format	= format_count($count, $count_field_width);
+	}
+	elsif ($converted && defined($highlight)) {
+		$result		= "*".$count;
+		$source_format	= '';
+		$count_format	= format_count($count, $count_field_width);
+	}
+	else {
+		$result		= $count;
+		$source_format	= '';
+		$count_format	= format_count($count, $count_field_width);
+	}
+	$result .= ":".$source;
+
+	# Write out a line number navigation anchor every $nav_resolution
+	# lines if necessary
+	$anchor_start	= "";
+	$anchor_end	= "";
+
+
+	# *************************************************************
+
+	$html = $anchor_start;
+	$html .= "".sprintf("%8d", $line)." ";
+	$html .= shift(@br_html).":" if ($br_coverage);
+	$html .= "$source_format$count_format : ";
+	$html .= escape_html($source);
+	$html .= "" if ($source_format);
+	$html .= $anchor_end."\n";
+
+	write_html($handle, $html);
+
+	if ($br_coverage) {
+		# Add lines for overlong branch information
+		foreach (@br_html) {
+			write_html($handle, "".
+				   "         $_\n");
+		}
+	}
+	# *************************************************************
+
+	return($result);
+}
+
+
+#
+# write_source_epilog(filehandle)
+#
+# Write end of source code table.
+#
+
+sub write_source_epilog(*)
+{
+	# *************************************************************
+
+	write_html($_[0], <
+	      
+	    
+	  
+	  
+ +END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_html_epilog(filehandle, base_dir[, break_frames]) +# +# Write HTML page footer to FILEHANDLE. BREAK_FRAMES should be set when +# this page is embedded in a frameset, clicking the URL link will then +# break this frameset. +# + +sub write_html_epilog(*$;$) +{ + my $basedir = $_[1]; + my $break_code = ""; + my $epilog; + + if (defined($_[2])) + { + $break_code = " target=\"_parent\""; + } + + # ************************************************************* + + write_html($_[0], < + + Generated by: $lcov_version + +
+END_OF_HTML + ; + + $epilog = $html_epilog; + $epilog =~ s/\@basedir\@/$basedir/g; + + write_html($_[0], $epilog); +} + + +# +# write_frameset(filehandle, basedir, basename, pagetitle) +# +# + +sub write_frameset(*$$$) +{ + my $frame_width = $overview_width + 40; + + # ************************************************************* + + write_html($_[0], < + + + + + + $_[3] + + + + + + + + <center>Frames not supported by your browser!<br></center> + + + + +END_OF_HTML + ; + + # ************************************************************* +} + + +# +# sub write_overview_line(filehandle, basename, line, link) +# +# + +sub write_overview_line(*$$$) +{ + my $y1 = $_[2] - 1; + my $y2 = $y1 + $nav_resolution - 1; + my $x2 = $overview_width - 1; + + # ************************************************************* + + write_html($_[0], < +END_OF_HTML + ; + + # ************************************************************* +} + + +# +# write_overview(filehandle, basedir, basename, pagetitle, lines) +# +# + +sub write_overview(*$$$$) +{ + my $index; + my $max_line = $_[4] - 1; + my $offset; + + # ************************************************************* + + write_html($_[0], < + + + + + $_[3] + + + + + + +END_OF_HTML + ; + + # ************************************************************* + + # Make $offset the next higher multiple of $nav_resolution + $offset = ($nav_offset + $nav_resolution - 1) / $nav_resolution; + $offset = sprintf("%d", $offset ) * $nav_resolution; + + # Create image map for overview image + for ($index = 1; $index <= $_[4]; $index += $nav_resolution) + { + # Enforce nav_offset + if ($index < $offset + 1) + { + write_overview_line($_[0], $_[2], $index, 1); + } + else + { + write_overview_line($_[0], $_[2], $index, $index - $offset); + } + } + + # ************************************************************* + + write_html($_[0], < + +
+ Top

+ Overview +
+ + +END_OF_HTML + ; + + # ************************************************************* +} + + +sub max($$) +{ + my ($a, $b) = @_; + + return $a if ($a > $b); + return $b; +} + + +# +# write_header(filehandle, type, trunc_file_name, rel_file_name, lines_found, +# lines_hit, funcs_found, funcs_hit, sort_type) +# +# Write a complete standard page header. TYPE may be (0, 1, 2, 3, 4) +# corresponding to (directory view header, file view header, source view +# header, test case description header, function view header) +# + +sub write_header(*$$$$$$$$$$) +{ + local *HTML_HANDLE = $_[0]; + my $type = $_[1]; + my $trunc_name = $_[2]; + my $rel_filename = $_[3]; + my $lines_found = $_[4]; + my $lines_hit = $_[5]; + my $fn_found = $_[6]; + my $fn_hit = $_[7]; + my $br_found = $_[8]; + my $br_hit = $_[9]; + my $sort_type = $_[10]; + my $base_dir; + my $view; + my $test; + my $base_name; + my $style; + my $rate; + my @row_left; + my @row_right; + my $num_rows; + my $i; + my $esc_trunc_name = escape_html($trunc_name); + + $base_name = basename($rel_filename); + + # Prepare text for "current view" field + if ($type == $HDR_DIR) + { + # Main overview + $base_dir = ""; + $view = $overview_title; + } + elsif ($type == $HDR_FILE) + { + # Directory overview + $base_dir = get_relative_base_path($rel_filename); + $view = "". + "$overview_title - $esc_trunc_name"; + } + elsif ($type == $HDR_SOURCE || $type == $HDR_FUNC) + { + # File view + my $dir_name = dirname($rel_filename); + my $esc_base_name = escape_html($base_name); + my $esc_dir_name = escape_html($dir_name); + + $base_dir = get_relative_base_path($dir_name); + if ($frames) + { + # Need to break frameset when clicking any of these + # links + $view = "$overview_title - ". + "". + "$esc_trunc_name - $esc_base_name"; + } + else + { + $view = "". + "$overview_title - ". + "". + "$esc_trunc_name - $esc_base_name"; + } + + # Add function suffix + if ($func_coverage) { + $view .= ""; + if ($type == $HDR_SOURCE) { + if ($sort) { + $view .= " (source / functions)"; + } else { + $view .= " (source / functions)"; + } + } elsif ($type == $HDR_FUNC) { + $view .= " (source / functions)"; + } + $view .= ""; + } + } + elsif ($type == $HDR_TESTDESC) + { + # Test description header + $base_dir = ""; + $view = "". + "$overview_title - test case descriptions"; + } + + # Prepare text for "test" field + $test = escape_html($test_title); + + # Append link to test description page if available + if (%test_description && ($type != $HDR_TESTDESC)) + { + if ($frames && ($type == $HDR_SOURCE || $type == $HDR_FUNC)) + { + # Need to break frameset when clicking this link + $test .= " ( ". + "". + "view descriptions )"; + } + else + { + $test .= " ( ". + "". + "view descriptions )"; + } + } + + # Write header + write_header_prolog(*HTML_HANDLE, $base_dir); + + # Left row + push(@row_left, [[ "10%", "headerItem", "Current view:" ], + [ "35%", "headerValue", $view ]]); + push(@row_left, [[undef, "headerItem", "Test:"], + [undef, "headerValue", $test]]); + push(@row_left, [[undef, "headerItem", "Date:"], + [undef, "headerValue", $date]]); + + # Right row + if ($legend && ($type == $HDR_SOURCE || $type == $HDR_FUNC)) { + my $text = <hit
+ not hit +END_OF_HTML + if ($br_coverage) { + $text .= <+
taken + - not taken + # not executed +END_OF_HTML + } + push(@row_left, [[undef, "headerItem", "Legend:"], + [undef, "headerValueLeg", $text]]); + } elsif ($legend && ($type != $HDR_TESTDESC)) { + my $text = <low: < $med_limit % + medium: >= $med_limit % + high: >= $hi_limit % +END_OF_HTML + push(@row_left, [[undef, "headerItem", "Legend:"], + [undef, "headerValueLeg", $text]]); + } + if ($type == $HDR_TESTDESC) { + push(@row_right, [[ "55%" ]]); + } else { + push(@row_right, [["15%", undef, undef ], + ["10%", "headerCovTableHead", "Hit" ], + ["10%", "headerCovTableHead", "Total" ], + ["15%", "headerCovTableHead", "Coverage"]]); + } + # Line coverage + $style = $rate_name[classify_rate($lines_found, $lines_hit, + $med_limit, $hi_limit)]; + $rate = rate($lines_hit, $lines_found, " %"); + push(@row_right, [[undef, "headerItem", "Lines:"], + [undef, "headerCovTableEntry", $lines_hit], + [undef, "headerCovTableEntry", $lines_found], + [undef, "headerCovTableEntry$style", $rate]]) + if ($type != $HDR_TESTDESC); + # Function coverage + if ($func_coverage) { + $style = $rate_name[classify_rate($fn_found, $fn_hit, + $fn_med_limit, $fn_hi_limit)]; + $rate = rate($fn_hit, $fn_found, " %"); + push(@row_right, [[undef, "headerItem", "Functions:"], + [undef, "headerCovTableEntry", $fn_hit], + [undef, "headerCovTableEntry", $fn_found], + [undef, "headerCovTableEntry$style", $rate]]) + if ($type != $HDR_TESTDESC); + } + # Branch coverage + if ($br_coverage) { + $style = $rate_name[classify_rate($br_found, $br_hit, + $br_med_limit, $br_hi_limit)]; + $rate = rate($br_hit, $br_found, " %"); + push(@row_right, [[undef, "headerItem", "Branches:"], + [undef, "headerCovTableEntry", $br_hit], + [undef, "headerCovTableEntry", $br_found], + [undef, "headerCovTableEntry$style", $rate]]) + if ($type != $HDR_TESTDESC); + } + + # Print rows + $num_rows = max(scalar(@row_left), scalar(@row_right)); + for ($i = 0; $i < $num_rows; $i++) { + my $left = $row_left[$i]; + my $right = $row_right[$i]; + + if (!defined($left)) { + $left = [[undef, undef, undef], [undef, undef, undef]]; + } + if (!defined($right)) { + $right = []; + } + write_header_line(*HTML_HANDLE, @{$left}, + [ $i == 0 ? "5%" : undef, undef, undef], + @{$right}); + } + + # Fourth line + write_header_epilog(*HTML_HANDLE, $base_dir); +} + +sub get_sorted_by_rate($$) +{ + my ($hash, $type) = @_; + + if ($type == $SORT_LINE) { + # Sort by line coverage + return sort({$hash->{$a}[7] <=> $hash->{$b}[7]} keys(%{$hash})); + } elsif ($type == $SORT_FUNC) { + # Sort by function coverage; + return sort({$hash->{$a}[8] <=> $hash->{$b}[8]} keys(%{$hash})); + } elsif ($type == $SORT_BRANCH) { + # Sort by br coverage; + return sort({$hash->{$a}[9] <=> $hash->{$b}[9]} keys(%{$hash})); + } +} + +sub get_sorted_by_missed($$) +{ + my ($hash, $type) = @_; + + if ($type == $SORT_LINE) { + # Sort by number of instrumented lines without coverage + return sort( + { + ($hash->{$b}[0] - $hash->{$b}[1]) <=> + ($hash->{$a}[0] - $hash->{$a}[1]) + } keys(%{$hash})); + } elsif ($type == $SORT_FUNC) { + # Sort by number of instrumented functions without coverage + return sort( + { + ($hash->{$b}[2] - $hash->{$b}[3]) <=> + ($hash->{$a}[2] - $hash->{$a}[3]) + } keys(%{$hash})); + } elsif ($type == $SORT_BRANCH) { + # Sort by number of instrumented branches without coverage + return sort( + { + ($hash->{$b}[4] - $hash->{$b}[5]) <=> + ($hash->{$a}[4] - $hash->{$a}[5]) + } keys(%{$hash})); + } +} + +# +# get_sorted_keys(hash_ref, sort_type) +# +# hash_ref: filename -> stats +# stats: [ lines_found, lines_hit, fn_found, fn_hit, br_found, br_hit, +# link_name, line_rate, fn_rate, br_rate ] +# + +sub get_sorted_keys($$) +{ + my ($hash, $type) = @_; + + if ($type == $SORT_FILE) { + # Sort by name + return sort(keys(%{$hash})); + } elsif ($opt_missed) { + return get_sorted_by_missed($hash, $type); + } else { + return get_sorted_by_rate($hash, $type); + } +} + +sub get_sort_code($$$) +{ + my ($link, $alt, $base) = @_; + my $png; + my $link_start; + my $link_end; + + if (!defined($link)) { + $png = "glass.png"; + $link_start = ""; + $link_end = ""; + } else { + $png = "updown.png"; + $link_start = ''; + $link_end = ""; + } + + return ' '.$link_start. + ''.$link_end.''; +} + +sub get_file_code($$$$) +{ + my ($type, $text, $sort_button, $base) = @_; + my $result = $text; + my $link; + + if ($sort_button) { + if ($type == $HEAD_NO_DETAIL) { + $link = "index.$html_ext"; + } else { + $link = "index-detail.$html_ext"; + } + } + $result .= get_sort_code($link, "Sort by name", $base); + + return $result; +} + +sub get_line_code($$$$$) +{ + my ($type, $sort_type, $text, $sort_button, $base) = @_; + my $result = $text; + my $sort_link; + + if ($type == $HEAD_NO_DETAIL) { + # Just text + if ($sort_button) { + $sort_link = "index-sort-l.$html_ext"; + } + } elsif ($type == $HEAD_DETAIL_HIDDEN) { + # Text + link to detail view + $result .= ' ( show details )'; + if ($sort_button) { + $sort_link = "index-sort-l.$html_ext"; + } + } else { + # Text + link to standard view + $result .= ' ( hide details )'; + if ($sort_button) { + $sort_link = "index-detail-sort-l.$html_ext"; + } + } + # Add sort button + $result .= get_sort_code($sort_link, "Sort by line coverage", $base); + + return $result; +} + +sub get_func_code($$$$) +{ + my ($type, $text, $sort_button, $base) = @_; + my $result = $text; + my $link; + + if ($sort_button) { + if ($type == $HEAD_NO_DETAIL) { + $link = "index-sort-f.$html_ext"; + } else { + $link = "index-detail-sort-f.$html_ext"; + } + } + $result .= get_sort_code($link, "Sort by function coverage", $base); + return $result; +} + +sub get_br_code($$$$) +{ + my ($type, $text, $sort_button, $base) = @_; + my $result = $text; + my $link; + + if ($sort_button) { + if ($type == $HEAD_NO_DETAIL) { + $link = "index-sort-b.$html_ext"; + } else { + $link = "index-detail-sort-b.$html_ext"; + } + } + $result .= get_sort_code($link, "Sort by branch coverage", $base); + return $result; +} + +# +# write_file_table(filehandle, base_dir, overview, testhash, testfnchash, +# testbrhash, fileview, sort_type) +# +# Write a complete file table. OVERVIEW is a reference to a hash containing +# the following mapping: +# +# filename -> "lines_found,lines_hit,funcs_found,funcs_hit,page_link, +# func_link" +# +# TESTHASH is a reference to the following hash: +# +# filename -> \%testdata +# %testdata: name of test affecting this file -> \%testcount +# %testcount: line number -> execution count for a single test +# +# Heading of first column is "Filename" if FILEVIEW is true, "Directory name" +# otherwise. +# + +sub write_file_table(*$$$$$$$) +{ + local *HTML_HANDLE = $_[0]; + my $base_dir = $_[1]; + my $overview = $_[2]; + my $testhash = $_[3]; + my $testfnchash = $_[4]; + my $testbrhash = $_[5]; + my $fileview = $_[6]; + my $sort_type = $_[7]; + my $filename; + my $bar_graph; + my $hit; + my $found; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + my $page_link; + my $testname; + my $testdata; + my $testfncdata; + my $testbrdata; + my %affecting_tests; + my $line_code = ""; + my $func_code; + my $br_code; + my $file_code; + my @head_columns; + + # Determine HTML code for column headings + if (($base_dir ne "") && $show_details) + { + my $detailed = keys(%{$testhash}); + + $file_code = get_file_code($detailed ? $HEAD_DETAIL_HIDDEN : + $HEAD_NO_DETAIL, + $fileview ? "Filename" : "Directory", + $sort && $sort_type != $SORT_FILE, + $base_dir); + $line_code = get_line_code($detailed ? $HEAD_DETAIL_SHOWN : + $HEAD_DETAIL_HIDDEN, + $sort_type, + "Line Coverage", + $sort && $sort_type != $SORT_LINE, + $base_dir); + $func_code = get_func_code($detailed ? $HEAD_DETAIL_HIDDEN : + $HEAD_NO_DETAIL, + "Functions", + $sort && $sort_type != $SORT_FUNC, + $base_dir); + $br_code = get_br_code($detailed ? $HEAD_DETAIL_HIDDEN : + $HEAD_NO_DETAIL, + "Branches", + $sort && $sort_type != $SORT_BRANCH, + $base_dir); + } else { + $file_code = get_file_code($HEAD_NO_DETAIL, + $fileview ? "Filename" : "Directory", + $sort && $sort_type != $SORT_FILE, + $base_dir); + $line_code = get_line_code($HEAD_NO_DETAIL, $sort_type, "Line Coverage", + $sort && $sort_type != $SORT_LINE, + $base_dir); + $func_code = get_func_code($HEAD_NO_DETAIL, "Functions", + $sort && $sort_type != $SORT_FUNC, + $base_dir); + $br_code = get_br_code($HEAD_NO_DETAIL, "Branches", + $sort && $sort_type != $SORT_BRANCH, + $base_dir); + } + push(@head_columns, [ $line_code, 3 ]); + push(@head_columns, [ $func_code, 2]) if ($func_coverage); + push(@head_columns, [ $br_code, 2]) if ($br_coverage); + + write_file_table_prolog(*HTML_HANDLE, $file_code, @head_columns); + + foreach $filename (get_sorted_keys($overview, $sort_type)) + { + my @columns; + ($found, $hit, $fn_found, $fn_hit, $br_found, $br_hit, + $page_link) = @{$overview->{$filename}}; + + # Line coverage + push(@columns, [$found, $hit, $med_limit, $hi_limit, 1]); + # Function coverage + if ($func_coverage) { + push(@columns, [$fn_found, $fn_hit, $fn_med_limit, + $fn_hi_limit, 0]); + } + # Branch coverage + if ($br_coverage) { + push(@columns, [$br_found, $br_hit, $br_med_limit, + $br_hi_limit, 0]); + } + write_file_table_entry(*HTML_HANDLE, $base_dir, $filename, + $page_link, @columns); + + $testdata = $testhash->{$filename}; + $testfncdata = $testfnchash->{$filename}; + $testbrdata = $testbrhash->{$filename}; + + # Check whether we should write test specific coverage + # as well + if (!($show_details && $testdata)) { next; } + + # Filter out those tests that actually affect this file + %affecting_tests = %{ get_affecting_tests($testdata, + $testfncdata, $testbrdata) }; + + # Does any of the tests affect this file at all? + if (!%affecting_tests) { next; } + + foreach $testname (keys(%affecting_tests)) + { + my @results; + ($found, $hit, $fn_found, $fn_hit, $br_found, $br_hit) = + split(",", $affecting_tests{$testname}); + + # Insert link to description of available + if ($test_description{$testname}) + { + $testname = "". + "$testname"; + } + + push(@results, [$found, $hit]); + push(@results, [$fn_found, $fn_hit]) if ($func_coverage); + push(@results, [$br_found, $br_hit]) if ($br_coverage); + write_file_table_detail_entry(*HTML_HANDLE, $testname, + @results); + } + } + + write_file_table_epilog(*HTML_HANDLE); +} + + +# +# get_found_and_hit(hash) +# +# Return the count for entries (found) and entries with an execution count +# greater than zero (hit) in a hash (linenumber -> execution count) as +# a list (found, hit) +# + +sub get_found_and_hit($) +{ + my %hash = %{$_[0]}; + my $found = 0; + my $hit = 0; + + # Calculate sum + $found = 0; + $hit = 0; + + foreach (keys(%hash)) + { + $found++; + if ($hash{$_}>0) { $hit++; } + } + + return ($found, $hit); +} + + +# +# get_func_found_and_hit(sumfnccount) +# +# Return (f_found, f_hit) for sumfnccount +# + +sub get_func_found_and_hit($) +{ + my ($sumfnccount) = @_; + my $function; + my $fn_found; + my $fn_hit; + + $fn_found = scalar(keys(%{$sumfnccount})); + $fn_hit = 0; + foreach $function (keys(%{$sumfnccount})) { + if ($sumfnccount->{$function} > 0) { + $fn_hit++; + } + } + return ($fn_found, $fn_hit); +} + + +sub get_br_found_and_hit($) +{ + my ($brcount) = @_; + my $db; + + $db = brcount_to_db($brcount); + + return brcount_db_get_found_and_hit($db); +} + + +# +# get_affecting_tests(testdata, testfncdata, testbrdata) +# +# HASHREF contains a mapping filename -> (linenumber -> exec count). Return +# a hash containing mapping filename -> "lines found, lines hit" for each +# filename which has a nonzero hit count. +# + +sub get_affecting_tests($$$) +{ + my ($testdata, $testfncdata, $testbrdata) = @_; + my $testname; + my $testcount; + my $testfnccount; + my $testbrcount; + my %result; + my $found; + my $hit; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + + foreach $testname (keys(%{$testdata})) + { + # Get (line number -> count) hash for this test case + $testcount = $testdata->{$testname}; + $testfnccount = $testfncdata->{$testname}; + $testbrcount = $testbrdata->{$testname}; + + # Calculate sum + ($found, $hit) = get_found_and_hit($testcount); + ($fn_found, $fn_hit) = get_func_found_and_hit($testfnccount); + ($br_found, $br_hit) = get_br_found_and_hit($testbrcount); + + if ($hit>0) + { + $result{$testname} = "$found,$hit,$fn_found,$fn_hit,". + "$br_found,$br_hit"; + } + } + + return(\%result); +} + + +sub get_hash_reverse($) +{ + my ($hash) = @_; + my %result; + + foreach (keys(%{$hash})) { + $result{$hash->{$_}} = $_; + } + + return \%result; +} + +# +# write_source(filehandle, source_filename, count_data, checksum_data, +# converted_data, func_data, sumbrcount) +# +# Write an HTML view of a source code file. Returns a list containing +# data as needed by gen_png(). +# +# Die on error. +# + +sub write_source($$$$$$$) +{ + local *HTML_HANDLE = $_[0]; + local *SOURCE_HANDLE; + my $source_filename = $_[1]; + my %count_data; + my $line_number; + my @result; + my $checkdata = $_[3]; + my $converted = $_[4]; + my $funcdata = $_[5]; + my $sumbrcount = $_[6]; + my $datafunc = get_hash_reverse($funcdata); + my @file; + + if ($_[2]) + { + %count_data = %{$_[2]}; + } + + if (!open(SOURCE_HANDLE, "<", $source_filename)) { + my @lines; + my $last_line = 0; + + if (!$ignore[$ERROR_SOURCE]) { + die("ERROR: cannot read $source_filename\n"); + } + + # Continue without source file + warn("WARNING: cannot read $source_filename!\n"); + + @lines = sort( { $a <=> $b } keys(%count_data)); + if (@lines) { + $last_line = $lines[scalar(@lines) - 1]; + } + return ( ":" ) if ($last_line < 1); + + # Simulate gcov behavior + for ($line_number = 1; $line_number <= $last_line; + $line_number++) { + push(@file, "/* EOF */"); + } + } else { + @file = ; + } + + write_source_prolog(*HTML_HANDLE); + $line_number = 0; + foreach (@file) { + $line_number++; + chomp($_); + + # Also remove CR from line-end + s/\015$//; + + # Source code matches coverage data? + if (defined($checkdata->{$line_number}) && + ($checkdata->{$line_number} ne md5_base64($_))) + { + die("ERROR: checksum mismatch at $source_filename:". + "$line_number\n"); + } + + push (@result, + write_source_line(HTML_HANDLE, $line_number, + $_, $count_data{$line_number}, + $converted->{$line_number}, + $sumbrcount->{$line_number})); + } + + close(SOURCE_HANDLE); + write_source_epilog(*HTML_HANDLE); + return(@result); +} + + +sub funcview_get_func_code($$$) +{ + my ($name, $base, $type) = @_; + my $result; + my $link; + + if ($sort && $type == 1) { + $link = "$name.func.$html_ext"; + } + $result = "Function Name"; + $result .= get_sort_code($link, "Sort by function name", $base); + + return $result; +} + +sub funcview_get_count_code($$$) +{ + my ($name, $base, $type) = @_; + my $result; + my $link; + + if ($sort && $type == 0) { + $link = "$name.func-sort-c.$html_ext"; + } + $result = "Hit count"; + $result .= get_sort_code($link, "Sort by hit count", $base); + + return $result; +} + +# +# funcview_get_sorted(funcdata, sumfncdata, sort_type) +# +# Depending on the value of sort_type, return a list of functions sorted +# by name (type 0) or by the associated call count (type 1). +# + +sub funcview_get_sorted($$$) +{ + my ($funcdata, $sumfncdata, $type) = @_; + + if ($type == 0) { + return sort(keys(%{$funcdata})); + } + return sort({ + $sumfncdata->{$b} == $sumfncdata->{$a} ? + $a cmp $b : $sumfncdata->{$a} <=> $sumfncdata->{$b} + } keys(%{$sumfncdata})); +} + +sub demangle_list($) +{ + my ($list) = @_; + my $tmpfile; + my $handle; + my %demangle; + my $demangle_arg = ""; + my %versions; + my $cppfilt_params = ""; + + # Write function names to file + ($handle, $tmpfile) = tempfile(); + die("ERROR: could not create temporary file") if (!defined($tmpfile)); + print($handle join("\n", @$list)); + close($handle); + + # Extra flag necessary on OS X so that symbols listed by gcov get demangled + # properly. + if ($^O eq "darwin" || $os_is_windows) { + $demangle_arg = "--no-strip-underscores"; + } + + # Build translation hash from c++filt output + open($handle, "-|", "c++filt $demangle_arg < $tmpfile") or + die("ERROR: could not run c++filt: $!\n"); + foreach my $func (@$list) { + my $translated = <$handle>; + my $version; + + last if (!defined($translated)); + chomp($translated); + + $version = ++$versions{$translated}; + $translated .= ".$version" if ($version > 1); + $demangle{$func} = $translated; + } + close($handle); + + if (scalar(keys(%demangle)) != scalar(@$list)) { + die("ERROR: c++filt output not as expected (". + scalar(keys(%demangle))." vs ".scalar(@$list).") lines\n"); + } + + unlink($tmpfile) or + warn("WARNING: could not remove temporary file $tmpfile: $!\n"); + + return \%demangle; +} + +# +# write_function_table(filehandle, source_file, sumcount, funcdata, +# sumfnccount, testfncdata, sumbrcount, testbrdata, +# base_name, base_dir, sort_type) +# +# Write an HTML table listing all functions in a source file, including +# also function call counts and line coverages inside of each function. +# +# Die on error. +# + +sub write_function_table(*$$$$$$$$$$) +{ + local *HTML_HANDLE = $_[0]; + my $source = $_[1]; + my $sumcount = $_[2]; + my $funcdata = $_[3]; + my $sumfncdata = $_[4]; + my $testfncdata = $_[5]; + my $sumbrcount = $_[6]; + my $testbrdata = $_[7]; + my $name = $_[8]; + my $base = $_[9]; + my $type = $_[10]; + my $func; + my $func_code; + my $count_code; + my $demangle; + + # Get HTML code for headings + $func_code = funcview_get_func_code($name, $base, $type); + $count_code = funcview_get_count_code($name, $base, $type); + write_html(*HTML_HANDLE, < + + + + + + +END_OF_HTML + ; + + # Get demangle translation hash + if ($demangle_cpp) { + $demangle = demangle_list([ sort(keys(%{$funcdata})) ]); + } + + # Get a sorted table + foreach $func (funcview_get_sorted($funcdata, $sumfncdata, $type)) { + if (!defined($funcdata->{$func})) + { + next; + } + + my $startline = $funcdata->{$func} - $func_offset; + my $name = $func; + my $count = $sumfncdata->{$name}; + my $countstyle; + + # Replace function name with demangled version if available + $name = $demangle->{$name} if (exists($demangle->{$name})); + + # Escape special characters + $name = escape_html($name); + if ($startline < 1) { + $startline = 1; + } + if ($count == 0) { + $countstyle = "coverFnLo"; + } else { + $countstyle = "coverFnHi"; + } + + write_html(*HTML_HANDLE, < + + + +END_OF_HTML + ; + } + write_html(*HTML_HANDLE, < +
+ +END_OF_HTML + ; +} + + +# +# info(printf_parameter) +# +# Use printf to write PRINTF_PARAMETER to stdout only when the $quiet flag +# is not set. +# + +sub info(@) +{ + if (!$quiet) + { + # Print info string + printf(@_); + } +} + + +# +# subtract_counts(data_ref, base_ref) +# + +sub subtract_counts($$) +{ + my %data = %{$_[0]}; + my %base = %{$_[1]}; + my $line; + my $data_count; + my $base_count; + my $hit = 0; + my $found = 0; + + foreach $line (keys(%data)) + { + $found++; + $data_count = $data{$line}; + $base_count = $base{$line}; + + if (defined($base_count)) + { + $data_count -= $base_count; + + # Make sure we don't get negative numbers + if ($data_count<0) { $data_count = 0; } + } + + $data{$line} = $data_count; + if ($data_count > 0) { $hit++; } + } + + return (\%data, $found, $hit); +} + + +# +# subtract_fnccounts(data, base) +# +# Subtract function call counts found in base from those in data. +# Return (data, f_found, f_hit). +# + +sub subtract_fnccounts($$) +{ + my %data; + my %base; + my $func; + my $data_count; + my $base_count; + my $fn_hit = 0; + my $fn_found = 0; + + %data = %{$_[0]} if (defined($_[0])); + %base = %{$_[1]} if (defined($_[1])); + foreach $func (keys(%data)) { + $fn_found++; + $data_count = $data{$func}; + $base_count = $base{$func}; + + if (defined($base_count)) { + $data_count -= $base_count; + + # Make sure we don't get negative numbers + if ($data_count < 0) { + $data_count = 0; + } + } + + $data{$func} = $data_count; + if ($data_count > 0) { + $fn_hit++; + } + } + + return (\%data, $fn_found, $fn_hit); +} + + +# +# apply_baseline(data_ref, baseline_ref) +# +# Subtract the execution counts found in the baseline hash referenced by +# BASELINE_REF from actual data in DATA_REF. +# + +sub apply_baseline($$) +{ + my %data_hash = %{$_[0]}; + my %base_hash = %{$_[1]}; + my $filename; + my $testname; + my $data; + my $data_testdata; + my $data_funcdata; + my $data_checkdata; + my $data_testfncdata; + my $data_testbrdata; + my $data_count; + my $data_testfnccount; + my $data_testbrcount; + my $base; + my $base_checkdata; + my $base_sumfnccount; + my $base_sumbrcount; + my $base_count; + my $sumcount; + my $sumfnccount; + my $sumbrcount; + my $found; + my $hit; + my $fn_found; + my $fn_hit; + my $br_found; + my $br_hit; + + foreach $filename (keys(%data_hash)) + { + # Get data set for data and baseline + $data = $data_hash{$filename}; + $base = $base_hash{$filename}; + + # Skip data entries for which no base entry exists + if (!defined($base)) + { + next; + } + + # Get set entries for data and baseline + ($data_testdata, undef, $data_funcdata, $data_checkdata, + $data_testfncdata, undef, $data_testbrdata) = + get_info_entry($data); + (undef, $base_count, undef, $base_checkdata, undef, + $base_sumfnccount, undef, $base_sumbrcount) = + get_info_entry($base); + + # Check for compatible checksums + merge_checksums($data_checkdata, $base_checkdata, $filename); + + # sumcount has to be calculated anew + $sumcount = {}; + $sumfnccount = {}; + $sumbrcount = {}; + + # For each test case, subtract test specific counts + foreach $testname (keys(%{$data_testdata})) + { + # Get counts of both data and baseline + $data_count = $data_testdata->{$testname}; + $data_testfnccount = $data_testfncdata->{$testname}; + $data_testbrcount = $data_testbrdata->{$testname}; + + ($data_count, undef, $hit) = + subtract_counts($data_count, $base_count); + ($data_testfnccount) = + subtract_fnccounts($data_testfnccount, + $base_sumfnccount); + ($data_testbrcount) = + combine_brcount($data_testbrcount, + $base_sumbrcount, $BR_SUB); + + + # Check whether this test case did hit any line at all + if ($hit > 0) + { + # Write back resulting hash + $data_testdata->{$testname} = $data_count; + $data_testfncdata->{$testname} = + $data_testfnccount; + $data_testbrdata->{$testname} = + $data_testbrcount; + } + else + { + # Delete test case which did not impact this + # file + delete($data_testdata->{$testname}); + delete($data_testfncdata->{$testname}); + delete($data_testbrdata->{$testname}); + } + + # Add counts to sum of counts + ($sumcount, $found, $hit) = + add_counts($sumcount, $data_count); + ($sumfnccount, $fn_found, $fn_hit) = + add_fnccount($sumfnccount, $data_testfnccount); + ($sumbrcount, $br_found, $br_hit) = + combine_brcount($sumbrcount, $data_testbrcount, + $BR_ADD); + } + + # Write back resulting entry + set_info_entry($data, $data_testdata, $sumcount, $data_funcdata, + $data_checkdata, $data_testfncdata, $sumfnccount, + $data_testbrdata, $sumbrcount, $found, $hit, + $fn_found, $fn_hit, $br_found, $br_hit); + + $data_hash{$filename} = $data; + } + + return (\%data_hash); +} + + +# +# remove_unused_descriptions() +# +# Removes all test descriptions from the global hash %test_description which +# are not present in %info_data. +# + +sub remove_unused_descriptions() +{ + my $filename; # The current filename + my %test_list; # Hash containing found test names + my $test_data; # Reference to hash test_name -> count_data + my $before; # Initial number of descriptions + my $after; # Remaining number of descriptions + + $before = scalar(keys(%test_description)); + + foreach $filename (keys(%info_data)) + { + ($test_data) = get_info_entry($info_data{$filename}); + foreach (keys(%{$test_data})) + { + $test_list{$_} = ""; + } + } + + # Remove descriptions for tests which are not in our list + foreach (keys(%test_description)) + { + if (!defined($test_list{$_})) + { + delete($test_description{$_}); + } + } + + $after = scalar(keys(%test_description)); + if ($after < $before) + { + info("Removed ".($before - $after). + " unused descriptions, $after remaining.\n"); + } +} + +# +# apply_prefix(filename, PREFIXES) +# +# If FILENAME begins with PREFIX from PREFIXES, remove PREFIX from FILENAME +# and return resulting string, otherwise return FILENAME. +# + +sub apply_prefix($@) +{ + my $filename = shift; + my @dir_prefix = @_; + + if (@dir_prefix) + { + foreach my $prefix (@dir_prefix) + { + if ($prefix ne "" && $filename =~ /^\Q$prefix$dir_sep\E/) + { + return substr($filename, length($prefix.$dir_sep)); + } + } + } + + return $filename; +} + +# +# system_no_output(mode, parameters) +# +# Call an external program using PARAMETERS while suppressing depending on +# the value of MODE: +# +# MODE & 1: suppress STDOUT +# MODE & 2: suppress STDERR +# +# Return 0 on success, non-zero otherwise. +# + +sub system_no_output($@) +{ + my $mode = shift; + my $result; + local *OLD_STDERR; + local *OLD_STDOUT; + + # Save old stdout and stderr handles + ($mode & 1) && open(OLD_STDOUT, ">>&", "STDOUT"); + ($mode & 2) && open(OLD_STDERR, ">>&", "STDERR"); + + # Redirect to /dev/null + ($mode & 1) && open(STDOUT, ">", "/dev/null"); + ($mode & 2) && open(STDERR, ">", "/dev/null"); + + system(@_); + $result = $?; + + # Close redirected handles + ($mode & 1) && close(STDOUT); + ($mode & 2) && close(STDERR); + + # Restore old handles + ($mode & 1) && open(STDOUT, ">>&", "OLD_STDOUT"); + ($mode & 2) && open(STDERR, ">>&", "OLD_STDERR"); + + return $result; +} + + +# +# read_config(filename) +# +# Read configuration file FILENAME and return a reference to a hash containing +# all valid key=value pairs found. +# + +sub read_config($) +{ + my $filename = $_[0]; + my %result; + my $key; + my $value; + local *HANDLE; + + if (!open(HANDLE, "<", $filename)) + { + warn("WARNING: cannot read configuration file $filename\n"); + return undef; + } + while () + { + chomp; + # Skip comments + s/#.*//; + # Remove leading blanks + s/^\s+//; + # Remove trailing blanks + s/\s+$//; + next unless length; + ($key, $value) = split(/\s*=\s*/, $_, 2); + if (defined($key) && defined($value)) + { + $result{$key} = $value; + } + else + { + warn("WARNING: malformed statement in line $. ". + "of configuration file $filename\n"); + } + } + close(HANDLE); + return \%result; +} + + +# +# apply_config(REF) +# +# REF is a reference to a hash containing the following mapping: +# +# key_string => var_ref +# +# where KEY_STRING is a keyword and VAR_REF is a reference to an associated +# variable. If the global configuration hashes CONFIG or OPT_RC contain a value +# for keyword KEY_STRING, VAR_REF will be assigned the value for that keyword. +# + +sub apply_config($) +{ + my $ref = $_[0]; + + foreach (keys(%{$ref})) + { + if (defined($opt_rc{$_})) { + ${$ref->{$_}} = $opt_rc{$_}; + } elsif (defined($config->{$_})) { + ${$ref->{$_}} = $config->{$_}; + } + } +} + + +# +# get_html_prolog(FILENAME) +# +# If FILENAME is defined, return contents of file. Otherwise return default +# HTML prolog. Die on error. +# + +sub get_html_prolog($) +{ + my $filename = $_[0]; + my $result = ""; + + if (defined($filename)) + { + local *HANDLE; + + open(HANDLE, "<", $filename) + or die("ERROR: cannot open html prolog $filename!\n"); + while () + { + $result .= $_; + } + close(HANDLE); + } + else + { + $result = < + + + + + + \@pagetitle\@ + + + + + +END_OF_HTML + ; + } + + return $result; +} + + +# +# get_html_epilog(FILENAME) +# +# If FILENAME is defined, return contents of file. Otherwise return default +# HTML epilog. Die on error. +# +sub get_html_epilog($) +{ + my $filename = $_[0]; + my $result = ""; + + if (defined($filename)) + { + local *HANDLE; + + open(HANDLE, "<", $filename) + or die("ERROR: cannot open html epilog $filename!\n"); + while () + { + $result .= $_; + } + close(HANDLE); + } + else + { + $result = < + +END_OF_HTML + ; + } + + return $result; + +} + +sub warn_handler($) +{ + my ($msg) = @_; + + warn("$tool_name: $msg"); +} + +sub die_handler($) +{ + my ($msg) = @_; + + die("$tool_name: $msg"); +} + +# +# parse_ignore_errors(@ignore_errors) +# +# Parse user input about which errors to ignore. +# + +sub parse_ignore_errors(@) +{ + my (@ignore_errors) = @_; + my @items; + my $item; + + return if (!@ignore_errors); + + foreach $item (@ignore_errors) { + $item =~ s/\s//g; + if ($item =~ /,/) { + # Split and add comma-separated parameters + push(@items, split(/,/, $item)); + } else { + # Add single parameter + push(@items, $item); + } + } + foreach $item (@items) { + my $item_id = $ERROR_ID{lc($item)}; + + if (!defined($item_id)) { + die("ERROR: unknown argument for --ignore-errors: ". + "$item\n"); + } + $ignore[$item_id] = 1; + } +} + +# +# parse_dir_prefix(@dir_prefix) +# +# Parse user input about the prefix list +# + +sub parse_dir_prefix(@) +{ + my (@opt_dir_prefix) = @_; + my $item; + + return if (!@opt_dir_prefix); + + foreach $item (@opt_dir_prefix) { + if ($item =~ /,/) { + # Split and add comma-separated parameters + push(@dir_prefix, split(/,/, $item)); + } else { + # Add single parameter + push(@dir_prefix, $item); + } + } +} + +# +# rate(hit, found[, suffix, precision, width]) +# +# Return the coverage rate [0..100] for HIT and FOUND values. 0 is only +# returned when HIT is 0. 100 is only returned when HIT equals FOUND. +# PRECISION specifies the precision of the result. SUFFIX defines a +# string that is appended to the result if FOUND is non-zero. Spaces +# are added to the start of the resulting string until it is at least WIDTH +# characters wide. +# + +sub rate($$;$$$) +{ + my ($hit, $found, $suffix, $precision, $width) = @_; + my $rate; + + # Assign defaults if necessary + $precision = $default_precision if (!defined($precision)); + $suffix = "" if (!defined($suffix)); + $width = 0 if (!defined($width)); + + return sprintf("%*s", $width, "-") if (!defined($found) || $found == 0); + $rate = sprintf("%.*f", $precision, $hit * 100 / $found); + + # Adjust rates if necessary + if ($rate == 0 && $hit > 0) { + $rate = sprintf("%.*f", $precision, 1 / 10 ** $precision); + } elsif ($rate == 100 && $hit != $found) { + $rate = sprintf("%.*f", $precision, 100 - 1 / 10 ** $precision); + } + + return sprintf("%*s", $width, $rate.$suffix); +} diff --git a/.tools/open-coverage-report.ts b/.tools/open-coverage-report.ts new file mode 100644 index 000000000..5da82eb78 --- /dev/null +++ b/.tools/open-coverage-report.ts @@ -0,0 +1,3 @@ +import { open } from 'https://deno.land/x/opener@v1.0.1/mod.ts'; + +await open(`file://${Deno.cwd()}/.coverage/report/index.html`); diff --git a/README.md b/README.md index c188cae12..5487e0030 100644 --- a/README.md +++ b/README.md @@ -1,39 +1,16 @@ -# Unity - Builder +# GameCI CLI -(Not affiliated with Unity Technologies) +The CLI is currently a work in progress. -GitHub Action to -[build Unity projects](https://github.com/marketplace/actions/unity-builder) -for different platforms. +We expect it to be working by the end of 2022. -Part of the GameCI open source project. -
-
- -[![Actions status](https://github.com/game-ci/unity-builder/workflows/Builds/badge.svg?event=push&branch=main)](https://github.com/game-ci/unity-builder/actions?query=branch%3Amain+event%3Apush+workflow%3A%22Builds) -[![lgtm - code quality](https://img.shields.io/lgtm/grade/javascript/g/webbertakken/unity-builder.svg?logo=lgtm&logoWidth=18)](https://lgtm.com/projects/g/webbertakken/unity-builder/context:javascript) -[![codecov - test coverage](https://codecov.io/gh/game-ci/unity-builder/branch/master/graph/badge.svg)](https://codecov.io/gh/game-ci/unity-builder) -
-
- -## How to use - -Find the -[docs](https://game.ci/docs/github/builder) -on the GameCI -[documentation website](https://game.ci/docs). - -## Related actions - -Visit the -GameCI Unity Actions -status repository for related Actions. +See [our roadmap](https://github.com/orgs/game-ci/projects/4/views/1) to follow our progress. ## Community Feel free to join us on -Discord -and engage with the community. +Discord and engage with the +community. ## Contributing diff --git a/action.yml b/action.yml index 1e0e6a95e..cf63576bc 100644 --- a/action.yml +++ b/action.yml @@ -6,7 +6,7 @@ inputs: required: true default: '' description: 'Platform that the build should target.' - unityVersion: + engineVersion: required: false default: 'auto' description: 'Version of unity to use for building the project. Use "auto" to get from your ProjectSettings/ProjectVersion.txt' @@ -169,11 +169,63 @@ inputs: outputs: volume: description: 'The Persistent Volume (PV) where the build artifacts have been stored by Kubernetes' + value: '' buildVersion: description: 'The generated version used for the Unity build' + value: '' +runs: + using: 'composite' + steps: + - run: echo "Using GameCI CLI to build project" + shell: bash + - uses: denoland/setup-deno@v1 + with: + deno-version: v1.x + - run: | + deno run --allow-run ./src/index.ts build \ + --targetPlatform="${{ inputs.targetPlatform }}" \ + --engineVersion="${{ inputs.engineVersion }}" \ + --engineVersion="${{ inputs.engineVersion }}" \ + --customImage="${{ inputs.customImage }}" \ + --projectPath="${{ inputs.projectPath }}" \ + --buildName="${{ inputs.buildName }}" \ + --buildsPath="${{ inputs.buildsPath }}" \ + --buildMethod="${{ inputs.buildMethod }}" \ + --customParameters="${{ inputs.customParameters }}" \ + --versioning="${{ inputs.versioning }}" \ + --version="${{ inputs.version }}" \ + --androidVersionCode="${{ inputs.androidVersionCode }}" \ + --androidAppBundle="${{ inputs.androidAppBundle }}" \ + --androidKeystoreName="${{ inputs.androidKeystoreName }}" \ + --androidKeystoreBase64="${{ inputs.androidKeystoreBase64 }}" \ + --androidKeystorePass="${{ inputs.androidKeystorePass }}" \ + --androidKeyaliasName="${{ inputs.androidKeyaliasName }}" \ + --androidKeyaliasPass="${{ inputs.androidKeyaliasPass }}" \ + --androidTargetSdkVersion="${{ inputs.androidTargetSdkVersion }}" \ + --sshAgent="${{ inputs.sshAgent }}" \ + --gitPrivateToken="${{ inputs.gitPrivateToken }}" \ + --chownFilesTo="${{ inputs.chownFilesTo }}" \ + --allowDirtyBuild="${{ inputs.allowDirtyBuild }}" \ + --postBuildSteps="${{ inputs.postBuildSteps }}" \ + --preBuildSteps="${{ inputs.preBuildSteps }}" \ + --customJobHooks="${{ inputs.customJobHooks }}" \ + --customJob="${{ inputs.customJob }}" \ + --awsBaseStackName="${{ inputs.awsBaseStackName }}" \ + --cloudRunnerCluster="${{ inputs.cloudRunnerCluster }}" \ + --cloudRunnerCpu="${{ inputs.cloudRunnerCpu }}" \ + --cloudRunnerMemory="${{ inputs.cloudRunnerMemory }}" \ + --cachePushOverrideCommand="${{ inputs.cachePushOverrideCommand }}" \ + --cachePullOverrideCommand="${{ inputs.cachePullOverrideCommand }}" \ + --readInputFromOverrideList="${{ inputs.readInputFromOverrideList }}" \ + --readInputOverrideCommand="${{ inputs.readInputOverrideCommand }}" \ + --kubeConfig="${{ inputs.kubeConfig }}" \ + --kubeVolume="${{ inputs.kubeVolume }}" \ + --kubeStorageClass="${{ inputs.kubeStorageClass }}" \ + --kubeVolumeSize="${{ inputs.kubeVolumeSize }}" \ + --cacheKey="${{ inputs.cacheKey }}" \ + --checkDependencyHealthOverride="${{ inputs.checkDependencyHealthOverride }}" \ + --startDependenciesOverride="${{ inputs.startDependenciesOverride }}" + shell: bash branding: icon: 'box' color: 'gray-dark' -runs: - using: 'node12' - main: 'dist/index.js' diff --git a/deno.json b/deno.json new file mode 100644 index 000000000..ea6676af7 --- /dev/null +++ b/deno.json @@ -0,0 +1,48 @@ +{ + "tasks": { + "config": "deno run -A .tools/open-config-folder.ts", + "test": "deno test -A ./src/integrity.test.ts", + "coverage": "rm -rf .coverage ; deno test -A ./src/integrity.test.ts --coverage=.coverage/raw ; deno coverage .coverage/raw --lcov --output=.coverage/raw/.lcov ; perl .tools/lcov/genhtml.perl -q -o .coverage/report .coverage/raw/.lcov && deno run -A .tools/open-coverage-report.ts" + }, + "compilerOptions": { + "allowJs": true, + "lib": [ + "deno.window" + ], + "strict": true + }, + "lint": { + "files": { + "include": [ + "src/" + ], + "exclude": [] + }, + "rules": { + "tags": [ + "recommended" + ], + "include": [ + "ban-untagged-todo" + ], + "exclude": [ + "no-unused-vars" + ] + } + }, + "fmt": { + "files": { + "include": [ + "src/" + ], + "exclude": [] + }, + "options": { + "useTabs": false, + "lineWidth": 120, + "indentWidth": 2, + "singleQuote": true, + "proseWrap": "always" + } + } +} diff --git a/dist/platforms/windows/entrypoint.ps1 b/dist/platforms/windows/entrypoint.ps1 index b5255d11e..01f666b50 100644 --- a/dist/platforms/windows/entrypoint.ps1 +++ b/dist/platforms/windows/entrypoint.ps1 @@ -1,13 +1,13 @@ -# Activate Unity -& "c:\steps\activate.ps1" - # Import any necessary registry keys, ie: location of windows 10 sdk # No guarantee that there will be any necessary registry keys, ie: tvOS -Get-ChildItem -Path c:\regkeys -File | Foreach {reg import $_.fullname} +Get-ChildItem -Path c:\registry-keys -File | Foreach {reg import $_.fullname} # Register the Visual Studio installation so Unity can find it regsvr32 C:\ProgramData\Microsoft\VisualStudio\Setup\x64\Microsoft.VisualStudio.Setup.Configuration.Native.dll +# Activate Unity +& "c:\steps\activate.ps1" + # Build the project & "c:\steps\build.ps1" diff --git a/lefthook.yml b/lefthook.yml index 915525f8d..c31e23294 100644 --- a/lefthook.yml +++ b/lefthook.yml @@ -23,9 +23,9 @@ pre-commit: glob: '*.{js,jsx,ts,tsx}' exclude: 'dist/' run: yarn jest --passWithNoTests --findRelatedTests {staged_files} - build distributables: - skip: ['merge', 'rebase'] - run: yarn build && git add dist + # build distributables: + # skip: ['merge', 'rebase'] + # run: yarn build && git add dist make shell script executable: glob: '*.sh' run: git update-index --chmod=+x diff --git a/logs/.gitkeep b/logs/.gitkeep new file mode 100644 index 000000000..e69de29bb diff --git a/package.json b/package.json index 3b83cc4b9..61a081d65 100644 --- a/package.json +++ b/package.json @@ -24,46 +24,28 @@ "test-i-aws": "cross-env cloudRunnerTests=true cloudRunnerCluster=aws yarn test -i -t \"cloud runner\"", "test-i-k8s": "cross-env cloudRunnerTests=true cloudRunnerCluster=k8s yarn test -i -t \"cloud runner\"" }, - "dependencies": { - "@actions/core": "^1.9.1", - "@actions/exec": "^1.1.0", - "@actions/github": "^5.0.0", - "@kubernetes/client-node": "^0.16.3", - "@octokit/core": "^3.5.1", - "async-wait-until": "^2.0.12", - "aws-sdk": "^2.1081.0", - "base-64": "^1.0.0", - "commander": "^9.0.0", - "commander-ts": "^0.2.0", - "kubernetes-client": "^9.0.0", - "nanoid": "^3.3.1", - "reflect-metadata": "^0.1.13", - "semver": "^7.3.5", - "unity-changeset": "^1.6.0", - "uuid": "^8.3.2", - "yaml": "^1.10.2" - }, + "dependencies": {}, "devDependencies": { "@arkweid/lefthook": "^0.7.7", - "@types/jest": "^27.4.1", - "@types/node": "^17.0.23", + "@types/jest": "^28.1.6", + "@types/node": "^18.7.13", "@types/semver": "^7.3.9", - "@typescript-eslint/parser": "4.8.1", + "@typescript-eslint/parser": "5.32.0", "@vercel/ncc": "^0.33.3", "cross-env": "^7.0.3", - "eslint": "7.17.0", - "eslint-config-prettier": "8.1.0", - "eslint-plugin-github": "^4.1.1", - "eslint-plugin-jest": "24.1.3", - "eslint-plugin-prettier": "^3.3.1", - "eslint-plugin-unicorn": "28.0.2", - "jest": "^27.5.1", - "jest-circus": "^27.5.1", - "jest-fail-on-console": "^2.3.0", + "eslint": "^8.21.0", + "eslint-config-prettier": "8.5.0", + "eslint-plugin-github": "4.3.7", + "eslint-plugin-jest": "26.7.0", + "eslint-plugin-prettier": "4.2.1", + "eslint-plugin-unicorn": "43.0.2", + "jest": "^28.1.3", + "jest-circus": "^28.1.3", + "jest-fail-on-console": "^2.4.2", "js-yaml": "^4.1.0", - "prettier": "^2.5.1", - "ts-jest": "^27.1.3", + "prettier": "2.7.1", + "ts-jest": "^28.0.7", "ts-node": "10.4.0", - "typescript": "4.1.3" + "typescript": "4.3.5" } } diff --git a/src/cli.ts b/src/cli.ts new file mode 100644 index 000000000..2c0220c9b --- /dev/null +++ b/src/cli.ts @@ -0,0 +1,182 @@ +import { yargs, YargsInstance, YargsArguments, getHomeDir } from './dependencies.ts'; +import { engineDetection } from './middleware/engine-detection/index.ts'; +import { CommandInterface } from './command/command-interface.ts'; +import { configureLogger } from './middleware/logger-verbosity/index.ts'; +import { CommandFactory } from './command/command-factory.ts'; +import { Engine } from './model/engine/engine.ts'; +import { vcsDetection } from './middleware/vcs-detection/index.ts'; + +export class Cli { + private readonly yargs: YargsInstance; + private readonly cliStorageAbsolutePath: string; + private readonly cliStorageCanonicalPath: string; + private readonly configFileName: string; + private command: CommandInterface; + + constructor() { + this.cliStorageAbsolutePath = `${getHomeDir()}/.game-ci`; + this.cliStorageCanonicalPath = '~/.game-ci'; + this.configFileName = 'config.json'; + this.yargs = yargs(Deno.args); + } + + public async validateAndParseArguments() { + this.globalSettings(); + this.configureLogger(); + this.globalOptions(); + + await this.registerConfigCommand(); + await this.registerBuildCommand(); + + await this.parse(); + + if (log.isVeryVerbose) { + log.debug(`Parsed command: ${this.command.name} (${this.command.constructor.name})`); + log.debug(`Parsed arguments: ${JSON.stringify(this.options, null, 2)}`); + } + + return { + command: this.command, + options: this.options, + }; + } + + private globalSettings() { + const defaultCanonicalPath = `${this.cliStorageCanonicalPath}/${this.configFileName}`; + const defaultAbsolutePath = `${this.cliStorageAbsolutePath}/${this.configFileName}`; + + this.yargs + .config('config', `default: ${defaultCanonicalPath}`, async (override: string) => { + const configPath = override || defaultAbsolutePath; + + return JSON.parse(await Deno.readTextFile(configPath)); + }) + .parserConfiguration({ + 'dot-notation': false, + 'duplicate-arguments-array': false, + 'negation-prefix': false, + 'strip-aliased': true, + 'strip-dashed': true, + }); + + // Todo - enable `.env()` after this is merged: https://github.com/yargs/yargs/pull/2231 + // this.yargs.env(); + } + + private configureLogger() { + this.yargs + .options('quiet', { + alias: 'q', + description: 'Suppress all output', + type: 'boolean', + demandOption: false, + default: false, + }) + .options('verbose', { + alias: 'v', + description: 'Enable verbose logging', + type: 'boolean', + demandOption: false, + default: false, + }) + .options('veryVerbose', { + alias: 'vv', + description: 'Enable very verbose logging', + type: 'boolean', + demandOption: false, + default: false, + }) + .options('maxVerbose', { + alias: 'vvv', + description: 'Enable debug logging', + demandOption: false, + type: 'boolean', + default: false, + }) + .default([{ logLevel: 'placeholder' }, { logLevelName: 'placeholder' }]) + .middleware([configureLogger], true); + } + + private globalOptions() { + this.yargs + .fail(Cli.handleFailure) + .help(false) // Fixes broken `_handle` in yargs 17.0.0 + .version(false) // Fixes broken `_handle` in yargs 17.0.0 + .showHelpOnFail(false) // Fixes broken `_handle` in yargs 17.0.0 + .epilogue('for more information, find our manual at https://game.ci/docs/cli') + .middleware([]) + .exitProcess(true) // Fixes broken `_handle` in yargs 17.0.0 + .strict(true); + } + + private async registerBuildCommand() { + this.yargs.command('build [projectPath]', 'Builds a project that you want to build', async (yargs) => { + yargs + .positional('projectPath', { + describe: 'Path to the project', + type: 'string', + demandOption: false, + default: '.', + }) + .coerce('projectPath', async (arg) => { + return arg.replace(/^~/, getHomeDir()).replace(/\/$/, ''); + }) + .default('engine', '') + .default('engineVersion', '') + .middleware([engineDetection], true) + .default('branch', '') + .middleware([vcsDetection], true) + + // Todo - remove these lines with release 3.0.0 + .option('unityVersion', { + describe: 'Override the engine version to be used', + type: 'string', + default: '', + }) + .deprecateOption('unityVersion', 'This parameter will be removed. Use engineVersion instead') + .middleware( + [ + async (args) => { + if (!args.unityVersion || args.unityVersion === 'auto' || args.engine !== Engine.unity) return; + + args.engineVersion = args.unityVersion; + args.unityVersion = undefined; + }, + ], + true, + ) + + // End todo + .middleware([async (args) => this.registerCommand(args, yargs)]); + }); + } + + private async registerConfigCommand() { + this.yargs.command('config', 'GameCI CLI configuration', async (yargs) => { + yargs + .command('open', 'Opens the CLI configuration folder', async (yargs) => {}) + .middleware([async (args) => this.registerCommand(args, yargs)]); + }); + } + + private async registerCommand(args: YargsArguments, yargs: YargsInstance) { + const { engine, engineVersion, _: command } = args; + + this.command = new CommandFactory().selectEngine(engine, engineVersion).createCommand(command); + + await this.command.configureOptions(yargs); + } + + private async parse() { + const { _, $0, ...options } = await this.yargs.parseAsync(); + + this.options = options; + } + + private static handleFailure(message: string, error: Error, yargs: YargsInstance) { + if (error) throw error; + + log.warning(message); + Deno.exit(1); + } +} diff --git a/src/command-options/android-options.ts b/src/command-options/android-options.ts new file mode 100644 index 000000000..633e7d498 --- /dev/null +++ b/src/command-options/android-options.ts @@ -0,0 +1,83 @@ +import { YargsInstance, YargsArguments } from '../dependencies.ts'; + +export class AndroidOptions { + public static configureCommonOptions(yargs: YargsInstance): void { + yargs + .option('androidAppBundle', { + description: 'Build an Android App Bundle', + type: 'boolean', + demandOption: false, + default: false, + }) + .options({ + androidKeystoreName: { + description: 'Name of the keystore', + type: 'string', + demandOption: false, + default: '', + }, + androidKeystoreBase64: { + description: 'Base64 encoded contents of the keystore', + type: 'string', + demandOption: false, + default: '', + }, + androidKeystorePass: { + description: 'Password for the keystore', + type: 'string', + demandOption: false, + default: '', + deprecated: 'Use androidKeystorePassword instead', + }, + androidKeystorePassword: { + description: 'Password for the keystore', + type: 'string', + demandOption: false, + default: '', + }, + androidKeyAlias: { + description: 'Alias for the keystore', + type: 'string', + demandOption: false, + default: '', + }, + androidKeyAliasName: { + description: 'Name of the keystore', + type: 'string', + demandOption: false, + default: '', + deprecated: 'Use androidKeyAlias instead', + }, + androidKeyAliasPassword: { + description: 'Password for the androidKeyAlias', + type: 'string', + demandOption: false, + default: '', + requires: ['androidKeyAlias'], + }, + androidKeyAliasPass: { + description: 'Password for the androidKeyAlias', + type: 'string', + demandOption: false, + default: '', + deprecated: 'Use androidKeyAliasPassword instead', + }, + }) + .option('androidTargetSdkVersion', { + description: 'Custom Android SDK target version', + type: 'number', + demandOption: false, + default: '', + }) + .default('androidSdkManagerParameters', '') // Placeholder, consumed in middleware + .middleware([AndroidOptions.determineSdkManagerParameters]); + } + + private static determineSdkManagerParameters(argv: YargsArguments) { + const { androidTargetSdkVersion } = argv; + + if (!androidTargetSdkVersion) return; + + argv.androidSdkManagerParameters = `platforms;android-${androidTargetSdkVersion}`; + } +} diff --git a/src/command-options/build-options.ts b/src/command-options/build-options.ts new file mode 100644 index 000000000..628f41c9d --- /dev/null +++ b/src/command-options/build-options.ts @@ -0,0 +1,28 @@ +import { YargsInstance } from '../dependencies.ts'; +import Unity from '../model/unity/unity.ts'; + +export class BuildOptions { + public static configure(yargs: YargsInstance): void { + yargs + .demandOption('targetPlatform', 'Target platform is mandatory for builds') + .option('buildName', { + description: 'Name of the build', + type: 'string', + default: '', + }) + .option('buildsPath', { + description: 'Path for outputting the builds to', + type: 'string', + demandOption: false, + default: 'build', + }) + .default('buildPath', '') + .default('buildFile', '') + .middleware(async (argv) => { + const { buildName, buildsPath, targetPlatform, androidAppBundle } = argv; + argv.buildName = buildName || targetPlatform; + argv.buildPath = `${buildsPath}/${targetPlatform}`; + argv.buildFile = Unity.determineBuildFileName(buildName, targetPlatform, androidAppBundle); + }); + } +} diff --git a/src/command-options/unity-options.ts b/src/command-options/unity-options.ts new file mode 100644 index 000000000..1b8476238 --- /dev/null +++ b/src/command-options/unity-options.ts @@ -0,0 +1,61 @@ +import type { YargsInstance } from '../dependencies.ts'; +import UnityTargetPlatform from '../model/unity/unity-target-platform.ts'; +import { UnityTargetPlatforms } from '../model/unity/unity-target-platforms.ts'; + +export class UnityOptions { + public static configure = async (yargs: YargsInstance) => { + yargs + .option('targetPlatform', { + alias: 't', + description: 'The platform to build your project for', + choices: UnityTargetPlatforms.all, + demandOption: false, + default: UnityTargetPlatform.default, + }) + .options({ + unityEmail: { + alias: 'u', + description: 'Email address for your Unity account', + type: 'string', + demandOption: false, + default: '', + }, + unityPassword: { + alias: 'p', + description: 'Password for your Unity account', + type: 'string', + demandOption: false, + default: '', + }, + unitySerial: { + alias: 's', + description: 'Serial number identifying a pro-license seat', + type: 'string', + demandOption: false, + default: '', + }, + unityLicense: { + alias: 'l', + description: 'Contents of, or path to your Unity License File (.ulf)', + type: 'string', + demandOption: false, + default: '', + }, + }) + .coerce('unityLicense', async (arg) => { + return arg.endsWith('.ulf') ? Deno.readTextFile(arg, { encoding: 'utf8' }) : arg; + }) + .option('customImage', { + description: String.dedent` + Custom docker image to use inside the command. + For more information see https://game.ci/docs/docker/versions`, + type: 'string', + }) + .option('usymUploadAuthToken', { + description: '', + type: 'string', + demandOption: false, + default: '', + }); + }; +} diff --git a/src/command-options/versioning-options.ts b/src/command-options/versioning-options.ts new file mode 100644 index 000000000..58de00a79 --- /dev/null +++ b/src/command-options/versioning-options.ts @@ -0,0 +1,37 @@ +import { YargsInstance } from '../dependencies.ts'; +import { VersioningStrategies } from '../model/versioning/versioning-strategies.ts'; +import { VersioningStrategy } from '../model/versioning/versioning-strategy.ts'; +import { buildVersioning } from '../middleware/build-versioning/index.ts'; + +export class VersioningOptions { + public static async configure(yargs: YargsInstance): void { + yargs + .option('versioningStrategy', { + description: 'Versioning strategy', + choices: VersioningStrategies.all, + demandOption: true, + default: VersioningStrategy.Semantic, + }) + .option('version', { + description: String.dedent` + Custom version to use for the build. + Only used when versioningStrategy is set to Custom`, + type: 'string', + default: '', + }) + .option('androidVersionCode', { + description: String.dedent` + Custom version code for android specifically.`, + type: 'string', + default: '', + }) + .option('allowDirtyBuild', { + description: 'Allow a dirty build', + type: 'boolean', + demandOption: false, + default: false, + }) + .default('buildVersion', 'placeholder') + .middleware([buildVersioning]); + } +} diff --git a/src/command/build/unity-build-command.ts b/src/command/build/unity-build-command.ts new file mode 100644 index 000000000..8ec5ffec3 --- /dev/null +++ b/src/command/build/unity-build-command.ts @@ -0,0 +1,41 @@ +import { CommandInterface } from '../command-interface.ts'; +import { Action, Cache, Docker, ImageTag, Input, Output } from '../../model/index.ts'; +import PlatformSetup from '../../model/platform-setup.ts'; +import MacBuilder from '../../model/mac-builder.ts'; +import { CommandBase } from '../command-base.ts'; +import { UnityOptions } from '../../command-options/unity-options.ts'; +import { YargsInstance, YargsArguments } from '../../dependencies.ts'; +import { VersioningOptions } from '../../command-options/versioning-options.ts'; +import { BuildOptions } from '../../command-options/build-options.ts'; + +export class UnityBuildCommand extends CommandBase implements CommandInterface { + public async execute(options: YargsArguments): Promise { + // Todo - rework this without needing this.options, use parameters from cli instead. + // const { workspace, actionFolder } = Action; + // const { parameters, env } = this.options; + // + // Action.checkCompatibility(); + // Cache.verify(); + // + // const baseImage = new ImageTag(options); + // log.debug('baseImage', baseImage); + // + // await PlatformSetup.setup(parameters, actionFolder); + // if (env.getOS() === 'darwin') { + // MacBuilder.run(actionFolder, workspace, parameters); + // } else { + // await Docker.run(baseImage, { workspace, actionFolder, ...parameters }); + // } + // + // // Set output + // await Output.setBuildVersion(parameters.buildVersion); + + return true; + } + + public async configureOptions(yargs: YargsInstance): Promise { + await UnityOptions.configure(yargs); + await VersioningOptions.configure(yargs); + await BuildOptions.configure(yargs); + } +} diff --git a/src/command/command-base.ts b/src/command/command-base.ts new file mode 100644 index 000000000..1fb93e2ae --- /dev/null +++ b/src/command/command-base.ts @@ -0,0 +1,18 @@ +import { CommandInterface } from './command-interface.ts'; +import { YargsArguments, YargsInstance } from '../dependencies.ts'; + +export class CommandBase implements CommandInterface { + public readonly name: string; + + constructor(name: string) { + this.name = name.charAt(0).toUpperCase() + name.slice(1); + } + + public execute(options: YargsArguments): Promise { + throw new Error('Method not implemented.'); + } + + public configureOptions(yargs: YargsInstance): Promise { + throw new Error('Method not implemented.'); + } +} diff --git a/src/command/command-factory.ts b/src/command/command-factory.ts new file mode 100644 index 000000000..7c17159cb --- /dev/null +++ b/src/command/command-factory.ts @@ -0,0 +1,53 @@ +import { NonExistentCommand } from './null/non-existent-command.ts'; +import { UnityBuildCommand } from './build/unity-build-command.ts'; +import { CommandInterface } from './command-interface.ts'; +import { Engine } from '../model/engine/engine.ts'; +import { OpenConfigFolderCommand } from './config/open-config-folder-command.ts'; + +export class CommandFactory { + constructor() {} + + selectEngine(engine: string, engineVersion: string) { + this.engine = engine; + this.engineVersion = engineVersion; + + return this; + } + + public createCommand(commandArray: string[]): CommandInterface { + // Structure looks like: _: [ "build" ], or _: [ "config", "open" ] + const [command, ...subCommands] = commandArray; + + if (command === 'config') { + return this.createConfigCommand(command, subCommands); + } + + switch (this.engine) { + case Engine.unity: + return this.createUnityCommand(command, subCommands); + default: + throw new Error(`Engine ${this.engine} is not yet supported.`); + } + } + + private createConfigCommand(command: string, subCommands: string[]) { + switch (subCommands[0]) { + case 'open': + return new OpenConfigFolderCommand(command); + default: + return new NonExistentCommand([command, ...subCommands].join(' ')); + } + } + + private createUnityCommand(command: string, subCommands: string[]) { + switch (command) { + case 'build': + return new UnityBuildCommand(command); + + // case 'remote-build': + // return new UnityRemoteBuildCommand(commandName); + default: + return new NonExistentCommand([command, ...subCommands].join(' ')); + } + } +} diff --git a/src/command/command-interface.ts b/src/command/command-interface.ts new file mode 100644 index 000000000..e32430b5d --- /dev/null +++ b/src/command/command-interface.ts @@ -0,0 +1,7 @@ +import { YargsInstance, YargsArguments } from '../dependencies.ts'; + +export interface CommandInterface { + name: string; + execute: (options: YargsArguments) => Promise; + configureOptions: (instance: YargsInstance) => Promise; +} diff --git a/src/command/config/open-config-folder-command.ts b/src/command/config/open-config-folder-command.ts new file mode 100644 index 000000000..0bfda4df5 --- /dev/null +++ b/src/command/config/open-config-folder-command.ts @@ -0,0 +1,17 @@ +import { CommandInterface } from '../command-interface.ts'; +import { CommandBase } from '../command-base.ts'; +import { YargsInstance, YargsArguments } from '../../dependencies.ts'; + +import { default as getHomeDir } from 'https://deno.land/x/dir@1.5.1/home_dir/mod.ts'; +import { open } from 'https://deno.land/x/opener@v1.0.1/mod.ts'; + +export class OpenConfigFolderCommand extends CommandBase implements CommandInterface { + public async execute(options: YargsArguments): Promise { + const cliStorageAbsolutePath = `${getHomeDir()}/.game-ci`; + await open(`file://${cliStorageAbsolutePath}/`); + + return true; + } + + public async configureOptions(yargs: YargsInstance): Promise {} +} diff --git a/src/command/null/non-existent-command.ts b/src/command/null/non-existent-command.ts new file mode 100644 index 000000000..220959d3c --- /dev/null +++ b/src/command/null/non-existent-command.ts @@ -0,0 +1,11 @@ +import { CommandInterface } from '../command-interface.ts'; +import { YargsInstance, YargsArguments } from '../../dependencies.ts'; +import { CommandBase } from '../command-base.ts'; + +export class NonExistentCommand extends CommandBase implements CommandInterface { + public execute(options: YargsArguments): Promise { + throw new Error(`Command "${this.name}" does not exist`); + } + + public async configureOptions(yargs: YargsInstance): Promise {} +} diff --git a/src/command/remote/unity-remote-build-command.ts b/src/command/remote/unity-remote-build-command.ts new file mode 100644 index 000000000..5ef65cb1a --- /dev/null +++ b/src/command/remote/unity-remote-build-command.ts @@ -0,0 +1,68 @@ +import { CommandInterface } from '../command-interface.ts'; +import { Options } from '../../config/options.ts'; +import { CloudRunner, ImageTag, Input, Output } from '../../model/index.ts'; +import { core, nanoid } from '../../dependencies.ts'; +import Parameters from '../../model/parameters.ts'; +import { GitRepoReader } from '../../model/input-readers/git-repo.ts'; +import { Cli } from '../../model/cli/cli.ts'; +import CloudRunnerConstants from '../../model/cloud-runner/services/cloud-runner-constants.ts'; +import CloudRunnerBuildGuid from '../../model/cloud-runner/services/cloud-runner-guid.ts'; +import { GithubCliReader } from '../../model/input-readers/github-cli.ts'; +import { CommandBase } from '../command-base.ts'; + +// Todo - Verify this entire flow +export class UnityRemoteBuildCommand extends CommandBase implements CommandInterface { + public async validate() { + await super.validate(); + } + + public async parseParameters(input: Input, parameters: Parameters): Promise> { + const cloudRunnerCluster = Cli.isCliMode + ? this.input.getInput('cloudRunnerCluster') || 'aws' + : this.input.getInput('cloudRunnerCluster') || 'local'; + + return { + cloudRunnerCluster, + cloudRunnerBranch: input.cloudRunnerBranch.split('/').reverse()[0], + cloudRunnerIntegrationTests: input.cloudRunnerTests, + githubRepo: input.githubRepo || (await GitRepoReader.GetRemote()) || 'game-ci/unity-builder', + gitPrivateToken: parameters.gitPrivateToken || (await GithubCliReader.GetGitHubAuthToken()), + isCliMode: Cli.isCliMode, + awsStackName: input.awsBaseStackName, + cloudRunnerBuilderPlatform: input.cloudRunnerBuilderPlatform, + awsBaseStackName: input.awsBaseStackName, + kubeConfig: input.kubeConfig, + cloudRunnerMemory: input.cloudRunnerMemory, + cloudRunnerCpu: input.cloudRunnerCpu, + kubeVolumeSize: input.kubeVolumeSize, + kubeVolume: input.kubeVolume, + postBuildSteps: input.postBuildSteps, + preBuildSteps: input.preBuildSteps, + runNumber: input.runNumber, + gitSha: input.gitSha, + logId: nanoid.customAlphabet(CloudRunnerConstants.alphabet, 9)(), + buildGuid: CloudRunnerBuildGuid.generateGuid(input.runNumber, input.targetPlatform), + customJobHooks: input.customJobHooks(), + cachePullOverrideCommand: input.cachePullOverrideCommand(), + cachePushOverrideCommand: input.cachePushOverrideCommand(), + readInputOverrideCommand: input.readInputOverrideCommand(), + readInputFromOverrideList: input.readInputFromOverrideList(), + kubeStorageClass: input.kubeStorageClass, + checkDependencyHealthOverride: input.checkDependencyHealthOverride, + startDependenciesOverride: input.startDependenciesOverride, + cacheKey: input.cacheKey, + }; + } + + public async execute(options: Options): Promise { + const { buildParameters } = options; + const baseImage = new ImageTag(buildParameters); + + const result = await CloudRunner.run(buildParameters, baseImage.toString()); + const { status, output } = result; + + await Output.setBuildVersion(buildParameters.buildVersion); + + return status.success; + } +} diff --git a/src/config/options.ts b/src/config/options.ts new file mode 100644 index 000000000..cef90ca82 --- /dev/null +++ b/src/config/options.ts @@ -0,0 +1,27 @@ +import { CliArguments } from '../core/cli/cli-arguments.ts'; +import { Parameters, Input } from '../model/index.ts'; +import { CommandInterface } from '../command/command-interface.ts'; +import { Environment } from '../core/env/environment.ts'; + +export class Options { + public input: Input; + public parameters: Parameters; + private readonly env: Environment; + private command: CommandInterface; + + constructor(command: CommandInterface, env: Environment) { + this.env = env; + this.command = command; + + return this; + } + + public async generateParameters(args: CliArguments) { + this.input = new Input(args); + this.parameters = await new Parameters(this.input, this.env).registerCommand(this.command).parse(); + + log.info('Parameters generated.'); + + return this; + } +} diff --git a/src/core/cli/arguments-parser.ts b/src/core/cli/arguments-parser.ts new file mode 100644 index 000000000..80f5a6831 --- /dev/null +++ b/src/core/cli/arguments-parser.ts @@ -0,0 +1,28 @@ +import { parseArgv } from './parse-argv.ts'; + +export class ArgumentsParser { + public parse(cliArguments: string[]) { + const [commandName, ...rest] = cliArguments; + const { subCommands, args } = parseArgv(rest); + + let verbosity; + if (args.has('vvv') || args.has('max-verbose') || args.has('maxVerbose') || args.has('debug')) { + verbosity = 3; + } else if (args.has('vv') || args.has('very-verbose') || args.has('veryVerbose')) { + verbosity = 2; + } else if (args.has('v') || args.has('verbose')) { + verbosity = 1; + } else if (args.has('q') || args.has('quiet')) { + verbosity = -1; + } else { + verbosity = 0; + } + + return { + commandName, + subCommands, + args, + verbosity, + }; + } +} diff --git a/src/core/cli/cli-arguments.ts b/src/core/cli/cli-arguments.ts new file mode 100644 index 000000000..107b2e9c8 --- /dev/null +++ b/src/core/cli/cli-arguments.ts @@ -0,0 +1 @@ +export type CliArguments = Map; diff --git a/src/core/cli/parse-argv.ts b/src/core/cli/parse-argv.ts new file mode 100644 index 000000000..4ed972a64 --- /dev/null +++ b/src/core/cli/parse-argv.ts @@ -0,0 +1,66 @@ +import { CliArguments } from './cli-arguments.ts'; + +/** + * Parse command line arguments + * + * Usage: + * console.dir(parseArgv(Deno.args)); // Deno + * console.log(parseArgv(process.argv)); // Node + * + * Example: + * deno run my-script my-project -test1=1 -test2 "2" -test3 -test4 false -test5 "one" -test6= -test7=9BX9 + * + * Output: + * [ + * [ 'my-project' ], + * Map { + * "test1" => 1, + * "test2" => 2, + * "test3" => true, + * "test4" => false, + * "test5" => "one", + * "test6" => "", + * "test7" => "9BX9" + * } + * ] + */ +export const parseArgv = (argv: string[] = [], { verbose = false } = {}): CliArguments => { + const subCommands: string[] = []; + const args = new Map(); + + let hasParsedSubCommands = false; + for (let current = 0, next = 1; current < argv.length; current += 1, next += 1) { + // Detect subCommands + if (!hasParsedSubCommands) { + if (argv[current].startsWith('-')) { + hasParsedSubCommands = true; + } else { + subCommands.push(argv[current]); + continue; + } + } + + // Detect flag + if (!argv[current].startsWith('-')) continue; + let flag = argv[current].replace(/^-+/, ''); + + // Detect value + const hasNextArgument = next < argv.length && !argv[next].startsWith('-'); + let value: string | number | boolean = hasNextArgument ? argv[next] : 'true'; + + // Split combinations + const isCombination = flag.includes('='); + if (isCombination) [flag, value] = flag.split('='); + + // Parse types + if (['true', 'false'].includes(value)) value = value === 'true'; + else if (!Number.isNaN(Number(value)) && !Number.isNaN(Number.parseInt(value))) value = Number.parseInt(value); + + // Assign + // eslint-disable-next-line no-console + if (verbose) console.log(`Found flag "${flag}" with value "${value}" (${typeof value}).`); + args.set(flag, value); + } + + return { subCommands, args }; +}; diff --git a/src/core/env/env-variables.ts b/src/core/env/env-variables.ts new file mode 100644 index 000000000..2bf7e0b47 --- /dev/null +++ b/src/core/env/env-variables.ts @@ -0,0 +1 @@ +export type EnvVariables = { [index: string]: string }; diff --git a/src/core/env/environment.ts b/src/core/env/environment.ts new file mode 100644 index 000000000..2b2ce111a --- /dev/null +++ b/src/core/env/environment.ts @@ -0,0 +1,40 @@ +type EnvVariables = { [index: string]: string }; + +export class Environment implements EnvVariables { + public readonly os: string; + public readonly arch: string; + + constructor(env: Deno.env, envFile: EnvVariables) { + // Make an immutable copy of the environment variables. + for (const [key, value] of Object.entries(env.toObject())) { + // Todo - check if this ever happens at all + if (value === undefined) { + // eslint-disable-next-line no-console + console.error(`Environment variable ${key} is undefined.`); + } + + this[key] = value; + } + + // Override any env variables that are set in a .env file. + for (const [key, value] of Object.entries(envFile)) { + this[key] = value; + } + + // Override specific variables. + this.os = Deno.build.os; + this.arch = Deno.build.arch; + } + + public get(key: string): string | undefined { + return this[key]; + } + + public getOS(): string { + return this.os; + } + + public getArch(): string { + return this.arch; + } +} diff --git a/src/core/logger/formatter.ts b/src/core/logger/formatter.ts new file mode 100644 index 000000000..6467a974c --- /dev/null +++ b/src/core/logger/formatter.ts @@ -0,0 +1,76 @@ +import { pad } from 'https://deno.land/std@0.36.0/strings/pad.ts'; +import { FormatterFunction } from 'https://deno.land/std@0.151.0/log/handlers.ts'; +import { LogRecord } from 'https://deno.land/std@0.151.0/log/logger.ts'; + +// See: https://github.com/denoland/deno_std/blob/0.151.0/log/README.md#custom-message-format +export const createFormatter = ({ + showTime = false, + showLogger = false, + showLevel = false, + showLevelName = true, + showBrackets = true, + depth = 3, +} = {}): FormatterFunction => { + const column = (input: string, { width = 0, align = 'left' } = {}) => { + const totalWidth = showBrackets ? width + 2 : width; + const value = showBrackets ? `[${input}]` : ` ${input}`; + const paddingOptions = { side: align === 'left' ? 'right' : 'left' }; + + return pad(value, totalWidth, paddingOptions); + }; + + const formatValue = (value) => { + switch (typeof value) { + case 'object': + return Deno.inspect(value, { depth }); + case 'undefined': + return 'undefined'; + case 'string': + return value; + default: + return `${value} (${typeof value})`; + } + }; + + return ({ level, levelName, msg, args, loggerName }: LogRecord) => { + let line = ''; + + if (showLogger) { + line += column(loggerName); + } + + if (showTime) { + const now = new Date(); + const hours = pad(`${now.getHours()}`, 2, { char: '0' }); + const minutes = pad(`${now.getMinutes()}`, 2, { char: '0' }); + const seconds = pad(`${now.getSeconds()}`, 2, { char: '0' }); + const time = [hours, minutes, seconds].join(':'); + line += column(time); + } + + if (showLevelName) { + const shortName = levelName.length <= 5 ? levelName : levelName.slice(0, 4); + line += column(shortName, { width: 5, align: 'right' }); + } + + if (showLevel) { + line += column(level); + } + + if (msg) { + if (line.length > 0) line += ' '; + line += formatValue(msg); + } + + if (args) { + if (line.length > 0) line += ' '; + line += args.map((arg) => formatValue(arg)).join(' '); + } + + return line; + }; +}; + +export const formatter = createFormatter(); +export const consoleFormatter = createFormatter(); +export const fileFormatter = createFormatter({ showTime: true, showLevel: true, showBrackets: false }); diff --git a/src/core/logger/index.ts b/src/core/logger/index.ts new file mode 100644 index 000000000..cd9f4b125 --- /dev/null +++ b/src/core/logger/index.ts @@ -0,0 +1,71 @@ +import * as log from 'https://deno.land/std@0.151.0/log/mod.ts'; +import { fileFormatter, consoleFormatter } from './formatter.ts'; +import { getHomeDir, fsSync as fs } from '../../dependencies.ts'; + +export enum Verbosity { + quiet = -1, + normal = 0, + verbose = 1, + veryVerbose = 2, + maxVerbose = 3, +} + +export const configureLogger = async (verbosity: Verbosity) => { + // Verbosity + const isQuiet = verbosity === Verbosity.quiet; + const isVerbose = verbosity >= Verbosity.verbose; + const isVeryVerbose = verbosity >= Verbosity.veryVerbose; + const isMaxVerbose = verbosity >= Verbosity.maxVerbose; + + // Config folder + const configFolder = `${getHomeDir()}/.game-ci`; + await fs.ensureDir(configFolder); + + // Handlers + let consoleLevel = 'INFO'; + if (isQuiet) consoleLevel = 'ERROR'; + if (isVerbose) consoleLevel = 'DEBUG'; + const consoleHandler = new log.handlers.ConsoleHandler(consoleLevel, { formatter: consoleFormatter }); + const fileHandler = new log.handlers.FileHandler('WARNING', { + filename: `${configFolder}/game-ci.log`, + formatter: fileFormatter, + }); + + // Make sure it saves on Ctrl+C interrupt https://github.com/denoland/deno_std/issues/2193 + Deno.addSignalListener('SIGINT', () => fileHandler.flush()); + + await log.setup({ + handlers: { + consoleHandler, + fileHandler, + }, + + loggers: { + default: { + level: 'DEBUG', + handlers: ['consoleHandler', 'fileHandler'], + }, + }, + }); + + /** + * Allows using `log.debug` and other methods directly from anywhere + * + * Example + * log.debug('something', [{ a: { b: { c: { d: ['a', 'b'] } } } }], 'something', { + * a: { b: { c: { d: { e: { f: { g: 'foo' } } } } } }, + * }); + * + * Outputs: + * [DEBUG] something [ { a: { b: [Object] } } ] something { a: { b: { c: [Object] } } } + */ + window.log = log.getLogger(); + + // Verbosity + window.log.verbosity = verbosity; + window.log.verbosityName = Verbosity[verbosity]; + window.log.isQuiet = isQuiet; + window.log.isVerbose = isVerbose; + window.log.isVeryVerbose = isVeryVerbose; + window.log.isMaxVerbose = isMaxVerbose; +}; diff --git a/src/dependencies.ts b/src/dependencies.ts new file mode 100644 index 000000000..5e5213c05 --- /dev/null +++ b/src/dependencies.ts @@ -0,0 +1,85 @@ +// These are the packages from Deno that replace the ones from Node. +import * as assert from 'https://deno.land/std@0.144.0/testing/asserts.ts'; +import * as aws from 'https://deno.land/x/aws_api/client/mod.ts'; +import * as base64 from 'https://deno.land/std@0.145.0/encoding/base64.ts'; +import * as compress from 'https://deno.land/x/compress@v0.3.3/mod.ts'; +import * as fs from 'https://deno.land/std@0.152.0/node/fs/promises.ts'; +import * as fsSync from 'https://deno.land/std@0.152.0/fs/mod.ts'; +import * as k8s from 'https://deno.land/x/kubernetes_client/mod.ts'; +import * as k8sTypes from 'https://deno.land/x/kubernetes_apis/builtin/core@v1/mod.ts'; +import * as nanoid from 'https://deno.land/x/nanoid@v3.0.0/mod.ts'; +import * as path from 'https://deno.land/std@0.142.0/path/mod.ts'; +import * as process from 'https://deno.land/std@0.104.0/node/process.ts'; +import * as semver from 'https://deno.land/x/semver@v1.4.0/mod.ts'; +import * as yaml from 'https://deno.land/std@0.145.0/encoding/yaml.ts'; +import { crypto } from 'https://deno.land/std@0.142.0/crypto/mod.ts'; +import { v4 as uuid } from 'https://deno.land/std@0.142.0/uuid/mod.ts'; +import * as http from 'https://deno.land/std@0.145.0/node/http.ts'; +import * as string from 'https://deno.land/std@0.36.0/strings/mod.ts'; +import { Command } from 'https://deno.land/x/cmd@v1.2.0/commander/index.ts'; +import { getUnityChangeset as getUnityChangeSet } from 'https://deno.land/x/unity_changeset@2.0.0/src/index.ts'; +import { Buffer } from 'https://deno.land/std@0.151.0/io/buffer.ts'; +import { config, configSync } from 'https://deno.land/std@0.151.0/dotenv/mod.ts'; +import yargs from 'https://deno.land/x/yargs@v17.5.1-deno/deno.ts'; +import type { Arguments as YargsArguments } from 'https://deno.land/x/yargs@v17.5.1-deno/deno-types.ts'; +import { default as getHomeDir } from 'https://deno.land/x/dir@1.5.1/home_dir/mod.ts'; + +// Internally managed packages +import waitUntil from './module/wait-until.ts'; +import { core } from './module/actions/index.ts'; +import { dedent } from './module/dedent.ts'; + +// Polyfill for https://github.com/tc39/proposal-string-dedent +String.dedent = dedent; + +// Errors from yargs can be very verbose and not very descriptive +Error.stackTraceLimit = 15; + +class Writable { + constructor() { + throw new Error('Writable is not implemented'); // stream + } +} + +const __filename = path.fromFileUrl(import.meta.url); +const __dirname = path.dirname(path.fromFileUrl(import.meta.url)); + +const { V1EnvVar, V1EnvVarSource, V1SecretKeySelector } = k8s; + +type YargsInstance = yargs.Argv; + +export type { YargsArguments, YargsInstance }; +export { + __dirname, + __filename, + k8s, + k8sTypes, + V1EnvVar, + V1EnvVarSource, + V1SecretKeySelector, + assert, + aws, + base64, + Buffer, + Command, + compress, + config, + configSync, + core, + crypto, + fs, + fsSync, + getHomeDir, + getUnityChangeSet, + http, + nanoid, + path, + process, + semver, + string, + uuid, + waitUntil, + Writable, + yaml, + yargs, +}; diff --git a/src/deps-test.ts b/src/deps-test.ts new file mode 100644 index 000000000..3f0a26eb7 --- /dev/null +++ b/src/deps-test.ts @@ -0,0 +1 @@ +export { assert, assertFalse, assertEquals, assertThrows } from 'https://deno.land/std@0.153.0/testing/asserts.ts'; diff --git a/src/dev-dependencies.ts b/src/dev-dependencies.ts new file mode 100644 index 000000000..3894a9936 --- /dev/null +++ b/src/dev-dependencies.ts @@ -0,0 +1,6 @@ +import * as asserts from 'https://deno.land/std@0.142.0/testing/mod.ts'; + +// Dev dependencies also include all prod dependencies. +export * from './dependencies.ts'; + +export { asserts }; diff --git a/src/foo.ts b/src/foo.ts new file mode 100644 index 000000000..6ebf13c11 --- /dev/null +++ b/src/foo.ts @@ -0,0 +1,5 @@ +export class Foo { + public static bar() { + return 'bar'; + } +} diff --git a/src/global.d.ts b/src/global.d.ts new file mode 100644 index 000000000..8d2500a59 --- /dev/null +++ b/src/global.d.ts @@ -0,0 +1,27 @@ +import { Verbosity } from './core/logger/index.ts'; + +declare global { + interface String { + dedent(indentedString: string): string; + } + + let log: { + verbosity: Verbosity; + isQuiet: boolean; + isVerbose: boolean; + isVeryVerbose: boolean; + isMaxVerbose: boolean; + debug: (msg: any, ...args: any[]) => void; + info: (msg: any, ...args: any[]) => void; + warning: (msg: any, ...args: any[]) => void; + error: (msg: any, ...args: any[]) => void; + }; +} + +declare interface String { + dedent(indentedString: string): string; +} + +declare interface Window { + log: any; +} diff --git a/src/index.ts b/src/index.ts index 1582d7c03..dfcac95e2 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,39 +1,22 @@ -import * as core from '@actions/core'; -import { Action, BuildParameters, Cache, CloudRunner, Docker, ImageTag, Output } from './model'; -import { Cli } from './model/cli/cli'; -import MacBuilder from './model/mac-builder'; -import PlatformSetup from './model/platform-setup'; -async function runMain() { - try { - if (Cli.InitCliMode()) { - await Cli.RunCli(); +import { Cli } from './cli.ts'; - return; - } - Action.checkCompatibility(); - Cache.verify(); - - const { workspace, actionFolder } = Action; +class GameCI { + public static async run() { + try { + const { command, options } = await new Cli().validateAndParseArguments(); - const buildParameters = await BuildParameters.create(); - const baseImage = new ImageTag(buildParameters); + const success = await command.execute(options); - if (buildParameters.cloudRunnerCluster !== 'local') { - await CloudRunner.run(buildParameters, baseImage.toString()); - } else { - core.info('Building locally'); - await PlatformSetup.setup(buildParameters, actionFolder); - if (process.platform === 'darwin') { - MacBuilder.run(actionFolder, workspace, buildParameters); + if (success) { + log.info(`${command.name} done.`); } else { - await Docker.run(baseImage, { workspace, actionFolder, ...buildParameters }); + log.warning(`${command.constructor.name} failed.`); } + } catch (error) { + log.error(error); + Deno.exit(1); } - - // Set output - await Output.setBuildVersion(buildParameters.buildVersion); - } catch (error) { - core.setFailed((error as Error).message); } } -runMain(); + +await GameCI.run(); diff --git a/src/integrity.test.ts b/src/integrity.test.ts index 7763709ca..c523b5851 100644 --- a/src/integrity.test.ts +++ b/src/integrity.test.ts @@ -1,9 +1,25 @@ -import { stat } from 'fs/promises'; - -describe('Integrity tests', () => { - describe('package-lock.json', () => { - it('does not exist', async () => { - await expect(stat(`${process.cwd()}/package-lock.json`)).rejects.toThrowError(); - }); - }); +import { assert, assertFalse, assertEquals, assertThrows } from './deps-test.ts'; +import { existsSync } from 'https://deno.land/std/fs/mod.ts'; +import { Foo } from './foo.ts'; + +// Todo - Remove this test after we have some coverage +assertEquals(Foo.bar(), 'bar'); + +// Todo - Enable this test, once node_modules are completely phased out. +// Deno.test('package.json does not exist', async () => { +// assertFalse(existsSync(`${Deno.cwd()}/package.json`)); +// }); + +Deno.test('package-lock.json does not exist', async () => { + assertFalse(existsSync(`${Deno.cwd()}/package-lock.json`)); }); + +// Todo - Enable this test, once node_modules are completely phased out. +// Deno.test('yarn.lock does not exist', async () => { +// assertFalse(existsSync(`${Deno.cwd()}/yarn.lock`)); +// }); + +// Todo - Enable this test, once node_modules are completely phased out. +// Deno.test('node_modules does not exist', async () => { +// assertFalse(existsSync(`${Deno.cwd()}/node_modules`)); +// }); diff --git a/src/logic/unity/platform-setup/index.ts b/src/logic/unity/platform-setup/index.ts new file mode 100644 index 000000000..87862a214 --- /dev/null +++ b/src/logic/unity/platform-setup/index.ts @@ -0,0 +1,4 @@ +import SetupWindows from './setup-windows.ts'; +import SetupMac from './setup-mac.ts'; + +export { SetupWindows, SetupMac }; diff --git a/src/logic/unity/platform-setup/setup-mac.ts b/src/logic/unity/platform-setup/setup-mac.ts new file mode 100644 index 000000000..68cab0360 --- /dev/null +++ b/src/logic/unity/platform-setup/setup-mac.ts @@ -0,0 +1,72 @@ +import { Parameters } from '../../../model/index.ts'; +import { fsSync as fs, getUnityChangeSet } from '../../../dependencies.ts'; +import System from '../../../model/system/system.ts'; + +class SetupMac { + static unityHubPath = `"/Applications/Unity Hub.app/Contents/MacOS/Unity Hub"`; + + public static async setup(buildParameters: Parameters, actionFolder: string) { + const unityEditorPath = `/Applications/Unity/Hub/Editor/${buildParameters.editorVersion}/Unity.app/Contents/MacOS/Unity`; + + // Only install unity if the editor doesn't already exist + if (!fs.existsSync(unityEditorPath)) { + await SetupMac.installUnityHub(); + await SetupMac.installUnity(buildParameters); + } + + await SetupMac.setEnvironmentVariables(buildParameters, actionFolder); + } + + private static async installUnityHub(silent = false) { + const command = 'brew install unity-hub'; + if (!fs.existsSync(this.unityHubPath)) { + try { + await System.run(command, { silent, ignoreReturnCode: true }); + } catch (error) { + throw new Error(`There was an error installing the Unity Editor. See logs above for details. ${error}`); + } + } + } + + private static async installUnity(buildParameters: Parameters, silent = false) { + const unityChangeSet = await getUnityChangeSet(buildParameters.editorVersion); + const command = `${this.unityHubPath} -- --headless install \ + --version ${buildParameters.editorVersion} \ + --changeset ${unityChangeSet.changeset} \ + --module mac-il2cpp \ + --childModules`; + + try { + await System.run(command, { silent, ignoreReturnCode: true }); + } catch (error) { + throw new Error(`There was an error installing the Unity Editor. See logs above for details. ${error}`); + } + } + + private static async setEnvironmentVariables(parameters: Parameters, actionFolder: string) { + // Need to set environment variables from here because we execute + // the scripts on the host for mac + Deno.env.set('ACTION_FOLDER', actionFolder); + Deno.env.set('UNITY_VERSION', parameters.editorVersion); + Deno.env.set('UNITY_SERIAL', parameters.unitySerial); + Deno.env.set('PROJECT_PATH', parameters.projectPath); + Deno.env.set('BUILD_TARGET', parameters.targetPlatform); + Deno.env.set('BUILD_NAME', parameters.buildName); + Deno.env.set('BUILD_PATH', parameters.buildPath); + Deno.env.set('BUILD_FILE', parameters.buildFile); + Deno.env.set('BUILD_METHOD', parameters.buildMethod); + Deno.env.set('VERSION', parameters.buildVersion); + Deno.env.set('ANDROID_VERSION_CODE', parameters.androidVersionCode); + Deno.env.set('ANDROID_KEYSTORE_NAME', parameters.androidKeystoreName); + Deno.env.set('ANDROID_KEYSTORE_BASE64', parameters.androidKeystoreBase64); + Deno.env.set('ANDROID_KEYSTORE_PASS', parameters.androidKeystorePass); + Deno.env.set('ANDROID_KEYALIAS_NAME', parameters.androidKeyaliasName); + Deno.env.set('ANDROID_KEYALIAS_PASS', parameters.androidKeyaliasPass); + Deno.env.set('ANDROID_TARGET_SDK_VERSION', parameters.androidTargetSdkVersion); + Deno.env.set('ANDROID_SDK_MANAGER_PARAMETERS', parameters.androidSdkManagerParameters); + Deno.env.set('CUSTOM_PARAMETERS', parameters.customParameters); + Deno.env.set('CHOWN_FILES_TO', parameters.chownFilesTo); + } +} + +export default SetupMac; diff --git a/src/logic/unity/platform-setup/setup-windows.ts b/src/logic/unity/platform-setup/setup-windows.ts new file mode 100644 index 000000000..1d5144752 --- /dev/null +++ b/src/logic/unity/platform-setup/setup-windows.ts @@ -0,0 +1,25 @@ +import { fsSync as fs } from '../../../dependencies.ts'; +import { Parameters } from '../../../model/index.ts'; +import ValidateWindows from '../platform-validation/validate-windows.ts'; +import System from '../../../model/system/system.ts'; + +class SetupWindows { + public static async setup(parameters: Parameters) { + ValidateWindows.validate(parameters); + await this.generateWinSdkRegistryKey(parameters); + } + + private static async generateWinSdkRegistryKey(parameters) { + const { targetPlatform, cliStoragePath } = parameters; + + if (!['StandaloneWindows', 'StandaloneWindows64', 'WSAPlayer'].includes(targetPlatform)) return; + + const registryKeysPath = `${cliStoragePath}/registry-keys`; + const copyWinSdkRegistryKeyCommand = `reg export "HKLM\\SOFTWARE\\WOW6432Node\\Microsoft\\Microsoft SDKs\\Windows\\v10.0" ${registryKeysPath}/winsdk.reg /y`; + + await fs.ensureDir(registryKeysPath); + await System.run(copyWinSdkRegistryKeyCommand); + } +} + +export default SetupWindows; diff --git a/src/model/platform-validation/validate-windows.ts b/src/logic/unity/platform-validation/validate-windows.ts similarity index 53% rename from src/model/platform-validation/validate-windows.ts rename to src/logic/unity/platform-validation/validate-windows.ts index 0a2c56bac..f1651b554 100644 --- a/src/model/platform-validation/validate-windows.ts +++ b/src/logic/unity/platform-validation/validate-windows.ts @@ -1,13 +1,15 @@ -import fs from 'fs'; -import { BuildParameters } from '..'; +import { fsSync as fs } from '../../../dependencies.ts'; +import { Parameters } from '../../../model/index.ts'; class ValidateWindows { - public static validate(buildParameters: BuildParameters) { - ValidateWindows.validateWindowsPlatformRequirements(buildParameters.targetPlatform); - if (!(process.env.UNITY_EMAIL && process.env.UNITY_PASSWORD)) { - throw new Error(`Unity email and password must be set for Windows based builds to - authenticate the license. Make sure to set them inside UNITY_EMAIL - and UNITY_PASSWORD in Github Secrets and pass them into the environment.`); + public static validate(parameters: Parameters) { + ValidateWindows.validateWindowsPlatformRequirements(parameters.targetPlatform); + if (!parameters.unityEmail || !parameters.unityPassword) { + throw new Error(String.dedent` + Unity email and password must be set for Windows based builds to authenticate the license. + + Please make sure to set the unityEmail (UNITY_EMAIL) and unityPassword (UNITY_PASSWORD) parameters. + `); } } @@ -35,9 +37,11 @@ class ValidateWindows { // Check for Windows 10 SDK on runner const windows10SDKPathExists = fs.existsSync('C:/Program Files (x86)/Windows Kits'); if (!windows10SDKPathExists) { - throw new Error(`Windows 10 SDK not found in default location. Make sure - the runner has a Windows 10 SDK installed in the default - location.`); + throw new Error(String.dedent` + Windows 10 SDK not found in default location. Please make sure this machine has a Windows 10 SDK installed. + + Download here: https://developer.microsoft.com/en-us/windows/downloads/windows-sdk/ + `); } } @@ -47,9 +51,13 @@ class ValidateWindows { const visualStudioDataPathExists = fs.existsSync('C:/ProgramData/Microsoft/VisualStudio'); if (!visualStudioInstallPathExists || !visualStudioDataPathExists) { - throw new Error(`Visual Studio not found at the default location. - Make sure the runner has Visual Studio installed in the - default location`); + throw new Error(String.dedent` + Visual Studio not found at the default location. + + Please make sure the runner has Visual Studio installed in the default location + + Download here: https://visualstudio.microsoft.com/downloads/ + `); } } } diff --git a/src/middleware/build-versioning/android-build-version-generator.ts b/src/middleware/build-versioning/android-build-version-generator.ts new file mode 100644 index 000000000..5831dc105 --- /dev/null +++ b/src/middleware/build-versioning/android-build-version-generator.ts @@ -0,0 +1,32 @@ +import { semver } from '../../dependencies.ts'; + +export default class AndroidBuildVersionGenerator { + public static determineVersionCode(version) { + if (version === 'none') { + log.info(`Versioning strategy is set to ${version}, so android version code should not be applied.`); + + return 0; + } + + const parsedVersion = semver.parse(version); + + if (!parsedVersion) { + log.warning(`Could not parse "${version}" to semver, defaulting android version code to 1`); + + return 1; + } + + // The greatest value Google Plays allows is 2100000000. + // Allow for 3 patch digits, 3 minor digits and 3 major digits. + const versionCode = parsedVersion.major * 1_000_000 + parsedVersion.minor * 1000 + parsedVersion.patch; + + if (versionCode >= 2_050_000_000) { + throw new Error( + `Generated versionCode ${versionCode} is dangerously close to the maximum allowed number 2100000000. Consider a different versioning scheme to be able to continue updating your application.`, + ); + } + log.info(`Using android versionCode ${versionCode}`); + + return versionCode; + } +} diff --git a/src/model/versioning.ts b/src/middleware/build-versioning/build-version-generator.ts similarity index 54% rename from src/model/versioning.ts rename to src/middleware/build-versioning/build-version-generator.ts index 7ef0ca47d..002e93c47 100644 --- a/src/model/versioning.ts +++ b/src/middleware/build-versioning/build-version-generator.ts @@ -1,113 +1,85 @@ -import * as core from '@actions/core'; -import NotImplementedException from './error/not-implemented-exception'; -import ValidationError from './error/validation-error'; -import Input from './input'; -import System from './system'; - -export default class Versioning { - static get projectPath() { - return Input.projectPath; +import NotImplementedException from '../../model/error/not-implemented-exception.ts'; +import Input from '../../model/input.ts'; +import System from '../../model/system/system.ts'; +import { Action } from '../../model/index.ts'; +import { VersioningStrategy } from '../../model/versioning/versioning-strategy.ts'; + +export default class BuildVersionGenerator { + private readonly maxDiffLines: number = 60; + private readonly projectPath: string; + + constructor(projectPath, currentBranch) { + this.projectPath = projectPath; + this.currentBranch = currentBranch; } - static get isDirtyAllowed() { - return Input.allowDirtyBuild; - } + // Todo - move more of the get detection logic to the vcs detection class + public async determineBuildVersion(strategy: string, inputVersion: string, allowDirtyBuild: boolean) { + log.info('Versioning strategy:', strategy); - static get strategies() { - return { None: 'None', Semantic: 'Semantic', Tag: 'Tag', Custom: 'Custom' }; - } + let version; + switch (strategy) { + case VersioningStrategy.None: + version = 'none'; + break; + case VersioningStrategy.Custom: + version = inputVersion; + break; + case VersioningStrategy.Semantic: + version = await this.generateSemanticVersion(allowDirtyBuild); + break; + case VersioningStrategy.Tag: + version = await this.generateTagVersion(); + break; + default: + throw new NotImplementedException(`Strategy ${strategy} is not implemented.`); + } - static get grepCompatibleInputVersionRegex() { - return '^v?([0-9]+\\.)*[0-9]+.*'; - } + log.info('Version of this build:', version); - /** - * Get the branch name of the (related) branch - */ - static get branch() { - // Todo - use optional chaining (https://github.com/zeit/ncc/issues/534) - return this.headRef || (this.ref && this.ref.slice(11)); + return version; } - /** - * For pull requests we can reliably use GITHUB_HEAD_REF - */ - static get headRef() { - return process.env.GITHUB_HEAD_REF; + private get grepCompatibleInputVersionRegex() { + return '^v?([0-9]+\\.)*[0-9]+.*'; } /** - * For branches GITHUB_REF will have format `refs/heads/feature-branch-1` + * Get the branch name of the (related) branch */ - static get ref() { - return process.env.GITHUB_REF; - } + private async getCurrentBranch() {} /** * The commit SHA that triggered the workflow run. + * @deprecated */ - static get sha() { - return process.env.GITHUB_SHA; - } - - /** - * Maximum number of lines to print when logging the git diff - */ - static get maxDiffLines() { - return 60; - } - - /** - * Log up to maxDiffLines of the git diff. - */ - static async logDiff() { - const diffCommand = `git --no-pager diff | head -n ${this.maxDiffLines.toString()}`; - await System.run('sh', undefined, { - input: Buffer.from(diffCommand), - silent: true, - }); + private get sha() { + return Deno.env.get('GITHUB_SHA'); } /** * Regex to parse version description into separate fields */ - static get descriptionRegex1() { + private get descriptionRegex1() { return /^v?([\d.]+)-(\d+)-g(\w+)-?(\w+)*/g; } - static get descriptionRegex2() { + private get descriptionRegex2() { return /^v?([\d.]+-\w+)-(\d+)-g(\w+)-?(\w+)*/g; } - static get descriptionRegex3() { + private get descriptionRegex3() { return /^v?([\d.]+-\w+\.\d+)-(\d+)-g(\w+)-?(\w+)*/g; } - static async determineBuildVersion(strategy: string, inputVersion: string) { - // Validate input - if (!Object.hasOwnProperty.call(this.strategies, strategy)) { - throw new ValidationError(`Versioning strategy should be one of ${Object.values(this.strategies).join(', ')}.`); - } - - let version; - switch (strategy) { - case this.strategies.None: - version = 'none'; - break; - case this.strategies.Custom: - version = inputVersion; - break; - case this.strategies.Semantic: - version = await this.generateSemanticVersion(); - break; - case this.strategies.Tag: - version = await this.generateTagVersion(); - break; - default: - throw new NotImplementedException(`Strategy ${strategy} is not implemented.`); - } + /** + * Log up to maxDiffLines of the git diff. + */ + static async logDiff() { + const diffCommand = `git --no-pager diff | head -n ${this.maxDiffLines.toString()}`; + const result = await System.shellRun(diffCommand); - return version; + log.debug(result.output); } /** @@ -120,20 +92,19 @@ export default class Versioning { * * @See: https://semver.org/ */ - static async generateSemanticVersion() { + private async generateSemanticVersion(allowDirtyBuild) { if (await this.isShallow()) { await this.fetch(); } - await this.logDiff(); - - if ((await this.isDirty()) && !this.isDirtyAllowed) { + if ((await this.isDirty()) && !allowDirtyBuild) { + await BuildVersionGenerator.logDiff(); throw new Error('Branch is dirty. Refusing to base semantic version on uncommitted changes'); } if (!(await this.hasAnyVersionTags())) { const version = `0.0.${await this.getTotalNumberOfCommits()}`; - core.info(`Generated version ${version} (no version tags found).`); + log.info(`Generated version ${version} (no version tags found).`); return version; } @@ -146,13 +117,13 @@ export default class Versioning { const [major, minor, patch] = `${tag}.${commits}`.split('.'); const threeDigitVersion = /^\d+$/.test(patch) ? `${major}.${minor}.${patch}` : `${major}.0.${minor}`; - core.info(`Found semantic version ${threeDigitVersion} for ${this.branch}@${hash}`); + log.info(`Found semantic version ${threeDigitVersion} for ${this.currentBranch}@${hash}`); return `${threeDigitVersion}`; } const version = `0.0.${await this.getTotalNumberOfCommits()}`; - core.info(`Generated version ${version} (semantic version couldn't be determined).`); + log.info(`Generated version ${version} (semantic version couldn't be determined).`); return version; } @@ -160,7 +131,7 @@ export default class Versioning { /** * Generate the proper version for unity based on an existing tag. */ - static async generateTagVersion() { + private async generateTagVersion() { let tag = await this.getTag(); if (tag.charAt(0) === 'v') { @@ -173,7 +144,7 @@ export default class Versioning { /** * Parses the versionDescription into their named parts. */ - static async parseSemanticVersion() { + private async parseSemanticVersion() { const description = await this.getVersionDescription(); try { @@ -206,9 +177,7 @@ export default class Versioning { hash, }; } catch { - core.warning( - `Failed to parse git describe output or version can not be determined through: "${description}".`, - ); + log.warning(`Failed to parse git describe output or version can not be determined through "${description}".`); return false; } @@ -219,10 +188,10 @@ export default class Versioning { /** * Returns whether the repository is shallow. */ - static async isShallow() { - const output = await this.git(['rev-parse', '--is-shallow-repository']); + private async isShallow() { + const output = await this.git('rev-parse --is-shallow-repository'); - return output !== 'false\n'; + return output !== 'false'; } /** @@ -232,12 +201,12 @@ export default class Versioning { * * Note: `--all` should not be used, and would break fetching for push event. */ - static async fetch() { + private async fetch() { try { - await this.git(['fetch', '--unshallow']); - } catch (error) { - core.warning(`Fetch --unshallow caught: ${error}`); - await this.git(['fetch']); + await this.git('fetch --unshallow'); + } catch { + log.warning(`fetch --unshallow did not work, falling back to regular fetch`); + await this.git('fetch'); } } @@ -249,20 +218,31 @@ export default class Versioning { * In this format v0.12 is the latest tag, 24 are the number of commits since, and gd2198ab * identifies the current commit. */ - static async getVersionDescription() { - return this.git(['describe', '--long', '--tags', '--always', this.sha]); + private async getVersionDescription() { + let commitIsh = ''; + + // In CI the repo is checked out in detached head mode. + // We MUST specify the commitIsh that triggered the job. + // Todo - make this compatible with more CI systems + if (!Action.isRunningLocally) { + commitIsh = this.sha; + } + + return this.git(`describe --long --tags --always ${commitIsh}`); } /** * Returns whether there are uncommitted changes that are not ignored. */ - static async isDirty() { - const output = await this.git(['status', '--porcelain']); + private async isDirty() { + const output = await this.git('status --porcelain'); const isDirty = output !== ''; if (isDirty) { - core.warning('Changes were made to the following files and folders:\n'); - core.warning(output); + log.warning( + `Changes were made to the following files and folders:\n\n A = addition, M = modification, D = deletion\n\n`, + output, + ); } return isDirty; @@ -271,8 +251,8 @@ export default class Versioning { /** * Get the tag if there is one pointing at HEAD */ - static async getTag() { - return (await this.git(['tag', '--points-at', 'HEAD'])).trim(); + private async getTag() { + return await this.git('tag --points-at HEAD'); } /** @@ -280,15 +260,19 @@ export default class Versioning { * * Note: Currently this is run in all OSes, so the syntax must be cross-platform. */ - static async hasAnyVersionTags() { - const numberOfTagsAsString = await System.run('sh', undefined, { - input: Buffer.from(`git tag --list --merged HEAD | grep -E '${this.grepCompatibleInputVersionRegex}' | wc -l`), - cwd: this.projectPath, - silent: false, - }); + private async hasAnyVersionTags() { + const command = `git tag --list --merged HEAD | grep -E '${this.grepCompatibleInputVersionRegex}' | wc -l`; + + // Todo - make sure this cwd is actually passed in somehow + const result = await System.shellRun(command, { cwd: this.projectPath, silent: false }); + log.debug(result); + + const { output: numberOfTagsAsString } = result; const numberOfTags = Number.parseInt(numberOfTagsAsString, 10); + log.debug('numberOfTags', numberOfTags); + return numberOfTags !== 0; } @@ -297,8 +281,8 @@ export default class Versioning { * * Note: HEAD should not be used, as it may be detached, resulting in an additional count. */ - static async getTotalNumberOfCommits() { - const numberOfCommitsAsString = await this.git(['rev-list', '--count', this.sha]); + private async getTotalNumberOfCommits() { + const numberOfCommitsAsString = await this.git(`rev-list --count ${this.sha}`); return Number.parseInt(numberOfCommitsAsString, 10); } @@ -306,7 +290,11 @@ export default class Versioning { /** * Run git in the specified project path */ - static async git(arguments_, options = {}) { - return System.run('git', arguments_, { cwd: this.projectPath, ...options }); + private async git(arguments_, options = {}) { + const result = await System.run(`git ${arguments_}`, { cwd: this.projectPath, ...options }); + + log.warning(result); + + return result.output; } } diff --git a/src/middleware/build-versioning/index.ts b/src/middleware/build-versioning/index.ts new file mode 100644 index 000000000..b49c7e392 --- /dev/null +++ b/src/middleware/build-versioning/index.ts @@ -0,0 +1,14 @@ +import BuildVersionGenerator from './build-version-generator.ts'; +import AndroidBuildVersionGenerator from './android-build-version-generator.ts'; + +export const buildVersioning = async (argv) => { + const { projectPath, versioningStrategy, version, allowDirtyBuild, androidVersionCode, buildVersion } = argv; + + const buildVersionGenerator = new BuildVersionGenerator(projectPath); + + argv.buildVersion = await buildVersionGenerator.determineBuildVersion(versioningStrategy, version, allowDirtyBuild); + + if (!androidVersionCode) { + argv.androidVersionCode = AndroidBuildVersionGenerator.determineVersionCode(buildVersion); + } +}; diff --git a/src/middleware/engine-detection/engine-detector.ts b/src/middleware/engine-detection/engine-detector.ts new file mode 100644 index 000000000..da8222326 --- /dev/null +++ b/src/middleware/engine-detection/engine-detector.ts @@ -0,0 +1,22 @@ +import UnityVersionDetector from './unity-version-detector.ts'; + +export class EngineDetector { + private readonly projectPath: string; + + constructor(projectPath) { + this.projectPath = projectPath; + } + + public async detect(): Promise<{ engine: string; engineVersion: string }> { + if (UnityVersionDetector.isUnityProject(this.projectPath)) { + const engineVersion = await UnityVersionDetector.getUnityVersion(this.projectPath); + + return { engine: 'unity', engineVersion }; + } + + return { + engine: 'unknown', + engineVersion: 'unknown', + }; + } +} diff --git a/src/middleware/engine-detection/index.ts b/src/middleware/engine-detection/index.ts new file mode 100644 index 000000000..8384cf8b4 --- /dev/null +++ b/src/middleware/engine-detection/index.ts @@ -0,0 +1,12 @@ +import { EngineDetector } from './engine-detector.ts'; + +export const engineDetection = async (argv) => { + const { projectPath } = argv; + + if (!projectPath) throw new Error('Unable to detect engine. No project path provided.'); + + const { engine, engineVersion } = await new EngineDetector(projectPath).detect(); + + argv.engine = engine; + argv.engineVersion = engineVersion; +}; diff --git a/src/model/unity-versioning.test.ts b/src/middleware/engine-detection/unity-version-detector.test.ts similarity index 51% rename from src/model/unity-versioning.test.ts rename to src/middleware/engine-detection/unity-version-detector.test.ts index 94aa3c6a4..b8006df18 100644 --- a/src/model/unity-versioning.test.ts +++ b/src/middleware/engine-detection/unity-version-detector.test.ts @@ -1,35 +1,35 @@ -import UnityVersioning from './unity-versioning'; +import UnityVersionDetector from './unity-version-detector.ts'; describe('Unity Versioning', () => { describe('parse', () => { it('throws for empty string', () => { - expect(() => UnityVersioning.parse('')).toThrow(Error); + expect(() => UnityVersionDetector.parse('')).toThrow(Error); }); it('parses from ProjectVersion.txt', () => { const projectVersionContents = `m_EditorVersion: 2019.2.11f1 m_EditorVersionWithRevision: 2019.2.11f1 (5f859a4cfee5)`; - expect(UnityVersioning.parse(projectVersionContents)).toBe('2019.2.11f1'); + expect(UnityVersionDetector.parse(projectVersionContents)).toBe('2019.2.11f1'); }); }); describe('read', () => { it('throws for invalid path', () => { - expect(() => UnityVersioning.read('')).toThrow(Error); + expect(() => UnityVersionDetector.read('')).toThrow(Error); }); it('reads from test-project', () => { - expect(UnityVersioning.read('./test-project')).toBe('2019.2.11f1'); + expect(UnityVersionDetector.read('./test-project')).toBe('2019.2.11f1'); }); }); describe('determineUnityVersion', () => { it('defaults to parsed version', () => { - expect(UnityVersioning.determineUnityVersion('./test-project', 'auto')).toBe('2019.2.11f1'); + expect(UnityVersionDetector.determineUnityVersion('./test-project', 'auto')).toBe('2019.2.11f1'); }); it('use specified unityVersion', () => { - expect(UnityVersioning.determineUnityVersion('./test-project', '1.2.3')).toBe('1.2.3'); + expect(UnityVersionDetector.determineUnityVersion('./test-project', '1.2.3')).toBe('1.2.3'); }); }); }); diff --git a/src/model/unity-versioning.ts b/src/middleware/engine-detection/unity-version-detector.ts similarity index 52% rename from src/model/unity-versioning.ts rename to src/middleware/engine-detection/unity-version-detector.ts index a7ef2e26d..9a5094d77 100644 --- a/src/model/unity-versioning.ts +++ b/src/middleware/engine-detection/unity-version-detector.ts @@ -1,17 +1,22 @@ -import * as fs from 'fs'; -import path from 'path'; +import { fsSync as fs, path } from '../../dependencies.ts'; -export default class UnityVersioning { +export default class UnityVersionDetector { static get versionPattern() { return /20\d{2}\.\d\.\w{3,4}|3/; } - static determineUnityVersion(projectPath, unityVersion) { - if (unityVersion === 'auto') { - return UnityVersioning.read(projectPath); + public static isUnityProject(projectPath) { + try { + UnityVersionDetector.read(projectPath); + + return true; + } catch { + return false; } + } - return unityVersion; + static getUnityVersion(projectPath) { + return UnityVersionDetector.read(projectPath); } static read(projectPath) { @@ -20,11 +25,11 @@ export default class UnityVersioning { throw new Error(`Project settings file not found at "${filePath}". Have you correctly set the projectPath?`); } - return UnityVersioning.parse(fs.readFileSync(filePath, 'utf8')); + return UnityVersionDetector.parse(Deno.readTextFileSync(filePath, 'utf8')); } static parse(projectVersionTxt) { - const matches = projectVersionTxt.match(UnityVersioning.versionPattern); + const matches = projectVersionTxt.match(UnityVersionDetector.versionPattern); if (!matches || matches.length === 0) { throw new Error(`Failed to parse version from "${projectVersionTxt}".`); } diff --git a/src/middleware/logger-verbosity/index.ts b/src/middleware/logger-verbosity/index.ts new file mode 100644 index 000000000..e021b8bc7 --- /dev/null +++ b/src/middleware/logger-verbosity/index.ts @@ -0,0 +1,28 @@ +import { configureLogger as createLoggerAndSetVerbosity } from '../../core/logger/index.ts'; + +export const configureLogger = async (argv) => { + const { quiet, verbose, veryVerbose, maxVerbose } = argv; + + let verbosity; + if (maxVerbose) { + verbosity = 3; + } else if (veryVerbose) { + verbosity = 2; + } else if (verbose) { + verbosity = 1; + } else if (quiet) { + verbosity = -1; + } else { + verbosity = 0; + } + + await createLoggerAndSetVerbosity(verbosity); + + argv.logLevel = log.verbosity; + argv.logLevelName = log.verbosityName; + + argv.quiet = undefined; + argv.verbose = undefined; + argv.veryVerbose = undefined; + argv.maxVerbose = undefined; +}; diff --git a/src/middleware/vcs-detection/git-detector.ts b/src/middleware/vcs-detection/git-detector.ts new file mode 100644 index 000000000..3f3f19b01 --- /dev/null +++ b/src/middleware/vcs-detection/git-detector.ts @@ -0,0 +1,45 @@ +import System from '../../model/system/system.ts'; + +export class GitDetector { + private readonly projectPath: string; + + constructor(projectPath) { + this.projectPath = projectPath; + } + + public async isGitRepository() { + const { status } = await System.shellRun(`git -C '${this.projectPath}' rev-parse 2>/dev/null`); + + return status.code === 0; + } + + public async getCurrentBranch() { + // GitHub pull request, GitHub non pull request + let branchName = this.headRef || this.ref?.slice(11); + + // Local + if (!branchName) { + const { status, output } = await System.shellRun('git branch --show-current', { cwd: this.projectPath }); + if (!status.success) throw new Error('did not expect "git branch --show-current"'); + branchName = output; + } + + return branchName; + } + + /** + * For pull requests we can reliably use GITHUB_HEAD_REF + * @deprecated + */ + private get headRef() { + return Deno.env.get('GITHUB_HEAD_REF'); + } + + /** + * For branches GITHUB_REF will have format `refs/heads/feature-branch-1` + * @deprecated + */ + private get ref() { + return Deno.env.get('GITHUB_REF'); + } +} diff --git a/src/middleware/vcs-detection/index.ts b/src/middleware/vcs-detection/index.ts new file mode 100644 index 000000000..e7aba2b0e --- /dev/null +++ b/src/middleware/vcs-detection/index.ts @@ -0,0 +1,10 @@ +import { GitDetector } from './git-detector.ts'; + +export const vcsDetection = async (argv) => { + const { projectPath } = argv; + + const gitDetector = new GitDetector(projectPath); + if (await gitDetector.isGitRepository()) { + argv.branch = await gitDetector.getCurrentBranch(projectPath); + } +}; diff --git a/src/model/__data__/versions.ts b/src/model/__data__/versions.ts index 56ad4a9c7..e7a0c443e 100644 --- a/src/model/__data__/versions.ts +++ b/src/model/__data__/versions.ts @@ -75,7 +75,7 @@ export const notCompletelyValidSemanticVersions = [ '99999999999999999999999.999999999999999999.99999999999999999----RC-SNAPSHOT.12.09.1--------------------------------..12', ]; -const addVariantsPrependingV = (array: string[]) => array.map((tag) => [tag, `v${tag}`]).flat(); +const addVariantsPrependingV = (array: string[]) => array.flatMap((tag) => [tag, `v${tag}`]); /** * Array of versions that will be detected as version tags. Not all of these are diff --git a/src/model/__mocks__/input.ts b/src/model/__mocks__/input.ts index 72645c0fd..4a478b51b 100644 --- a/src/model/__mocks__/input.ts +++ b/src/model/__mocks__/input.ts @@ -1,11 +1,11 @@ // Import this named export into your test file: -import Platform from '../platform'; +import UnityTargetPlatform from '../unity/unity-target-platform.ts'; export const mockGetFromUser = jest.fn().mockResolvedValue({ editorVersion: '', - targetPlatform: Platform.types.Test, + targetPlatform: UnityTargetPlatform.Test, projectPath: '.', - buildName: Platform.types.Test, + buildName: UnityTargetPlatform.Test, buildsPath: 'build', buildMethod: undefined, buildVersion: '1.3.37', diff --git a/src/model/action.test.ts b/src/model/action.test.ts index 76fef312a..2e40f2eac 100644 --- a/src/model/action.test.ts +++ b/src/model/action.test.ts @@ -1,6 +1,5 @@ -import path from 'path'; -import fs from 'fs'; -import Action from './action'; +import { fs, path } from '../dependencies.ts'; +import Action from './action.ts'; describe('Action', () => { describe('compatibility check', () => { diff --git a/src/model/action.ts b/src/model/action.ts index 28d079638..7b01948a8 100644 --- a/src/model/action.ts +++ b/src/model/action.ts @@ -1,4 +1,4 @@ -import path from 'path'; +import { path, __dirname, __filename } from '../dependencies.ts'; class Action { static get supportedPlatforms() { @@ -6,7 +6,7 @@ class Action { } static get isRunningLocally() { - return process.env.RUNNER_WORKSPACE === undefined; + return Deno.env.get('RUNNER_WORKSPACE') === undefined; } static get isRunningFromSource() { @@ -30,7 +30,9 @@ class Action { } static get workspace() { - return process.env.GITHUB_WORKSPACE; + if (Action.isRunningLocally) return Deno.cwd(); + + return Deno.env.get('GITHUB_WORKSPACE'); } static checkCompatibility() { diff --git a/src/model/android-versioning.test.ts b/src/model/android-versioning.test.ts index 59d3b64a6..d09aac8da 100644 --- a/src/model/android-versioning.test.ts +++ b/src/model/android-versioning.test.ts @@ -1,41 +1,41 @@ -import AndroidVersioning from './android-versioning'; +import AndroidBuildVersionGenerator from '../middleware/build-versioning/android-build-version-generator.ts'; describe('Android Versioning', () => { describe('versionToVersionCode', () => { it('defaults to 0 when versioning strategy is none', () => { - expect(AndroidVersioning.versionToVersionCode('none')).toBe(0); + expect(AndroidBuildVersionGenerator.versionToVersionCode('none')).toBe(0); }); it('defaults to 1 when version is not a valid semver', () => { - expect(AndroidVersioning.versionToVersionCode('abcd')).toBe(1); + expect(AndroidBuildVersionGenerator.versionToVersionCode('abcd')).toBe(1); }); it('returns a number', () => { - expect(AndroidVersioning.versionToVersionCode('123.456.789')).toBe(123456789); + expect(AndroidBuildVersionGenerator.versionToVersionCode('123.456.789')).toBe(123_456_789); }); it('throw when generated version code is too large', () => { - expect(() => AndroidVersioning.versionToVersionCode('2050.0.0')).toThrow(); + expect(() => AndroidBuildVersionGenerator.versionToVersionCode('2050.0.0')).toThrow(); }); }); describe('determineVersionCode', () => { it('defaults to parsed version', () => { - expect(AndroidVersioning.determineVersionCode('1.2.3', '')).toBe(1002003); + expect(AndroidBuildVersionGenerator.determineVersionCode('1.2.3', '')).toBe(1_002_003); }); it('use specified code', () => { - expect(AndroidVersioning.determineVersionCode('1.2.3', 2)).toBe(2); + expect(AndroidBuildVersionGenerator.determineVersionCode('1.2.3', 2)).toBe(2); }); }); describe('determineSdkManagerParameters', () => { it('defaults to blank', () => { - expect(AndroidVersioning.determineSdkManagerParameters('AndroidApiLevelAuto')).toBe(''); + expect(AndroidBuildVersionGenerator.determineSdkManagerParameters('AndroidApiLevelAuto')).toBe(''); }); it('uses the specified api level', () => { - expect(AndroidVersioning.determineSdkManagerParameters('AndroidApiLevel30')).toBe('platforms;android-30'); + expect(AndroidBuildVersionGenerator.determineSdkManagerParameters('AndroidApiLevel30')).toBe('platforms;android-30'); }); }); }); diff --git a/src/model/android-versioning.ts b/src/model/android-versioning.ts deleted file mode 100644 index 3fb5a5a3c..000000000 --- a/src/model/android-versioning.ts +++ /dev/null @@ -1,47 +0,0 @@ -import * as core from '@actions/core'; -import * as semver from 'semver'; - -export default class AndroidVersioning { - static determineVersionCode(version, inputVersionCode) { - if (!inputVersionCode) { - return AndroidVersioning.versionToVersionCode(version); - } - - return inputVersionCode; - } - - static versionToVersionCode(version) { - if (version === 'none') { - core.info(`Versioning strategy is set to ${version}, so android version code should not be applied.`); - - return 0; - } - - const parsedVersion = semver.parse(version); - - if (!parsedVersion) { - core.warning(`Could not parse "${version}" to semver, defaulting android version code to 1`); - - return 1; - } - - // The greatest value Google Plays allows is 2100000000. - // Allow for 3 patch digits, 3 minor digits and 3 major digits. - const versionCode = parsedVersion.major * 1000000 + parsedVersion.minor * 1000 + parsedVersion.patch; - - if (versionCode >= 2050000000) { - throw new Error( - `Generated versionCode ${versionCode} is dangerously close to the maximum allowed number 2100000000. Consider a different versioning scheme to be able to continue updating your application.`, - ); - } - core.info(`Using android versionCode ${versionCode}`); - - return versionCode; - } - - static determineSdkManagerParameters(targetSdkVersion) { - const parsedVersion = Number.parseInt(targetSdkVersion.slice(-2), 10); - - return Number.isNaN(parsedVersion) ? '' : `platforms;android-${parsedVersion}`; - } -} diff --git a/src/model/build-parameters.test.ts b/src/model/build-parameters.test.ts index 1cb37ae54..238c9e361 100644 --- a/src/model/build-parameters.test.ts +++ b/src/model/build-parameters.test.ts @@ -1,21 +1,21 @@ -import Versioning from './versioning'; -import UnityVersioning from './unity-versioning'; -import AndroidVersioning from './android-versioning'; -import BuildParameters from './build-parameters'; -import Input from './input'; -import Platform from './platform'; +import BuildVersionGenerator from '../middleware/build-versioning/build-version-generator.ts'; +import UnityVersionDetector from '../middleware/engine-detection/unity-version-detector.ts'; +import AndroidBuildVersionGenerator from '../middleware/build-versioning/android-build-version-generator.ts'; +import Parameters from './parameters.ts'; +import Input from './input.ts'; +import UnityTargetPlatform from './unity/unity-target-platform.ts'; // Todo - Don't use process.env directly, that's what the input model class is for. const testLicense = '\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \nm0Db8UK+ktnOLJBtHybkfetpcKo=o/pUbSQAukz7+ZYAWhnA0AJbIlyyCPL7bKVEM2lVqbrXt7cyey+umkCXamuOgsWPVUKBMkXtMH8L\n5etLmD0getWIhTGhzOnDCk+gtIPfL4jMo9tkEuOCROQAXCci23VFscKcrkB+3X6h4wEOtA2APhOY\nB+wvC794o8/82ffjP79aVAi57rp3Wmzx+9pe9yMwoJuljAy2sc2tIMgdQGWVmOGBpQm3JqsidyzI\nJWG2kjnc7pDXK9pwYzXoKiqUqqrut90d+kQqRyv7MSZXR50HFqD/LI69h68b7P8Bjo3bPXOhNXGR\n9YCoemH6EkfCJxp2gIjzjWW+l2Hj2EsFQi8YXw=='; -process.env.UNITY_LICENSE = testLicense; +Deno.env.set('UNITY_LICENSE', testLicense); -const determineVersion = jest.spyOn(Versioning, 'determineBuildVersion').mockImplementation(async () => '1.3.37'); +const determineVersion = jest.spyOn(BuildVersionGenerator, 'determineBuildVersion').mockImplementation(async () => '1.3.37'); const determineUnityVersion = jest - .spyOn(UnityVersioning, 'determineUnityVersion') + .spyOn(UnityVersionDetector, 'determineUnityVersion') .mockImplementation(() => '2019.2.11f1'); const determineSdkManagerParameters = jest - .spyOn(AndroidVersioning, 'determineSdkManagerParameters') + .spyOn(AndroidBuildVersionGenerator, 'determineSdkManagerParameters') .mockImplementation(() => 'platforms;android-30'); afterEach(() => { @@ -25,52 +25,52 @@ afterEach(() => { describe('BuildParameters', () => { describe('create', () => { it('does not throw', async () => { - await expect(BuildParameters.create()).resolves.not.toThrow(); + await expect(Parameters.create()).resolves.not.toThrow(); }); it('determines the version only once', async () => { - await BuildParameters.create(); + await Parameters.create(); expect(determineVersion).toHaveBeenCalledTimes(1); }); it('determines the unity version only once', async () => { - await BuildParameters.create(); + await Parameters.create(); expect(determineUnityVersion).toHaveBeenCalledTimes(1); }); it('returns the android version code with provided input', async () => { const mockValue = '42'; jest.spyOn(Input, 'androidVersionCode', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidVersionCode: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidVersionCode: mockValue })); }); it('returns the android version code from version by default', async () => { const mockValue = ''; jest.spyOn(Input, 'androidVersionCode', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidVersionCode: 1003037 })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidVersionCode: 1_003_037 })); }); it('determines the android sdk manager parameters only once', async () => { - await BuildParameters.create(); + await Parameters.create(); expect(determineSdkManagerParameters).toHaveBeenCalledTimes(1); }); it('returns the targetPlatform', async () => { const mockValue = 'somePlatform'; jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ targetPlatform: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ targetPlatform: mockValue })); }); it('returns the project path', async () => { const mockValue = 'path/to/project'; jest.spyOn(Input, 'projectPath', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ projectPath: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ projectPath: mockValue })); }); it('returns the build name', async () => { const mockValue = 'someBuildName'; jest.spyOn(Input, 'buildName', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildName: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildName: mockValue })); }); it('returns the build path', async () => { @@ -79,92 +79,84 @@ describe('BuildParameters', () => { const expectedBuildPath = `${mockPath}/${mockPlatform}`; jest.spyOn(Input, 'buildsPath', 'get').mockReturnValue(mockPath); jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(mockPlatform); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildPath: expectedBuildPath })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildPath: expectedBuildPath })); }); it('returns the build file', async () => { const mockValue = 'someBuildName'; jest.spyOn(Input, 'buildName', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: mockValue })); }); - test.each([Platform.types.StandaloneWindows, Platform.types.StandaloneWindows64])( + test.each([UnityTargetPlatform.StandaloneWindows, UnityTargetPlatform.StandaloneWindows64])( 'appends exe for %s', async (targetPlatform) => { jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); - expect(BuildParameters.create()).resolves.toEqual( - expect.objectContaining({ buildFile: `${targetPlatform}.exe` }), - ); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: `${targetPlatform}.exe` })); }, ); - test.each([Platform.types.Android])('appends apk for %s', async (targetPlatform) => { + test.each([UnityTargetPlatform.Android])('appends apk for %s', async (targetPlatform) => { jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); jest.spyOn(Input, 'androidAppBundle', 'get').mockReturnValue(false); - expect(BuildParameters.create()).resolves.toEqual( - expect.objectContaining({ buildFile: `${targetPlatform}.apk` }), - ); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: `${targetPlatform}.apk` })); }); - test.each([Platform.types.Android])('appends aab for %s', async (targetPlatform) => { + test.each([UnityTargetPlatform.Android])('appends aab for %s', async (targetPlatform) => { jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); jest.spyOn(Input, 'androidAppBundle', 'get').mockReturnValue(true); - expect(BuildParameters.create()).resolves.toEqual( - expect.objectContaining({ buildFile: `${targetPlatform}.aab` }), - ); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: `${targetPlatform}.aab` })); }); it('returns the build method', async () => { const mockValue = 'Namespace.ClassName.BuildMethod'; jest.spyOn(Input, 'buildMethod', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildMethod: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ buildMethod: mockValue })); }); it('returns the android keystore name', async () => { const mockValue = 'keystore.keystore'; jest.spyOn(Input, 'androidKeystoreName', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystoreName: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystoreName: mockValue })); }); it('returns the android keystore base64-encoded content', async () => { const mockValue = 'secret'; jest.spyOn(Input, 'androidKeystoreBase64', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystoreBase64: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystoreBase64: mockValue })); }); it('returns the android keystore pass', async () => { const mockValue = 'secret'; jest.spyOn(Input, 'androidKeystorePass', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystorePass: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystorePass: mockValue })); }); it('returns the android keyalias name', async () => { const mockValue = 'secret'; jest.spyOn(Input, 'androidKeyaliasName', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeyaliasName: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidKeyaliasName: mockValue })); }); it('returns the android keyalias pass', async () => { const mockValue = 'secret'; jest.spyOn(Input, 'androidKeyaliasPass', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeyaliasPass: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidKeyaliasPass: mockValue })); }); it('returns the android target sdk version', async () => { const mockValue = 'AndroidApiLevelAuto'; jest.spyOn(Input, 'androidTargetSdkVersion', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual( - expect.objectContaining({ androidTargetSdkVersion: mockValue }), - ); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ androidTargetSdkVersion: mockValue })); }); it('returns the custom parameters', async () => { const mockValue = '-profile SomeProfile -someBoolean -someValue exampleValue'; jest.spyOn(Input, 'customParameters', 'get').mockReturnValue(mockValue); - expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ customParameters: mockValue })); + expect(Parameters.create()).resolves.toEqual(expect.objectContaining({ customParameters: mockValue })); }); }); }); diff --git a/src/model/build-parameters.ts b/src/model/build-parameters.ts deleted file mode 100644 index 4dcb38546..000000000 --- a/src/model/build-parameters.ts +++ /dev/null @@ -1,176 +0,0 @@ -import { customAlphabet } from 'nanoid'; -import AndroidVersioning from './android-versioning'; -import CloudRunnerConstants from './cloud-runner/services/cloud-runner-constants'; -import CloudRunnerBuildGuid from './cloud-runner/services/cloud-runner-guid'; -import Input from './input'; -import Platform from './platform'; -import UnityVersioning from './unity-versioning'; -import Versioning from './versioning'; -import { GitRepoReader } from './input-readers/git-repo'; -import { GithubCliReader } from './input-readers/github-cli'; -import { Cli } from './cli/cli'; - -class BuildParameters { - public editorVersion!: string; - public customImage!: string; - public unitySerial!: string; - public runnerTempPath: string | undefined; - public targetPlatform!: string; - public projectPath!: string; - public buildName!: string; - public buildPath!: string; - public buildFile!: string; - public buildMethod!: string; - public buildVersion!: string; - public androidVersionCode!: string; - public androidKeystoreName!: string; - public androidKeystoreBase64!: string; - public androidKeystorePass!: string; - public androidKeyaliasName!: string; - public androidKeyaliasPass!: string; - public androidTargetSdkVersion!: string; - public androidSdkManagerParameters!: string; - public customParameters!: string; - public sshAgent!: string; - public cloudRunnerCluster!: string; - public awsBaseStackName!: string; - public gitPrivateToken!: string; - public awsStackName!: string; - public kubeConfig!: string; - public cloudRunnerMemory!: string; - public cloudRunnerCpu!: string; - public kubeVolumeSize!: string; - public kubeVolume!: string; - public kubeStorageClass!: string; - public chownFilesTo!: string; - public customJobHooks!: string; - public cachePushOverrideCommand!: string; - public cachePullOverrideCommand!: string; - public readInputFromOverrideList!: string; - public readInputOverrideCommand!: string; - public checkDependencyHealthOverride!: string; - public startDependenciesOverride!: string; - public cacheKey!: string; - - public postBuildSteps!: string; - public preBuildSteps!: string; - public customJob!: string; - public runNumber!: string; - public branch!: string; - public githubRepo!: string; - public gitSha!: string; - public logId!: string; - public buildGuid!: string; - public cloudRunnerBranch!: string; - public cloudRunnerIntegrationTests!: boolean; - public cloudRunnerBuilderPlatform!: string | undefined; - public isCliMode!: boolean; - - static async create(): Promise { - const buildFile = this.parseBuildFile(Input.buildName, Input.targetPlatform, Input.androidAppBundle); - const editorVersion = UnityVersioning.determineUnityVersion(Input.projectPath, Input.unityVersion); - const buildVersion = await Versioning.determineBuildVersion(Input.versioningStrategy, Input.specifiedVersion); - const androidVersionCode = AndroidVersioning.determineVersionCode(buildVersion, Input.androidVersionCode); - const androidSdkManagerParameters = AndroidVersioning.determineSdkManagerParameters(Input.androidTargetSdkVersion); - - // Todo - Don't use process.env directly, that's what the input model class is for. - // --- - let unitySerial = ''; - if (!process.env.UNITY_SERIAL && Input.githubInputEnabled) { - // No serial was present, so it is a personal license that we need to convert - if (!process.env.UNITY_LICENSE) { - throw new Error(`Missing Unity License File and no Serial was found. If this - is a personal license, make sure to follow the activation - steps and set the UNITY_LICENSE GitHub secret or enter a Unity - serial number inside the UNITY_SERIAL GitHub secret.`); - } - unitySerial = this.getSerialFromLicenseFile(process.env.UNITY_LICENSE); - } else { - unitySerial = process.env.UNITY_SERIAL!; - } - - return { - editorVersion, - customImage: Input.customImage, - unitySerial, - - runnerTempPath: process.env.RUNNER_TEMP, - targetPlatform: Input.targetPlatform, - projectPath: Input.projectPath, - buildName: Input.buildName, - buildPath: `${Input.buildsPath}/${Input.targetPlatform}`, - buildFile, - buildMethod: Input.buildMethod, - buildVersion, - androidVersionCode, - androidKeystoreName: Input.androidKeystoreName, - androidKeystoreBase64: Input.androidKeystoreBase64, - androidKeystorePass: Input.androidKeystorePass, - androidKeyaliasName: Input.androidKeyaliasName, - androidKeyaliasPass: Input.androidKeyaliasPass, - androidTargetSdkVersion: Input.androidTargetSdkVersion, - androidSdkManagerParameters, - customParameters: Input.customParameters, - sshAgent: Input.sshAgent, - gitPrivateToken: Input.gitPrivateToken || (await GithubCliReader.GetGitHubAuthToken()), - chownFilesTo: Input.chownFilesTo, - cloudRunnerCluster: Input.cloudRunnerCluster, - cloudRunnerBuilderPlatform: Input.cloudRunnerBuilderPlatform, - awsBaseStackName: Input.awsBaseStackName, - kubeConfig: Input.kubeConfig, - cloudRunnerMemory: Input.cloudRunnerMemory, - cloudRunnerCpu: Input.cloudRunnerCpu, - kubeVolumeSize: Input.kubeVolumeSize, - kubeVolume: Input.kubeVolume, - postBuildSteps: Input.postBuildSteps, - preBuildSteps: Input.preBuildSteps, - customJob: Input.customJob, - runNumber: Input.runNumber, - branch: Input.branch.replace('/head', '') || (await GitRepoReader.GetBranch()), - cloudRunnerBranch: Input.cloudRunnerBranch.split('/').reverse()[0], - cloudRunnerIntegrationTests: Input.cloudRunnerTests, - githubRepo: Input.githubRepo || (await GitRepoReader.GetRemote()) || 'game-ci/unity-builder', - isCliMode: Cli.isCliMode, - awsStackName: Input.awsBaseStackName, - gitSha: Input.gitSha, - logId: customAlphabet(CloudRunnerConstants.alphabet, 9)(), - buildGuid: CloudRunnerBuildGuid.generateGuid(Input.runNumber, Input.targetPlatform), - customJobHooks: Input.customJobHooks(), - cachePullOverrideCommand: Input.cachePullOverrideCommand(), - cachePushOverrideCommand: Input.cachePushOverrideCommand(), - readInputOverrideCommand: Input.readInputOverrideCommand(), - readInputFromOverrideList: Input.readInputFromOverrideList(), - kubeStorageClass: Input.kubeStorageClass, - checkDependencyHealthOverride: Input.checkDependencyHealthOverride, - startDependenciesOverride: Input.startDependenciesOverride, - cacheKey: Input.cacheKey, - }; - } - - static parseBuildFile(filename, platform, androidAppBundle) { - if (Platform.isWindows(platform)) { - return `${filename}.exe`; - } - - if (Platform.isAndroid(platform)) { - return androidAppBundle ? `${filename}.aab` : `${filename}.apk`; - } - - return filename; - } - - static getSerialFromLicenseFile(license) { - const startKey = ``; - const startIndex = license.indexOf(startKey) + startKey.length; - if (startIndex < 0) { - throw new Error(`License File was corrupted, unable to locate serial`); - } - const endIndex = license.indexOf(endKey, startIndex); - - // Slice off the first 4 characters as they are garbage values - return Buffer.from(license.slice(startIndex, endIndex), 'base64').toString('binary').slice(4); - } -} - -export default BuildParameters; diff --git a/src/model/cache.test.ts b/src/model/cache.test.ts index 29afe8faf..4b85b72f2 100644 --- a/src/model/cache.test.ts +++ b/src/model/cache.test.ts @@ -1,4 +1,4 @@ -import Cache from './cache'; +import Cache from './cache.ts'; jest.mock('./input'); diff --git a/src/model/cache.ts b/src/model/cache.ts index de963bc88..39877c1ba 100644 --- a/src/model/cache.ts +++ b/src/model/cache.ts @@ -1,11 +1,10 @@ -import * as core from '@actions/core'; -import fs from 'fs'; -import Action from './action'; -import Project from './project'; +import { fsSync, core } from '../dependencies.ts'; +import Action from './action.ts'; +import Project from './project.ts'; class Cache { static verify() { - if (!fs.existsSync(Project.libraryFolder)) { + if (!fsSync.existsSync(Project.libraryFolder)) { this.notifyAboutCachingPossibility(); } } @@ -15,7 +14,7 @@ class Cache { return; } - core.warning(` + log.warning(` Library folder does not exist. Consider setting up caching to speed up your workflow, if this is not your first build. diff --git a/src/model/cli/cli-functions-repository.ts b/src/model/cli/cli-functions-repository.ts index 6cc66f305..26b1aad78 100644 --- a/src/model/cli/cli-functions-repository.ts +++ b/src/model/cli/cli-functions-repository.ts @@ -34,7 +34,6 @@ export class CliFunctionsRepository { }); } - // eslint-disable-next-line no-unused-vars public static PushCliFunctionSource(cliFunction: any) {} } diff --git a/src/model/cli/cli.ts b/src/model/cli/cli.ts index de7854e79..746c804a0 100644 --- a/src/model/cli/cli.ts +++ b/src/model/cli/cli.ts @@ -1,14 +1,13 @@ -import { Command } from 'commander-ts'; -import { BuildParameters, CloudRunner, ImageTag, Input } from '..'; -import * as core from '@actions/core'; -import { ActionYamlReader } from '../input-readers/action-yaml'; -import CloudRunnerLogger from '../cloud-runner/services/cloud-runner-logger'; -import CloudRunnerQueryOverride from '../cloud-runner/services/cloud-runner-query-override'; -import { CliFunction, CliFunctionsRepository } from './cli-functions-repository'; -import { AwsCliCommands } from '../cloud-runner/providers/aws/commands/aws-cli-commands'; -import { Caching } from '../cloud-runner/remote-client/caching'; -import { LfsHashing } from '../cloud-runner/services/lfs-hashing'; -import { RemoteClient } from '../cloud-runner/remote-client'; +import { Parameters, CloudRunner, ImageTag, Input } from '../index.ts'; +import { Command, core } from '../../dependencies.ts'; +import { ActionYamlReader } from '../input-readers/action-yaml.ts'; +import CloudRunnerLogger from '../cloud-runner/services/cloud-runner-logger.ts'; +import CloudRunnerQueryOverride from '../cloud-runner/services/cloud-runner-query-override.ts'; +import { CliFunction, CliFunctionsRepository } from './cli-functions-repository.ts'; +import { AwsCliCommands } from '../cloud-runner/providers/aws/commands/aws-cli-commands.ts'; +import { Caching } from '../cloud-runner/remote-client/caching.ts'; +import { LfsHashing } from '../cloud-runner/services/lfs-hashing.ts'; +import { RemoteClient } from '../cloud-runner/remote-client/index.ts'; export class Cli { public static options; @@ -69,8 +68,8 @@ export class Cli { @CliFunction(`print-input`, `prints all input`) private static logInput() { - core.info(`\n`); - core.info(`INPUT:`); + log.info(`\n`); + log.info(`INPUT:`); const properties = Object.getOwnPropertyNames(Input); for (const element of properties) { if ( @@ -81,15 +80,15 @@ export class Cli { element !== 'cliOptions' && element !== 'prototype' ) { - core.info(`${element} ${Input[element]}`); + log.info(`${element} ${Input[element]}`); } } - core.info(`\n`); + log.info(`\n`); } @CliFunction(`cli`, `runs a cloud runner build`) public static async CLIBuild(): Promise { - const buildParameter = await BuildParameters.create(); + const buildParameter = await Parameters.create(); const baseImage = new ImageTag(buildParameter); return await CloudRunner.run(buildParameter, baseImage.toString()); diff --git a/src/model/cloud-runner/cloud-runner-step-state.ts b/src/model/cloud-runner/cloud-runner-step-state.ts index 94f744bb1..602882164 100644 --- a/src/model/cloud-runner/cloud-runner-step-state.ts +++ b/src/model/cloud-runner/cloud-runner-step-state.ts @@ -1,5 +1,5 @@ -import CloudRunnerEnvironmentVariable from './services/cloud-runner-environment-variable'; -import CloudRunnerSecret from './services/cloud-runner-secret'; +import CloudRunnerEnvironmentVariable from './services/cloud-runner-environment-variable.ts'; +import CloudRunnerSecret from './services/cloud-runner-secret.ts'; export class CloudRunnerStepState { public image: string; diff --git a/src/model/cloud-runner/cloud-runner.test.ts b/src/model/cloud-runner/cloud-runner.test.ts index b8e51ed34..e09cb6009 100644 --- a/src/model/cloud-runner/cloud-runner.test.ts +++ b/src/model/cloud-runner/cloud-runner.test.ts @@ -1,12 +1,12 @@ -import { BuildParameters, ImageTag } from '..'; -import CloudRunner from './cloud-runner'; -import Input from '../input'; -import { CloudRunnerStatics } from './cloud-runner-statics'; -import { TaskParameterSerializer } from './services/task-parameter-serializer'; -import UnityVersioning from '../unity-versioning'; -import { Cli } from '../cli/cli'; -import CloudRunnerLogger from './services/cloud-runner-logger'; -import { v4 as uuidv4 } from 'uuid'; +import { Parameters, ImageTag } from '..'; +import CloudRunner from './cloud-runner.ts'; +import Input from '../input.ts'; +import { CloudRunnerStatics } from './cloud-runner-statics.ts'; +import { TaskParameterSerializer } from './services/task-parameter-serializer.ts'; +import UnityVersionDetector from '../../middleware/engine-detection/unity-version-detector.ts'; +import { Cli } from '../cli/cli.ts'; +import CloudRunnerLogger from './services/cloud-runner-logger.ts'; +import { v4 as uuidv4 } from '../../../node_modules/uuid'; describe('Cloud Runner', () => { it('responds', () => {}); @@ -20,7 +20,7 @@ describe('Cloud Runner', () => { Cli.options = { versioning: 'None', projectPath: 'test-project', - unityVersion: UnityVersioning.read('test-project'), + engineVersion: UnityVersionDetector.read('test-project'), targetPlatform: 'StandaloneLinux64', customJob: ` - name: 'step 1' @@ -34,7 +34,7 @@ describe('Cloud Runner', () => { Input.githubInputEnabled = false; // Setup parameters - const buildParameter = await BuildParameters.create(); + const buildParameter = await Parameters.create(); Input.githubInputEnabled = true; const baseImage = new ImageTag(buildParameter); @@ -43,7 +43,7 @@ describe('Cloud Runner', () => { // Assert results expect(file).toContain(JSON.stringify(buildParameter)); - expect(file).toContain(`${Input.ToEnvVarFormat(testSecretName)}=${testSecretValue}`); + expect(file).toContain(`${Input.toEnvVarFormat(testSecretName)}=${testSecretValue}`); const environmentVariables = TaskParameterSerializer.readBuildEnvironmentVariables(); const newLinePurgedFile = file .replace(/\s+/g, '') @@ -63,17 +63,17 @@ describe('Cloud Runner', () => { } } delete Cli.options; - }, 1000000); + }, 1_000_000); it('Run one build it should not use cache, run subsequent build which should use cache', async () => { Cli.options = { versioning: 'None', projectPath: 'test-project', - unityVersion: UnityVersioning.determineUnityVersion('test-project', UnityVersioning.read('test-project')), + engineVersion: UnityVersionDetector.determineUnityVersion('test-project', UnityVersionDetector.read('test-project')), targetPlatform: 'StandaloneLinux64', cacheKey: `test-case-${uuidv4()}`, }; Input.githubInputEnabled = false; - const buildParameter = await BuildParameters.create(); + const buildParameter = await Parameters.create(); const baseImage = new ImageTag(buildParameter); const results = await CloudRunner.run(buildParameter, baseImage.toString()); const libraryString = 'Rebuilding Library because the asset database could not be found!'; @@ -81,7 +81,7 @@ describe('Cloud Runner', () => { expect(results).toContain(libraryString); expect(results).toContain(buildSucceededString); CloudRunnerLogger.log(`run 1 succeeded`); - const buildParameter2 = await BuildParameters.create(); + const buildParameter2 = await Parameters.create(); const baseImage2 = new ImageTag(buildParameter2); const results2 = await CloudRunner.run(buildParameter2, baseImage2.toString()); CloudRunnerLogger.log(`run 2 succeeded`); @@ -89,14 +89,14 @@ describe('Cloud Runner', () => { expect(results2).toEqual(expect.not.stringContaining(libraryString)); Input.githubInputEnabled = true; delete Cli.options; - }, 1000000); + }, 1_000_000); } it('Local cloud runner returns commands', async () => { // Build parameters Cli.options = { versioning: 'None', projectPath: 'test-project', - unityVersion: UnityVersioning.read('test-project'), + engineVersion: UnityVersionDetector.read('test-project'), cloudRunnerCluster: 'local-system', targetPlatform: 'StandaloneLinux64', customJob: ` @@ -111,32 +111,32 @@ describe('Cloud Runner', () => { Input.githubInputEnabled = false; // Setup parameters - const buildParameter = await BuildParameters.create(); + const buildParameter = await Parameters.create(); const baseImage = new ImageTag(buildParameter); // Run the job await expect(CloudRunner.run(buildParameter, baseImage.toString())).resolves.not.toThrow(); Input.githubInputEnabled = true; delete Cli.options; - }, 1000000); + }, 1_000_000); it('Test cloud runner returns commands', async () => { // Build parameters Cli.options = { versioning: 'None', projectPath: 'test-project', - unityVersion: UnityVersioning.read('test-project'), + engineVersion: UnityVersionDetector.read('test-project'), cloudRunnerCluster: 'test', targetPlatform: 'StandaloneLinux64', }; Input.githubInputEnabled = false; // Setup parameters - const buildParameter = await BuildParameters.create(); + const buildParameter = await Parameters.create(); const baseImage = new ImageTag(buildParameter); // Run the job await expect(CloudRunner.run(buildParameter, baseImage.toString())).resolves.not.toThrow(); Input.githubInputEnabled = true; delete Cli.options; - }, 1000000); + }, 1_000_000); }); diff --git a/src/model/cloud-runner/cloud-runner.ts b/src/model/cloud-runner/cloud-runner.ts index 9234f427a..260f72534 100644 --- a/src/model/cloud-runner/cloud-runner.ts +++ b/src/model/cloud-runner/cloud-runner.ts @@ -1,25 +1,25 @@ -import AwsBuildPlatform from './providers/aws'; -import { BuildParameters, Input } from '..'; -import Kubernetes from './providers/k8s'; -import CloudRunnerLogger from './services/cloud-runner-logger'; -import { CloudRunnerStepState } from './cloud-runner-step-state'; -import { WorkflowCompositionRoot } from './workflows/workflow-composition-root'; -import { CloudRunnerError } from './error/cloud-runner-error'; -import { TaskParameterSerializer } from './services/task-parameter-serializer'; -import * as core from '@actions/core'; -import CloudRunnerSecret from './services/cloud-runner-secret'; -import { ProviderInterface } from './providers/provider-interface'; -import CloudRunnerEnvironmentVariable from './services/cloud-runner-environment-variable'; -import TestCloudRunner from './providers/test'; -import LocalCloudRunner from './providers/local'; -import LocalDockerCloudRunner from './providers/local-docker'; +import AwsBuildPlatform from './providers/aws/index.ts'; +import { Parameters, Input } from '../index.ts'; +import Kubernetes from './providers/k8s/index.ts'; +import CloudRunnerLogger from './services/cloud-runner-logger.ts'; +import { CloudRunnerStepState } from './cloud-runner-step-state.ts'; +import { WorkflowCompositionRoot } from './workflows/workflow-composition-root.ts'; +import { CloudRunnerError } from './error/cloud-runner-error.ts'; +import { TaskParameterSerializer } from './services/task-parameter-serializer.ts'; +import { core } from '../../dependencies.ts'; +import CloudRunnerSecret from './services/cloud-runner-secret.ts'; +import { ProviderInterface } from './providers/provider-interface.ts'; +import CloudRunnerEnvironmentVariable from './services/cloud-runner-environment-variable.ts'; +import TestCloudRunner from './providers/test/index.ts'; +import LocalCloudRunner from './providers/local/index.ts'; +import LocalDockerCloudRunner from './providers/local-docker/index.ts'; class CloudRunner { public static Provider: ProviderInterface; - static buildParameters: BuildParameters; + static buildParameters: Parameters; public static defaultSecrets: CloudRunnerSecret[]; public static cloudRunnerEnvironmentVariables: CloudRunnerEnvironmentVariable[]; - private static setup(buildParameters: BuildParameters) { + private static setup(buildParameters: Parameters) { CloudRunnerLogger.setup(); CloudRunner.buildParameters = buildParameters; CloudRunner.setupBuildPlatform(); @@ -28,10 +28,10 @@ class CloudRunner { if (!buildParameters.isCliMode) { const buildParameterPropertyNames = Object.getOwnPropertyNames(buildParameters); for (const element of CloudRunner.cloudRunnerEnvironmentVariables) { - core.setOutput(Input.ToEnvVarFormat(element.name), element.value); + core.setOutput(Input.toEnvVarFormat(element.name), element.value); } for (const element of buildParameterPropertyNames) { - core.setOutput(Input.ToEnvVarFormat(element), buildParameters[element]); + core.setOutput(Input.toEnvVarFormat(element), buildParameters[element]); } } } @@ -57,7 +57,7 @@ class CloudRunner { } } - static async run(buildParameters: BuildParameters, baseImage: string) { + static async run(buildParameters: Parameters, baseImage: string) { CloudRunner.setup(buildParameters); try { if (!CloudRunner.buildParameters.isCliMode) core.startGroup('Setup shared cloud runner resources'); diff --git a/src/model/cloud-runner/error/cloud-runner-error.ts b/src/model/cloud-runner/error/cloud-runner-error.ts index c290dcceb..73a98105f 100644 --- a/src/model/cloud-runner/error/cloud-runner-error.ts +++ b/src/model/cloud-runner/error/cloud-runner-error.ts @@ -1,16 +1,19 @@ -import CloudRunnerLogger from '../services/cloud-runner-logger'; -import * as core from '@actions/core'; -import CloudRunner from '../cloud-runner'; +import CloudRunnerLogger from '../services/cloud-runner-logger.ts'; +import { core } from '../../../dependencies.ts'; +import CloudRunner from '../cloud-runner.ts'; export class CloudRunnerError { public static async handleException(error: unknown) { CloudRunnerLogger.error(JSON.stringify(error, undefined, 4)); - core.setFailed('Cloud Runner failed'); + log.error('Cloud runner failed'); + await CloudRunner.Provider.cleanup( CloudRunner.buildParameters.buildGuid, CloudRunner.buildParameters, CloudRunner.buildParameters.branch, CloudRunner.defaultSecrets, ); + + Deno.exit(1); } } diff --git a/src/model/cloud-runner/providers/aws/aws-base-stack.ts b/src/model/cloud-runner/providers/aws/aws-base-stack.ts index a60cc43a5..d34eb668c 100644 --- a/src/model/cloud-runner/providers/aws/aws-base-stack.ts +++ b/src/model/cloud-runner/providers/aws/aws-base-stack.ts @@ -1,8 +1,6 @@ -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import * as core from '@actions/core'; -import * as SDK from 'aws-sdk'; -import { BaseStackFormation } from './cloud-formations/base-stack-formation'; -const crypto = require('crypto'); +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { core, aws, crypto } from '../../../../dependencies.ts'; +import { BaseStackFormation } from './cloud-formations/base-stack-formation.ts'; export class AWSBaseStack { constructor(baseStackName: string) { @@ -10,33 +8,33 @@ export class AWSBaseStack { } private baseStackName: string; - async setupBaseStack(CF: SDK.CloudFormation) { + async setupBaseStack(CF: aws.CloudFormation) { const baseStackName = this.baseStackName; const baseStack = BaseStackFormation.formation; // Cloud Formation Input - const describeStackInput: SDK.CloudFormation.DescribeStacksInput = { + const describeStackInput: aws.CloudFormation.DescribeStacksInput = { StackName: baseStackName, }; - const parametersWithoutHash: SDK.CloudFormation.Parameter[] = [ + const parametersWithoutHash: aws.CloudFormation.Parameter[] = [ { ParameterKey: 'EnvironmentName', ParameterValue: baseStackName }, ]; const parametersHash = crypto .createHash('md5') .update(baseStack + JSON.stringify(parametersWithoutHash)) .digest('hex'); - const parameters: SDK.CloudFormation.Parameter[] = [ + const parameters: aws.CloudFormation.Parameter[] = [ ...parametersWithoutHash, - ...[{ ParameterKey: 'Version', ParameterValue: parametersHash }], + { ParameterKey: 'Version', ParameterValue: parametersHash }, ]; - const updateInput: SDK.CloudFormation.UpdateStackInput = { + const updateInput: aws.CloudFormation.UpdateStackInput = { StackName: baseStackName, TemplateBody: baseStack, Parameters: parameters, Capabilities: ['CAPABILITY_IAM'], }; - const createStackInput: SDK.CloudFormation.CreateStackInput = { + const createStackInput: aws.CloudFormation.CreateStackInput = { StackName: baseStackName, TemplateBody: baseStack, Parameters: parameters, @@ -57,7 +55,7 @@ export class AWSBaseStack { await CF.createStack(createStackInput).promise(); CloudRunnerLogger.log(`created stack (version: ${parametersHash})`); } - const CFState = await describeStack(); + let CFState = await describeStack(); let stack = CFState.Stacks?.[0]; if (!stack) { throw new Error(`Base stack doesn't exist, even after creation, stackExists check: ${stackExists}`); @@ -86,7 +84,9 @@ export class AWSBaseStack { } else { CloudRunnerLogger.log(`No update required`); } - stack = (await describeStack()).Stacks?.[0]; + + CFState = await describeStack(); + stack = CFState.Stacks?.[0]; if (!stack) { throw new Error( `Base stack doesn't exist, even after updating and creation, stackExists check: ${stackExists}`, @@ -98,7 +98,7 @@ export class AWSBaseStack { } CloudRunnerLogger.log('base stack is now ready'); } catch (error) { - core.error(JSON.stringify(await describeStack(), undefined, 4)); + log.error(JSON.stringify(await describeStack(), undefined, 4)); throw error; } } diff --git a/src/model/cloud-runner/providers/aws/aws-cloud-formation-templates.ts b/src/model/cloud-runner/providers/aws/aws-cloud-formation-templates.ts index 0d0f83a45..fd3ea2322 100644 --- a/src/model/cloud-runner/providers/aws/aws-cloud-formation-templates.ts +++ b/src/model/cloud-runner/providers/aws/aws-cloud-formation-templates.ts @@ -1,4 +1,4 @@ -import { TaskDefinitionFormation } from './cloud-formations/task-definition-formation'; +import { TaskDefinitionFormation } from './cloud-formations/task-definition-formation.ts'; export class AWSCloudFormationTemplates { public static getParameterTemplate(p1) { diff --git a/src/model/cloud-runner/providers/aws/aws-error.ts b/src/model/cloud-runner/providers/aws/aws-error.ts index 7acbe3c5b..26b706c32 100644 --- a/src/model/cloud-runner/providers/aws/aws-error.ts +++ b/src/model/cloud-runner/providers/aws/aws-error.ts @@ -1,15 +1,15 @@ -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import * as SDK from 'aws-sdk'; -import * as core from '@actions/core'; -import CloudRunner from '../../cloud-runner'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { aws } from '../../../../dependencies.ts'; +import CloudRunner from '../../cloud-runner.ts'; export class AWSError { - static async handleStackCreationFailure(error: any, CF: SDK.CloudFormation, taskDefStackName: string) { + static async handleStackCreationFailure(error: any, CF: aws.CloudFormation, taskDefStackName: string) { CloudRunnerLogger.log('aws error: '); - core.error(JSON.stringify(error, undefined, 4)); + log.error(JSON.stringify(error, undefined, 4)); if (CloudRunner.buildParameters.cloudRunnerIntegrationTests) { CloudRunnerLogger.log('Getting events and resources for task stack'); - const events = (await CF.describeStackEvents({ StackName: taskDefStackName }).promise()).StackEvents; + const stackEventsDescription = await CF.describeStackEvents({ StackName: taskDefStackName }).promise(); + const events = stackEventsDescription.StackEvents; CloudRunnerLogger.log(JSON.stringify(events, undefined, 4)); } } diff --git a/src/model/cloud-runner/providers/aws/aws-job-stack.ts b/src/model/cloud-runner/providers/aws/aws-job-stack.ts index c970faa98..8dba533b4 100644 --- a/src/model/cloud-runner/providers/aws/aws-job-stack.ts +++ b/src/model/cloud-runner/providers/aws/aws-job-stack.ts @@ -1,10 +1,10 @@ -import * as SDK from 'aws-sdk'; -import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; -import { AWSCloudFormationTemplates } from './aws-cloud-formation-templates'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { AWSError } from './aws-error'; -import CloudRunner from '../../cloud-runner'; +import { aws } from '../../../../dependencies.ts'; +import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; +import { AWSCloudFormationTemplates } from './aws-cloud-formation-templates.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { AWSError } from './aws-error.ts'; +import CloudRunner from '../../cloud-runner.ts'; export class AWSJobStack { private baseStackName: string; @@ -13,7 +13,7 @@ export class AWSJobStack { } public async setupCloudFormations( - CF: SDK.CloudFormation, + CF: aws.CloudFormation, buildGuid: string, image: string, entrypoint: string[], @@ -118,7 +118,7 @@ export class AWSJobStack { } } } - const createStackInput: SDK.CloudFormation.CreateStackInput = { + const createStackInput: aws.CloudFormation.CreateStackInput = { StackName: taskDefStackName, TemplateBody: taskDefCloudFormation, Capabilities: ['CAPABILITY_IAM'], @@ -134,13 +134,13 @@ export class AWSJobStack { throw error; } - const taskDefResources = ( - await CF.describeStackResources({ - StackName: taskDefStackName, - }).promise() - ).StackResources; + const { StackResources: taskDefResources } = await CF.describeStackResources({ + StackName: taskDefStackName, + }).promise(); - const baseResources = (await CF.describeStackResources({ StackName: this.baseStackName }).promise()).StackResources; + const { StackResources: baseResources } = await CF.describeStackResources({ + StackName: this.baseStackName, + }).promise(); return { taskDefStackName, diff --git a/src/model/cloud-runner/providers/aws/aws-task-runner.ts b/src/model/cloud-runner/providers/aws/aws-task-runner.ts index 1efb22942..424e89f0c 100644 --- a/src/model/cloud-runner/providers/aws/aws-task-runner.ts +++ b/src/model/cloud-runner/providers/aws/aws-task-runner.ts @@ -1,19 +1,17 @@ -import * as AWS from 'aws-sdk'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import * as core from '@actions/core'; -import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def'; -import * as zlib from 'zlib'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { Input } from '../../..'; -import CloudRunner from '../../cloud-runner'; -import { CloudRunnerBuildCommandProcessor } from '../../services/cloud-runner-build-command-process'; -import { FollowLogStreamService } from '../../services/follow-log-stream-service'; +import { aws, core, compress } from '../../../../dependencies.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { Input } from '../../../index.ts'; +import CloudRunner from '../../cloud-runner.ts'; +import { CloudRunnerBuildCommandProcessor } from '../../services/cloud-runner-build-command-process.ts'; +import { FollowLogStreamService } from '../../services/follow-log-stream-service.ts'; class AWSTaskRunner { static async runTask( taskDef: CloudRunnerAWSTaskDef, - ECS: AWS.ECS, - CF: AWS.CloudFormation, + ECS: aws.ECS, + CF: aws.CloudFormation, environment: CloudRunnerEnvironmentVariable[], buildGuid: string, commands: string, @@ -55,9 +53,8 @@ class AWSTaskRunner { const taskArn = task.tasks?.[0].taskArn || ''; CloudRunnerLogger.log('Cloud runner job is starting'); await AWSTaskRunner.waitUntilTaskRunning(ECS, taskArn, cluster); - CloudRunnerLogger.log( - `Cloud runner job status is running ${(await AWSTaskRunner.describeTasks(ECS, cluster, taskArn))?.lastStatus}`, - ); + const { lastStatus } = await AWSTaskRunner.describeTasks(ECS, cluster, taskArn); + CloudRunnerLogger.log(`Cloud runner job status is running ${lastStatus}`); const { output, shouldCleanup } = await this.streamLogsUntilTaskStops( ECS, CF, @@ -85,24 +82,21 @@ class AWSTaskRunner { } } - private static async waitUntilTaskRunning(ECS: AWS.ECS, taskArn: string, cluster: string) { + private static async waitUntilTaskRunning(ECS: aws.ECS, taskArn: string, cluster: string) { try { await ECS.waitFor('tasksRunning', { tasks: [taskArn], cluster }).promise(); } catch (error_) { const error = error_ as Error; await new Promise((resolve) => setTimeout(resolve, 3000)); - CloudRunnerLogger.log( - `Cloud runner job has ended ${ - (await AWSTaskRunner.describeTasks(ECS, cluster, taskArn)).containers?.[0].lastStatus - }`, - ); - - core.setFailed(error); - core.error(error); + const tasksDescription = await AWSTaskRunner.describeTasks(ECS, cluster, taskArn); + CloudRunnerLogger.log(`Cloud runner job has ended ${tasksDescription.containers?.[0].lastStatus}`); + + log.error(error); + Deno.exit(1); } } - static async describeTasks(ECS: AWS.ECS, clusterName: string, taskArn: string) { + static async describeTasks(ECS: aws.ECS, clusterName: string, taskArn: string) { const tasks = await ECS.describeTasks({ cluster: clusterName, tasks: [taskArn], @@ -115,14 +109,14 @@ class AWSTaskRunner { } static async streamLogsUntilTaskStops( - ECS: AWS.ECS, - CF: AWS.CloudFormation, + ECS: aws.ECS, + CF: aws.CloudFormation, taskDef: CloudRunnerAWSTaskDef, clusterName: string, taskArn: string, kinesisStreamName: string, ) { - const kinesis = new AWS.Kinesis(); + const kinesis = new aws.Kinesis(); const stream = await AWSTaskRunner.getLogStream(kinesis, kinesisStreamName); let iterator = await AWSTaskRunner.getLogIterator(kinesis, stream); @@ -150,7 +144,7 @@ class AWSTaskRunner { } private static async handleLogStreamIteration( - kinesis: AWS.Kinesis, + kinesis: aws.Kinesis, iterator: string, shouldReadLogs: boolean, taskDef: CloudRunnerAWSTaskDef, @@ -175,7 +169,7 @@ class AWSTaskRunner { return { iterator, shouldReadLogs, output, shouldCleanup }; } - private static checkStreamingShouldContinue(taskData: AWS.ECS.Task, timestamp: number, shouldReadLogs: boolean) { + private static checkStreamingShouldContinue(taskData: aws.ECS.Task, timestamp: number, shouldReadLogs: boolean) { if (taskData?.lastStatus === 'UNKNOWN') { CloudRunnerLogger.log('## Cloud runner job unknwon'); } @@ -184,7 +178,7 @@ class AWSTaskRunner { CloudRunnerLogger.log('## Cloud runner job stopped, streaming end of logs'); timestamp = Date.now(); } - if (timestamp !== 0 && Date.now() - timestamp > 30000) { + if (timestamp !== 0 && Date.now() - timestamp > 30_000) { CloudRunnerLogger.log('## Cloud runner status is not RUNNING for 30 seconds, last query for logs'); shouldReadLogs = false; } @@ -205,7 +199,7 @@ class AWSTaskRunner { if (records.Records.length > 0 && iterator) { for (let index = 0; index < records.Records.length; index++) { const json = JSON.parse( - zlib.gunzipSync(Buffer.from(records.Records[index].Data as string, 'base64')).toString('utf8'), + compress.gunzipSync(Buffer.from(records.Records[index].Data as string, 'base64')).toString('utf8'), ); if (json.messageType === 'DATA_MESSAGE') { for (let logEventsIndex = 0; logEventsIndex < json.logEvents.length; logEventsIndex++) { @@ -224,7 +218,7 @@ class AWSTaskRunner { return { shouldReadLogs, output, shouldCleanup }; } - private static async getLogStream(kinesis: AWS.Kinesis, kinesisStreamName: string) { + private static async getLogStream(kinesis: aws.Kinesis, kinesisStreamName: string) { return await kinesis .describeStream({ StreamName: kinesisStreamName, @@ -232,18 +226,16 @@ class AWSTaskRunner { .promise(); } - private static async getLogIterator(kinesis: AWS.Kinesis, stream) { - return ( - ( - await kinesis - .getShardIterator({ - ShardIteratorType: 'TRIM_HORIZON', - StreamName: stream.StreamDescription.StreamName, - ShardId: stream.StreamDescription.Shards[0].ShardId, - }) - .promise() - ).ShardIterator || '' - ); + private static async getLogIterator(kinesis: aws.Kinesis, stream) { + const description = await kinesis + .getShardIterator({ + ShardIteratorType: 'TRIM_HORIZON', + StreamName: stream.StreamDescription.StreamName, + ShardId: stream.StreamDescription.Shards[0].ShardId, + }) + .promise(); + + return description.ShardIterator || ''; } } export default AWSTaskRunner; diff --git a/src/model/cloud-runner/providers/aws/cloud-runner-aws-task-def.ts b/src/model/cloud-runner/providers/aws/cloud-runner-aws-task-def.ts index 4ec161965..5439bff66 100644 --- a/src/model/cloud-runner/providers/aws/cloud-runner-aws-task-def.ts +++ b/src/model/cloud-runner/providers/aws/cloud-runner-aws-task-def.ts @@ -1,9 +1,9 @@ -import * as AWS from 'aws-sdk'; +import { aws } from '../../../../dependencies.ts'; class CloudRunnerAWSTaskDef { public taskDefStackName!: string; public taskDefCloudFormation!: string; - public taskDefResources: AWS.CloudFormation.StackResources | undefined; - public baseResources: AWS.CloudFormation.StackResources | undefined; + public taskDefResources: aws.CloudFormation.StackResources | undefined; + public baseResources: aws.CloudFormation.StackResources | undefined; } export default CloudRunnerAWSTaskDef; diff --git a/src/model/cloud-runner/providers/aws/commands/aws-cli-commands.ts b/src/model/cloud-runner/providers/aws/commands/aws-cli-commands.ts index 7563606a7..1aee8e121 100644 --- a/src/model/cloud-runner/providers/aws/commands/aws-cli-commands.ts +++ b/src/model/cloud-runner/providers/aws/commands/aws-cli-commands.ts @@ -1,8 +1,8 @@ -import AWS from 'aws-sdk'; -import { CliFunction } from '../../../../cli/cli-functions-repository'; -import Input from '../../../../input'; -import CloudRunnerLogger from '../../../services/cloud-runner-logger'; -import { BaseStackFormation } from '../cloud-formations/base-stack-formation'; +import { aws } from '../../../../../dependencies.ts'; +import { CliFunction } from '../../../../cli/cli-functions-repository.ts'; +import Input from '../../../../input.ts'; +import CloudRunnerLogger from '../../../services/cloud-runner-logger.ts'; +import { BaseStackFormation } from '../cloud-formations/base-stack-formation.ts'; export class AwsCliCommands { @CliFunction(`aws-list-all`, `List all resources`) @@ -33,10 +33,11 @@ export class AwsCliCommands { } @CliFunction(`aws-list-stacks`, `List stacks`) static async awsListStacks(perResultCallback: any = false, verbose: boolean = false) { - process.env.AWS_REGION = Input.region; - const CF = new AWS.CloudFormation(); + Deno.env.set('AWS_REGION', Input.region); + const CF = new aws.CloudFormation(); + let cfStacks = await CF.listStacks().promise(); const stacks = - (await CF.listStacks().promise()).StackSummaries?.filter( + cfStacks.StackSummaries?.filter( (_x) => _x.StackStatus !== 'DELETE_COMPLETE', // && // _x.TemplateDescription === TaskDefinitionFormation.description.replace('\n', ''), ) || []; @@ -49,8 +50,9 @@ export class AwsCliCommands { ); if (perResultCallback) await perResultCallback(element); } + cfStacks = await CF.listStacks().promise(); const baseStacks = - (await CF.listStacks().promise()).StackSummaries?.filter( + cfStacks.StackSummaries?.filter( (_x) => _x.StackStatus !== 'DELETE_COMPLETE' && _x.TemplateDescription === BaseStackFormation.baseStackDecription, ) || []; @@ -71,19 +73,22 @@ export class AwsCliCommands { } @CliFunction(`aws-list-tasks`, `List tasks`) static async awsListTasks(perResultCallback: any = false) { - process.env.AWS_REGION = Input.region; - const ecs = new AWS.ECS(); - const clusters = (await ecs.listClusters().promise()).clusterArns || []; + Deno.env.set('AWS_REGION', Input.region); + const ecs = new aws.ECS(); + const ecsClusters = await ecs.listClusters().promise(); + const clusters = ecsClusters.clusterArns || []; CloudRunnerLogger.log(`Clusters ${clusters.length}`); for (const element of clusters) { - const input: AWS.ECS.ListTasksRequest = { + const input: aws.ECS.ListTasksRequest = { cluster: element, }; - const list = (await ecs.listTasks(input).promise()).taskArns || []; + const listedTasks = await ecs.listTasks(input).promise(); + const list = listedTasks.taskArns || []; if (list.length > 0) { - const describeInput: AWS.ECS.DescribeTasksRequest = { tasks: list, cluster: element }; - const describeList = (await ecs.describeTasks(describeInput).promise()).tasks || []; + const describeInput: aws.ECS.DescribeTasksRequest = { tasks: list, cluster: element }; + const tasksDescription = await ecs.describeTasks(describeInput).promise(); + const describeList = tasksDescription.tasks || []; if (describeList === []) { continue; } @@ -105,9 +110,9 @@ export class AwsCliCommands { } @CliFunction(`aws-list-log-groups`, `List tasks`) static async awsListLogGroups(perResultCallback: any = false, verbose: boolean = false) { - process.env.AWS_REGION = Input.region; - const ecs = new AWS.CloudWatchLogs(); - let logStreamInput: AWS.CloudWatchLogs.DescribeLogGroupsRequest = { + Deno.env.set('AWS_REGION', Input.region); + const ecs = new aws.CloudWatchLogs(); + let logStreamInput: aws.CloudWatchLogs.DescribeLogGroupsRequest = { /* logGroupNamePrefix: 'game-ci' */ }; let logGroupsDescribe = await ecs.describeLogGroups(logStreamInput).promise(); @@ -138,10 +143,10 @@ export class AwsCliCommands { } private static async cleanup(deleteResources = false, OneDayOlderOnly: boolean = false) { - process.env.AWS_REGION = Input.region; - const CF = new AWS.CloudFormation(); - const ecs = new AWS.ECS(); - const cwl = new AWS.CloudWatchLogs(); + Deno.env.set('AWS_REGION', Input.region); + const CF = new aws.CloudFormation(); + const ecs = new aws.ECS(); + const cwl = new aws.CloudWatchLogs(); await AwsCliCommands.awsListStacks(async (element) => { if (deleteResources && (!OneDayOlderOnly || AwsCliCommands.isOlderThan1day(element.CreationTime))) { if (element.StackName === 'game-ci' || element.TemplateDescription === 'Game-CI base stack') { @@ -150,7 +155,7 @@ export class AwsCliCommands { return; } CloudRunnerLogger.log(`Deleting ${element.logGroupName}`); - const deleteStackInput: AWS.CloudFormation.DeleteStackInput = { StackName: element.StackName }; + const deleteStackInput: aws.CloudFormation.DeleteStackInput = { StackName: element.StackName }; await CF.deleteStack(deleteStackInput).promise(); } }); diff --git a/src/model/cloud-runner/providers/aws/index.ts b/src/model/cloud-runner/providers/aws/index.ts index 4be2f66f7..b59b34759 100644 --- a/src/model/cloud-runner/providers/aws/index.ts +++ b/src/model/cloud-runner/providers/aws/index.ts @@ -1,39 +1,31 @@ -import * as SDK from 'aws-sdk'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def'; -import AWSTaskRunner from './aws-task-runner'; -import { ProviderInterface } from '../provider-interface'; -import BuildParameters from '../../../build-parameters'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { AWSJobStack } from './aws-job-stack'; -import { AWSBaseStack } from './aws-base-stack'; -import { Input } from '../../..'; +import { aws } from '../../../../dependencies.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def.ts'; +import AWSTaskRunner from './aws-task-runner.ts'; +import { ProviderInterface } from '../provider-interface.ts'; +import Parameters from '../../../parameters.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { AWSJobStack } from './aws-job-stack.ts'; +import { AWSBaseStack } from './aws-base-stack.ts'; +import { Input } from '../../../index.ts'; class AWSBuildEnvironment implements ProviderInterface { private baseStackName: string; - constructor(buildParameters: BuildParameters) { + constructor(buildParameters: Parameters) { this.baseStackName = buildParameters.awsBaseStackName; } async cleanup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} async setup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} @@ -46,9 +38,9 @@ class AWSBuildEnvironment implements ProviderInterface { environment: CloudRunnerEnvironmentVariable[], secrets: CloudRunnerSecret[], ): Promise { - process.env.AWS_REGION = Input.region; - const ECS = new SDK.ECS(); - const CF = new SDK.CloudFormation(); + Deno.env.set('AWS_REGION', Input.region); + const ECS = new aws.ECS(); + const CF = new aws.CloudFormation(); CloudRunnerLogger.log(`AWS Region: ${CF.config.region}`); const entrypoint = ['/bin/sh']; const startTimeMs = Date.now(); @@ -86,7 +78,7 @@ class AWSBuildEnvironment implements ProviderInterface { } } - async cleanupResources(CF: SDK.CloudFormation, taskDef: CloudRunnerAWSTaskDef) { + async cleanupResources(CF: aws.CloudFormation, taskDef: CloudRunnerAWSTaskDef) { CloudRunnerLogger.log('Cleanup starting'); await CF.deleteStack({ StackName: taskDef.taskDefStackName, diff --git a/src/model/cloud-runner/providers/k8s/index.ts b/src/model/cloud-runner/providers/k8s/index.ts index 65cb4ed2d..454a984c2 100644 --- a/src/model/cloud-runner/providers/k8s/index.ts +++ b/src/model/cloud-runner/providers/k8s/index.ts @@ -1,25 +1,22 @@ -import * as k8s from '@kubernetes/client-node'; -import { BuildParameters, Output } from '../../..'; -import * as core from '@actions/core'; -import { ProviderInterface } from '../provider-interface'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; -import KubernetesStorage from './kubernetes-storage'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import KubernetesTaskRunner from './kubernetes-task-runner'; -import KubernetesSecret from './kubernetes-secret'; -import waitUntil from 'async-wait-until'; -import KubernetesJobSpecFactory from './kubernetes-job-spec-factory'; -import KubernetesServiceAccount from './kubernetes-service-account'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { CoreV1Api } from '@kubernetes/client-node'; -import DependencyOverrideService from '../../services/depdency-override-service'; +import { Parameters, Output } from '../../../index.ts'; +import { k8sTypes, k8s, waitUntil } from '../../../../dependencies.ts'; +import { ProviderInterface } from '../provider-interface.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; +import KubernetesStorage from './kubernetes-storage.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import KubernetesTaskRunner from './kubernetes-task-runner.ts'; +import KubernetesSecret from './kubernetes-secret.ts'; +import KubernetesJobSpecFactory from './kubernetes-job-spec-factory.ts'; +import KubernetesServiceAccount from './kubernetes-service-account.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import DependencyOverrideService from '../../services/depdency-override-service.ts'; class Kubernetes implements ProviderInterface { - private kubeConfig: k8s.KubeConfig; - private kubeClient: k8s.CoreV1Api; - private kubeClientBatch: k8s.BatchV1Api; + private kubeConfig: k8sTypes.KubeConfig; + private kubeClient: k8sTypes.CoreV1Api; + private kubeClientBatch: k8sTypes.BatchV1Api; private buildGuid: string = ''; - private buildParameters: BuildParameters; + private buildParameters: Parameters; private pvcName: string = ''; private secretName: string = ''; private jobName: string = ''; @@ -29,10 +26,10 @@ class Kubernetes implements ProviderInterface { private cleanupCronJobName: string = ''; private serviceAccountName: string = ''; - constructor(buildParameters: BuildParameters) { + constructor(buildParameters: Parameters) { this.kubeConfig = new k8s.KubeConfig(); this.kubeConfig.loadFromDefault(); - this.kubeClient = this.kubeConfig.makeApiClient(k8s.CoreV1Api); + this.kubeClient = this.kubeConfig.makeApiClient(k8sTypes.CoreV1Api); this.kubeClientBatch = this.kubeConfig.makeApiClient(k8s.BatchV1Api); CloudRunnerLogger.log('Loaded default Kubernetes configuration for this environment'); @@ -41,10 +38,8 @@ class Kubernetes implements ProviderInterface { } public async setup( buildGuid: string, - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) { try { @@ -122,9 +117,7 @@ class Kubernetes implements ProviderInterface { ); break; } catch (error: any) { - if (error.message.includes(`HTTP`)) { - continue; - } else { + if (!error.message.includes(`HTTP`)) { throw error; } } @@ -134,7 +127,7 @@ class Kubernetes implements ProviderInterface { return output; } catch (error) { CloudRunnerLogger.log('Running job failed'); - core.error(JSON.stringify(error, undefined, 4)); + log.error(JSON.stringify(error, undefined, 4)); await this.cleanupTaskResources(); throw error; } @@ -154,33 +147,32 @@ class Kubernetes implements ProviderInterface { await new Promise((promise) => setTimeout(promise, 5000)); } catch (error) { CloudRunnerLogger.log('Failed to cleanup, error:'); - core.error(JSON.stringify(error, undefined, 4)); + log.error(JSON.stringify(error, undefined, 4)); CloudRunnerLogger.log('Abandoning cleanup, build error:'); throw error; } try { await waitUntil( async () => { - const jobBody = (await this.kubeClientBatch.readNamespacedJob(this.jobName, this.namespace)).body; - const podBody = (await this.kubeClient.readNamespacedPod(this.podName, this.namespace)).body; + const { body: jobBody } = await this.kubeClientBatch.readNamespacedJob(this.jobName, this.namespace); + const { body: podBody } = await this.kubeClient.readNamespacedPod(this.podName, this.namespace); return (jobBody === null || jobBody.status?.active === 0) && podBody === null; }, { - timeout: 500000, - intervalBetweenAttempts: 15000, + timeout: 500_000, + intervalBetweenAttempts: 15_000, }, ); - // eslint-disable-next-line no-empty - } catch {} + } catch { + log.debug('Moved into empty catch block'); + } } async cleanup( buildGuid: string, - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) { CloudRunnerLogger.log(`deleting PVC`); @@ -190,7 +182,7 @@ class Kubernetes implements ProviderInterface { process.exit(); } - static async findPodFromJob(kubeClient: CoreV1Api, jobName: string, namespace: string) { + static async findPodFromJob(kubeClient: k8sTypes.CoreV1Api, jobName: string, namespace: string) { const namespacedPods = await kubeClient.listNamespacedPod(namespace); const pod = namespacedPods.body.items.find((x) => x.metadata?.labels?.['job-name'] === jobName); if (pod === undefined) { diff --git a/src/model/cloud-runner/providers/k8s/kubernetes-job-spec-factory.ts b/src/model/cloud-runner/providers/k8s/kubernetes-job-spec-factory.ts index 0bb9294a5..dc1e90ab3 100644 --- a/src/model/cloud-runner/providers/k8s/kubernetes-job-spec-factory.ts +++ b/src/model/cloud-runner/providers/k8s/kubernetes-job-spec-factory.ts @@ -1,9 +1,9 @@ -import { V1EnvVar, V1EnvVarSource, V1SecretKeySelector } from '@kubernetes/client-node'; -import BuildParameters from '../../../build-parameters'; -import { CloudRunnerBuildCommandProcessor } from '../../services/cloud-runner-build-command-process'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; -import CloudRunner from '../../cloud-runner'; +import { V1EnvVar, V1EnvVarSource, V1SecretKeySelector } from '../../../../dependencies.ts'; +import Parameters from '../../../parameters.ts'; +import { CloudRunnerBuildCommandProcessor } from '../../services/cloud-runner-build-command-process.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; +import CloudRunner from '../../cloud-runner.ts'; class KubernetesJobSpecFactory { static getJobSpec( @@ -14,67 +14,65 @@ class KubernetesJobSpecFactory { environment: CloudRunnerEnvironmentVariable[], secrets: CloudRunnerSecret[], buildGuid: string, - buildParameters: BuildParameters, + buildParameters: Parameters, secretName, pvcName, jobName, k8s, ) { environment.push( - ...[ - { - name: 'GITHUB_SHA', - value: buildGuid, - }, - { - name: 'GITHUB_WORKSPACE', - value: '/data/repo', - }, - { - name: 'PROJECT_PATH', - value: buildParameters.projectPath, - }, - { - name: 'BUILD_PATH', - value: buildParameters.buildPath, - }, - { - name: 'BUILD_FILE', - value: buildParameters.buildFile, - }, - { - name: 'BUILD_NAME', - value: buildParameters.buildName, - }, - { - name: 'BUILD_METHOD', - value: buildParameters.buildMethod, - }, - { - name: 'CUSTOM_PARAMETERS', - value: buildParameters.customParameters, - }, - { - name: 'CHOWN_FILES_TO', - value: buildParameters.chownFilesTo, - }, - { - name: 'BUILD_TARGET', - value: buildParameters.targetPlatform, - }, - { - name: 'ANDROID_VERSION_CODE', - value: buildParameters.androidVersionCode.toString(), - }, - { - name: 'ANDROID_KEYSTORE_NAME', - value: buildParameters.androidKeystoreName, - }, - { - name: 'ANDROID_KEYALIAS_NAME', - value: buildParameters.androidKeyaliasName, - }, - ], + { + name: 'GITHUB_SHA', + value: buildGuid, + }, + { + name: 'GITHUB_WORKSPACE', + value: '/data/repo', + }, + { + name: 'PROJECT_PATH', + value: buildParameters.projectPath, + }, + { + name: 'BUILD_PATH', + value: buildParameters.buildPath, + }, + { + name: 'BUILD_FILE', + value: buildParameters.buildFile, + }, + { + name: 'BUILD_NAME', + value: buildParameters.buildName, + }, + { + name: 'BUILD_METHOD', + value: buildParameters.buildMethod, + }, + { + name: 'CUSTOM_PARAMETERS', + value: buildParameters.customParameters, + }, + { + name: 'CHOWN_FILES_TO', + value: buildParameters.chownFilesTo, + }, + { + name: 'BUILD_TARGET', + value: buildParameters.targetPlatform, + }, + { + name: 'ANDROID_VERSION_CODE', + value: buildParameters.androidVersionCode.toString(), + }, + { + name: 'ANDROID_KEYSTORE_NAME', + value: buildParameters.androidKeystoreName, + }, + { + name: 'ANDROID_KEYALIAS_NAME', + value: buildParameters.androidKeyaliasName, + }, ); const job = new k8s.V1Job(); job.apiVersion = 'batch/v1'; @@ -114,7 +112,7 @@ class KubernetesJobSpecFactory { }, env: [ ...environment.map((x) => { - const environmentVariable = new V1EnvVar(); + const environmentVariable = new k8s.V1EnvVar(); environmentVariable.name = x.name; environmentVariable.value = x.value; diff --git a/src/model/cloud-runner/providers/k8s/kubernetes-secret.ts b/src/model/cloud-runner/providers/k8s/kubernetes-secret.ts index 76e3935bf..f439796db 100644 --- a/src/model/cloud-runner/providers/k8s/kubernetes-secret.ts +++ b/src/model/cloud-runner/providers/k8s/kubernetes-secret.ts @@ -1,14 +1,12 @@ -import { CoreV1Api } from '@kubernetes/client-node'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; -import * as k8s from '@kubernetes/client-node'; -const base64 = require('base-64'); +import { k8sTypes, k8s, base64 } from '../../../../dependencies.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; class KubernetesSecret { static async createSecret( secrets: CloudRunnerSecret[], secretName: string, namespace: string, - kubeClient: CoreV1Api, + kubeClient: k8sTypes.CoreV1Api, ) { const secret = new k8s.V1Secret(); secret.apiVersion = 'v1'; diff --git a/src/model/cloud-runner/providers/k8s/kubernetes-service-account.ts b/src/model/cloud-runner/providers/k8s/kubernetes-service-account.ts index 311027f32..225577e1a 100644 --- a/src/model/cloud-runner/providers/k8s/kubernetes-service-account.ts +++ b/src/model/cloud-runner/providers/k8s/kubernetes-service-account.ts @@ -1,8 +1,7 @@ -import { CoreV1Api } from '@kubernetes/client-node'; -import * as k8s from '@kubernetes/client-node'; +import { k8s, k8sTypes } from '../../../../dependencies.ts'; class KubernetesServiceAccount { - static async createServiceAccount(serviceAccountName: string, namespace: string, kubeClient: CoreV1Api) { + static async createServiceAccount(serviceAccountName: string, namespace: string, kubeClient: k8sTypes.CoreV1Api) { const serviceAccount = new k8s.V1ServiceAccount(); serviceAccount.apiVersion = 'v1'; serviceAccount.kind = 'ServiceAccount'; diff --git a/src/model/cloud-runner/providers/k8s/kubernetes-storage.ts b/src/model/cloud-runner/providers/k8s/kubernetes-storage.ts index 215948825..c450a756e 100644 --- a/src/model/cloud-runner/providers/k8s/kubernetes-storage.ts +++ b/src/model/cloud-runner/providers/k8s/kubernetes-storage.ts @@ -1,16 +1,12 @@ -import waitUntil from 'async-wait-until'; -import * as core from '@actions/core'; -import * as k8s from '@kubernetes/client-node'; -import BuildParameters from '../../../build-parameters'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import YAML from 'yaml'; -import { IncomingMessage } from 'http'; +import { k8sTypes, k8s, core, yaml, waitUntil, http } from '../../../../dependencies.ts'; +import Parameters from '../../../parameters.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; class KubernetesStorage { public static async createPersistentVolumeClaim( - buildParameters: BuildParameters, + buildParameters: Parameters, pvcName: string, - kubeClient: k8s.CoreV1Api, + kubeClient: k8sTypes.CoreV1Api, namespace: string, ) { if (buildParameters.kubeVolume) { @@ -19,9 +15,9 @@ class KubernetesStorage { return; } - const pvcList = (await kubeClient.listNamespacedPersistentVolumeClaim(namespace)).body.items.map( - (x) => x.metadata?.name, - ); + + const listedPvcs = await kubeClient.listNamespacedPersistentVolumeClaim(namespace); + const pvcList = listedPvcs.body.items.map((x) => x.metadata?.name); CloudRunnerLogger.log(`Current PVCs in namespace ${namespace}`); CloudRunnerLogger.log(JSON.stringify(pvcList, undefined, 4)); if (pvcList.includes(pvcName)) { @@ -37,17 +33,19 @@ class KubernetesStorage { await KubernetesStorage.handleResult(result, kubeClient, namespace, pvcName); } - public static async getPVCPhase(kubeClient: k8s.CoreV1Api, name: string, namespace: string) { + public static async getPVCPhase(kubeClient: k8sTypes.CoreV1Api, name: string, namespace: string) { try { - return (await kubeClient.readNamespacedPersistentVolumeClaim(name, namespace)).body.status?.phase; + const pvc = await kubeClient.readNamespacedPersistentVolumeClaim(name, namespace); + + return pvc.body.status?.phase; } catch (error) { - core.error('Failed to get PVC phase'); - core.error(JSON.stringify(error, undefined, 4)); + log.error('Failed to get PVC phase'); + log.error(JSON.stringify(error, undefined, 4)); throw error; } } - public static async watchUntilPVCNotPending(kubeClient: k8s.CoreV1Api, name: string, namespace: string) { + public static async watchUntilPVCNotPending(kubeClient: k8sTypes.CoreV1Api, name: string, namespace: string) { try { CloudRunnerLogger.log(`watch Until PVC Not Pending ${name} ${namespace}`); CloudRunnerLogger.log(`${await this.getPVCPhase(kubeClient, name, namespace)}`); @@ -56,28 +54,23 @@ class KubernetesStorage { return (await this.getPVCPhase(kubeClient, name, namespace)) === 'Pending'; }, { - timeout: 750000, - intervalBetweenAttempts: 15000, + timeout: 750_000, + intervalBetweenAttempts: 15_000, }, ); } catch (error: any) { - core.error('Failed to watch PVC'); - core.error(error.toString()); - core.error( - `PVC Body: ${JSON.stringify( - (await kubeClient.readNamespacedPersistentVolumeClaim(name, namespace)).body, - undefined, - 4, - )}`, - ); + log.error('Failed to watch PVC'); + log.error(error.toString()); + const pvc = await kubeClient.readNamespacedPersistentVolumeClaim(name, namespace); + log.error(`PVC Body: ${JSON.stringify(pvc.body, undefined, 4)}`); throw error; } } private static async createPVC( pvcName: string, - buildParameters: BuildParameters, - kubeClient: k8s.CoreV1Api, + buildParameters: Parameters, + kubeClient: k8sTypes.CoreV1Api, namespace: string, ) { const pvc = new k8s.V1PersistentVolumeClaim(); @@ -95,8 +88,8 @@ class KubernetesStorage { }, }, }; - if (process.env.K8s_STORAGE_PVC_SPEC) { - YAML.parse(process.env.K8s_STORAGE_PVC_SPEC); + if (Deno.env.get('K8s_STORAGE_PVC_SPEC')) { + yaml.parse(Deno.env.get('K8s_STORAGE_PVC_SPEC')); } const result = await kubeClient.createNamespacedPersistentVolumeClaim(namespace, pvc); @@ -104,8 +97,8 @@ class KubernetesStorage { } private static async handleResult( - result: { response: IncomingMessage; body: k8s.V1PersistentVolumeClaim }, - kubeClient: k8s.CoreV1Api, + result: { response: http.IncomingMessage; body: k8s.V1PersistentVolumeClaim }, + kubeClient: k8sTypes.CoreV1Api, namespace: string, pvcName: string, ) { diff --git a/src/model/cloud-runner/providers/k8s/kubernetes-task-runner.ts b/src/model/cloud-runner/providers/k8s/kubernetes-task-runner.ts index da70be190..ccb2074cf 100644 --- a/src/model/cloud-runner/providers/k8s/kubernetes-task-runner.ts +++ b/src/model/cloud-runner/providers/k8s/kubernetes-task-runner.ts @@ -1,15 +1,12 @@ -import { CoreV1Api, KubeConfig, Log } from '@kubernetes/client-node'; -import { Writable } from 'stream'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import * as core from '@actions/core'; -import { CloudRunnerStatics } from '../../cloud-runner-statics'; -import waitUntil from 'async-wait-until'; -import { FollowLogStreamService } from '../../services/follow-log-stream-service'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { k8sTypes, k8s, Writable, waitUntil } from '../../../../dependencies.ts'; +import { CloudRunnerStatics } from '../../cloud-runner-statics.ts'; +import { FollowLogStreamService } from '../../services/follow-log-stream-service.ts'; class KubernetesTaskRunner { static async runTask( - kubeConfig: KubeConfig, - kubeClient: CoreV1Api, + kubeConfig: k8sTypes.KubeConfig, + kubeClient: k8sTypes.CoreV1Api, jobName: string, podName: string, containerName: string, @@ -40,18 +37,19 @@ class KubernetesTaskRunner { }; try { const resultError = await new Promise((resolve) => - new Log(kubeConfig).log(namespace, podName, containerName, stream, resolve, logOptions), + new k8s.Log(kubeConfig).log(namespace, podName, containerName, stream, resolve, logOptions), ); stream.destroy(); if (resultError) { throw resultError; } if (!didStreamAnyLogs) { - core.error('Failed to stream any logs, listing namespace events, check for an error with the container'); - core.error( + log.error('Failed to stream any logs, listing namespace events, check for an error with the container'); + const listedEvents = await kubeClient.listNamespacedEvent(namespace); + log.error( JSON.stringify( { - events: (await kubeClient.listNamespacedEvent(namespace)).body.items + events: listedEvents.body.items .filter((x) => { return x.involvedObject.name === podName || x.involvedObject.name === jobName; }) @@ -80,7 +78,7 @@ class KubernetesTaskRunner { return output; } - static async watchUntilPodRunning(kubeClient: CoreV1Api, podName: string, namespace: string) { + static async watchUntilPodRunning(kubeClient: k8sTypes.CoreV1Api, podName: string, namespace: string) { let success: boolean = false; CloudRunnerLogger.log(`Watching ${podName} ${namespace}`); await waitUntil( @@ -93,13 +91,12 @@ class KubernetesTaskRunner { status.body.status?.conditions?.[0].message || '' }`, ); - if (success || phase !== 'Pending') return true; - return false; + return success || phase !== 'Pending'; }, { - timeout: 2000000, - intervalBetweenAttempts: 15000, + timeout: 2_000_000, + intervalBetweenAttempts: 15_000, }, ); diff --git a/src/model/cloud-runner/providers/local-docker/index.ts b/src/model/cloud-runner/providers/local-docker/index.ts index 397582824..9e8669564 100644 --- a/src/model/cloud-runner/providers/local-docker/index.ts +++ b/src/model/cloud-runner/providers/local-docker/index.ts @@ -1,43 +1,30 @@ -import BuildParameters from '../../../build-parameters'; -import { CloudRunnerSystem } from '../../services/cloud-runner-system'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { ProviderInterface } from '../provider-interface'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; +import Parameters from '../../../parameters.ts'; +import { CloudRunnerSystem } from '../../services/cloud-runner-system.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { ProviderInterface } from '../provider-interface.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; class LocalDockerCloudRunner implements ProviderInterface { cleanup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} setup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} public runTask( commands: string, buildGuid: string, - // eslint-disable-next-line no-unused-vars image: string, - // eslint-disable-next-line no-unused-vars mountdir: string, - // eslint-disable-next-line no-unused-vars workingdir: string, - // eslint-disable-next-line no-unused-vars environment: CloudRunnerEnvironmentVariable[], - // eslint-disable-next-line no-unused-vars secrets: CloudRunnerSecret[], ): Promise { CloudRunnerLogger.log(buildGuid); diff --git a/src/model/cloud-runner/providers/local/index.ts b/src/model/cloud-runner/providers/local/index.ts index 0980d6c76..d958624e4 100644 --- a/src/model/cloud-runner/providers/local/index.ts +++ b/src/model/cloud-runner/providers/local/index.ts @@ -1,42 +1,30 @@ -import BuildParameters from '../../../build-parameters'; -import { CloudRunnerSystem } from '../../services/cloud-runner-system'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { ProviderInterface } from '../provider-interface'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; +import Parameters from '../../../parameters.ts'; +import { CloudRunnerSystem } from '../../services/cloud-runner-system.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { ProviderInterface } from '../provider-interface.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; class LocalCloudRunner implements ProviderInterface { cleanup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} public setup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} public async runTask( buildGuid: string, image: string, commands: string, - // eslint-disable-next-line no-unused-vars mountdir: string, - // eslint-disable-next-line no-unused-vars workingdir: string, - // eslint-disable-next-line no-unused-vars environment: CloudRunnerEnvironmentVariable[], - // eslint-disable-next-line no-unused-vars secrets: CloudRunnerSecret[], ): Promise { CloudRunnerLogger.log(image); diff --git a/src/model/cloud-runner/providers/provider-interface.ts b/src/model/cloud-runner/providers/provider-interface.ts index f6e9d1715..a82b1f90f 100644 --- a/src/model/cloud-runner/providers/provider-interface.ts +++ b/src/model/cloud-runner/providers/provider-interface.ts @@ -1,42 +1,27 @@ -import BuildParameters from '../../build-parameters'; -import CloudRunnerEnvironmentVariable from '../services/cloud-runner-environment-variable'; -import CloudRunnerSecret from '../services/cloud-runner-secret'; +import Parameters from '../../parameters.ts'; +import CloudRunnerEnvironmentVariable from '../services/cloud-runner-environment-variable.ts'; +import CloudRunnerSecret from '../services/cloud-runner-secret.ts'; export interface ProviderInterface { cleanup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ); setup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ); runTask( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars image: string, - // eslint-disable-next-line no-unused-vars commands: string, - // eslint-disable-next-line no-unused-vars mountdir: string, - // eslint-disable-next-line no-unused-vars workingdir: string, - // eslint-disable-next-line no-unused-vars environment: CloudRunnerEnvironmentVariable[], - // eslint-disable-next-line no-unused-vars secrets: CloudRunnerSecret[], ): Promise; } diff --git a/src/model/cloud-runner/providers/test/index.ts b/src/model/cloud-runner/providers/test/index.ts index f7cb9630b..c4325fff1 100644 --- a/src/model/cloud-runner/providers/test/index.ts +++ b/src/model/cloud-runner/providers/test/index.ts @@ -1,41 +1,29 @@ -import BuildParameters from '../../../build-parameters'; -import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable'; -import CloudRunnerLogger from '../../services/cloud-runner-logger'; -import { ProviderInterface } from '../provider-interface'; -import CloudRunnerSecret from '../../services/cloud-runner-secret'; +import Parameters from '../../../parameters.ts'; +import CloudRunnerEnvironmentVariable from '../../services/cloud-runner-environment-variable.ts'; +import CloudRunnerLogger from '../../services/cloud-runner-logger.ts'; +import { ProviderInterface } from '../provider-interface.ts'; +import CloudRunnerSecret from '../../services/cloud-runner-secret.ts'; class TestCloudRunner implements ProviderInterface { cleanup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} setup( - // eslint-disable-next-line no-unused-vars buildGuid: string, - // eslint-disable-next-line no-unused-vars - buildParameters: BuildParameters, - // eslint-disable-next-line no-unused-vars + buildParameters: Parameters, branchName: string, - // eslint-disable-next-line no-unused-vars defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], ) {} public async runTask( commands: string, buildGuid: string, image: string, - // eslint-disable-next-line no-unused-vars mountdir: string, - // eslint-disable-next-line no-unused-vars workingdir: string, - // eslint-disable-next-line no-unused-vars environment: CloudRunnerEnvironmentVariable[], - // eslint-disable-next-line no-unused-vars secrets: CloudRunnerSecret[], ): Promise { CloudRunnerLogger.log(image); diff --git a/src/model/cloud-runner/remote-client/caching.test.ts b/src/model/cloud-runner/remote-client/caching.test.ts index 1da3eb1dc..7f7d86c11 100644 --- a/src/model/cloud-runner/remote-client/caching.test.ts +++ b/src/model/cloud-runner/remote-client/caching.test.ts @@ -1,13 +1,11 @@ -import fs from 'fs'; -import path from 'path'; -import BuildParameters from '../../build-parameters'; -import { Cli } from '../../cli/cli'; -import Input from '../../input'; -import UnityVersioning from '../../unity-versioning'; -import CloudRunner from '../cloud-runner'; -import { CloudRunnerSystem } from '../services/cloud-runner-system'; -import { Caching } from './caching'; -import { v4 as uuidv4 } from 'uuid'; +import { fs, uuid, path, __dirname } from '../../../dependencies.ts'; +import Parameters from '../../parameters.ts'; +import { Cli } from '../../cli/cli.ts'; +import Input from '../../input.ts'; +import UnityVersionDetector from '../../../middleware/engine-detection/unity-version-detector.ts'; +import CloudRunner from '../cloud-runner.ts'; +import { CloudRunnerSystem } from '../services/cloud-runner-system/index.ts'; +import { Caching } from './caching.ts'; describe('Cloud Runner Caching', () => { it('responds', () => {}); @@ -18,12 +16,12 @@ describe('Cloud Runner Caching', () => { Cli.options = { versioning: 'None', projectPath: 'test-project', - unityVersion: UnityVersioning.read('test-project'), + engineVersion: UnityVersionDetector.read('test-project'), targetPlatform: 'StandaloneLinux64', - cacheKey: `test-case-${uuidv4()}`, + cacheKey: `test-case-${uuid()}`, }; Input.githubInputEnabled = false; - const buildParameter = await BuildParameters.create(); + const buildParameter = await Parameters.create(); CloudRunner.buildParameters = buildParameter; // Create test folder @@ -50,7 +48,7 @@ describe('Cloud Runner Caching', () => { await CloudRunnerSystem.Run(`tree ${cacheFolder}`); // Compare validity to original hash - expect(fs.readFileSync(path.resolve(testFolder, 'test.txt'), { encoding: 'utf8' }).toString()).toContain( + expect(Deno.readTextFileSync(path.resolve(testFolder, 'test.txt'), { encoding: 'utf8' }).toString()).toContain( Cli.options.cacheKey, ); fs.rmdirSync(testFolder, { recursive: true }); @@ -58,6 +56,6 @@ describe('Cloud Runner Caching', () => { Input.githubInputEnabled = true; delete Cli.options; - }, 1000000); + }, 1_000_000); } }); diff --git a/src/model/cloud-runner/remote-client/caching.ts b/src/model/cloud-runner/remote-client/caching.ts index 19cdbf03d..9ae5d7fa7 100644 --- a/src/model/cloud-runner/remote-client/caching.ts +++ b/src/model/cloud-runner/remote-client/caching.ts @@ -1,14 +1,12 @@ -import { assert } from 'console'; -import fs from 'fs'; -import path from 'path'; -import CloudRunner from '../cloud-runner'; -import CloudRunnerLogger from '../services/cloud-runner-logger'; -import { CloudRunnerFolders } from '../services/cloud-runner-folders'; -import { CloudRunnerSystem } from '../services/cloud-runner-system'; -import { LfsHashing } from '../services/lfs-hashing'; -import { RemoteClientLogger } from './remote-client-logger'; -import { Cli } from '../../cli/cli'; -import { CliFunction } from '../../cli/cli-functions-repository'; +import { fs, path, assert } from '../../../dependencies.ts'; +import CloudRunner from '../cloud-runner.ts'; +import CloudRunnerLogger from '../services/cloud-runner-logger.ts'; +import { CloudRunnerFolders } from '../services/cloud-runner-folders.ts'; +import { CloudRunnerSystem } from '../services/cloud-runner-system.ts'; +import { LfsHashing } from '../services/lfs-hashing.ts'; +import { RemoteClientLogger } from './remote-client-logger.ts'; +import { Cli } from '../../cli/cli.ts'; +import { CliFunction } from '../../cli/cli-functions-repository.ts'; // eslint-disable-next-line github/no-then const fileExists = async (fpath) => !!(await fs.promises.stat(fpath).catch(() => false)); @@ -16,7 +14,7 @@ export class Caching { @CliFunction(`cache-push`, `push to cache`) static async cachePush() { try { - const buildParameter = JSON.parse(process.env.BUILD_PARAMETERS || '{}'); + const buildParameter = JSON.parse(Deno.env.get('BUILD_PARAMETERS') || '{}'); CloudRunner.buildParameters = buildParameter; await Caching.PushToCache( Cli.options['cachePushTo'], @@ -31,7 +29,7 @@ export class Caching { @CliFunction(`cache-pull`, `pull from cache`) static async cachePull() { try { - const buildParameter = JSON.parse(process.env.BUILD_PARAMETERS || '{}'); + const buildParameter = JSON.parse(Deno.env.get('BUILD_PARAMETERS') || '{}'); CloudRunner.buildParameters = buildParameter; await Caching.PullFromCache( Cli.options['cachePushFrom'], @@ -90,6 +88,7 @@ export class Caching { } public static async PullFromCache(cacheFolder: string, destinationFolder: string, cacheArtifactName: string = ``) { cacheArtifactName = cacheArtifactName.replace(' ', ''); + const startPath = process.cwd(); RemoteClientLogger.log(`Caching for ${path.basename(destinationFolder)}`); try { @@ -101,18 +100,15 @@ export class Caching { await fs.promises.mkdir(destinationFolder); } - const latestInBranch = await (await CloudRunnerSystem.Run(`ls -t "${cacheFolder}" | grep .tar$ | head -1`)) - .replace(/\n/g, ``) - .replace('.tar', ''); + const latestInBranchRaw = await CloudRunnerSystem.Run(`ls -t "${cacheFolder}" | grep .tar$ | head -1`); + const latestInBranch = latestInBranchRaw.replace(/\n/g, ``).replace('.tar', ''); process.chdir(cacheFolder); - const cacheSelection = cacheArtifactName !== `` && (await fileExists(`${cacheArtifactName}.tar`)) ? cacheArtifactName : latestInBranch; await CloudRunnerLogger.log(`cache key ${cacheArtifactName} selection ${cacheSelection}`); - // eslint-disable-next-line func-style - const formatFunction = function (format: string) { + const formatFunction = (format: string) => { const arguments_ = Array.prototype.slice.call( [path.resolve(destinationFolder, '..'), cacheFolder, cacheArtifactName], 1, @@ -164,7 +160,7 @@ export class Caching { } public static async handleCachePurging() { - if (process.env.PURGE_REMOTE_BUILDER_CACHE !== undefined) { + if (Deno.env.get('PURGE_REMOTE_BUILDER_CACHE') !== undefined) { RemoteClientLogger.log(`purging ${CloudRunnerFolders.purgeRemoteCaching}`); fs.promises.rmdir(CloudRunnerFolders.cacheFolder, { recursive: true }); } diff --git a/src/model/cloud-runner/remote-client/index.ts b/src/model/cloud-runner/remote-client/index.ts index 39cfcf702..2b971b78b 100644 --- a/src/model/cloud-runner/remote-client/index.ts +++ b/src/model/cloud-runner/remote-client/index.ts @@ -1,14 +1,12 @@ -import fs from 'fs'; -import CloudRunner from '../cloud-runner'; -import { CloudRunnerFolders } from '../services/cloud-runner-folders'; -import { Caching } from './caching'; -import { LfsHashing } from '../services/lfs-hashing'; -import { RemoteClientLogger } from './remote-client-logger'; -import path from 'path'; -import { assert } from 'console'; -import CloudRunnerLogger from '../services/cloud-runner-logger'; -import { CliFunction } from '../../cli/cli-functions-repository'; -import { CloudRunnerSystem } from '../services/cloud-runner-system'; +import { fsSync as fs, assert, path } from '../../../dependencies.ts'; +import CloudRunner from '../cloud-runner.ts'; +import { CloudRunnerFolders } from '../services/cloud-runner-folders.ts'; +import { Caching } from './caching.ts'; +import { LfsHashing } from '../services/lfs-hashing.ts'; +import { RemoteClientLogger } from './remote-client-logger.ts'; +import CloudRunnerLogger from '../services/cloud-runner-logger.ts'; +import { CliFunction } from '../../cli/cli-functions-repository.ts'; +import { CloudRunnerSystem } from '../services/cloud-runner-system.ts'; export class RemoteClient { public static async bootstrapRepository() { @@ -70,7 +68,7 @@ export class RemoteClient { RemoteClientLogger.log(`${CloudRunner.buildParameters.branch}`); await CloudRunnerSystem.Run(`git checkout ${CloudRunner.buildParameters.branch}`); assert(fs.existsSync(path.join(`.git`, `lfs`)), 'LFS folder should not exist before caching'); - RemoteClientLogger.log(`Checked out ${process.env.GITHUB_SHA}`); + RemoteClientLogger.log(`Checked out ${Deno.env.get('GITHUB_SHA')}`); } catch (error) { throw error; } @@ -87,7 +85,7 @@ export class RemoteClient { @CliFunction(`remote-cli`, `sets up a repository, usually before a game-ci build`) static async runRemoteClientJob() { - const buildParameter = JSON.parse(process.env.BUILD_PARAMETERS || '{}'); + const buildParameter = JSON.parse(Deno.env.get('BUILD_PARAMETERS') || '{}'); RemoteClientLogger.log(`Build Params: ${JSON.stringify(buildParameter, undefined, 4)} `); diff --git a/src/model/cloud-runner/remote-client/remote-client-logger.ts b/src/model/cloud-runner/remote-client/remote-client-logger.ts index 5581b8bc1..2c29d7e02 100644 --- a/src/model/cloud-runner/remote-client/remote-client-logger.ts +++ b/src/model/cloud-runner/remote-client/remote-client-logger.ts @@ -1,4 +1,4 @@ -import CloudRunnerLogger from '../services/cloud-runner-logger'; +import CloudRunnerLogger from '../services/cloud-runner-logger.ts'; export class RemoteClientLogger { public static log(message: string) { diff --git a/src/model/cloud-runner/services/cloud-runner-build-command-process.ts b/src/model/cloud-runner/services/cloud-runner-build-command-process.ts index 4428d37a2..1358efad5 100644 --- a/src/model/cloud-runner/services/cloud-runner-build-command-process.ts +++ b/src/model/cloud-runner/services/cloud-runner-build-command-process.ts @@ -1,10 +1,10 @@ -import { BuildParameters } from '../..'; -import YAML from 'yaml'; -import CloudRunnerSecret from './cloud-runner-secret'; -import CloudRunner from '../cloud-runner'; +import { Parameters } from '../../index.ts'; +import { yaml } from '../../../dependencies.ts'; +import CloudRunnerSecret from './cloud-runner-secret.ts'; +import CloudRunner from '../cloud-runner.ts'; export class CloudRunnerBuildCommandProcessor { - public static ProcessCommands(commands: string, buildParameters: BuildParameters): string { + public static ProcessCommands(commands: string, buildParameters: Parameters): string { const hooks = CloudRunnerBuildCommandProcessor.getHooks(buildParameters.customJobHooks).filter((x) => x.step.includes(`all`), ); @@ -25,7 +25,7 @@ export class CloudRunnerBuildCommandProcessor { let output = new Array(); if (experimentHooks && experimentHooks !== '') { try { - output = YAML.parse(experimentHooks); + output = yaml.parse(experimentHooks); } catch (error) { throw error; } diff --git a/src/model/cloud-runner/services/cloud-runner-folders.ts b/src/model/cloud-runner/services/cloud-runner-folders.ts index 30cb4a6d2..de4bef622 100644 --- a/src/model/cloud-runner/services/cloud-runner-folders.ts +++ b/src/model/cloud-runner/services/cloud-runner-folders.ts @@ -1,5 +1,5 @@ -import path from 'path'; -import { CloudRunner } from '../..'; +import { path } from '../../../dependencies.ts'; +import { CloudRunner } from '../../index.ts'; export class CloudRunnerFolders { public static readonly repositoryFolder = 'repo'; @@ -44,7 +44,7 @@ export class CloudRunnerFolders { } public static get purgeRemoteCaching(): boolean { - return process.env.PURGE_REMOTE_BUILDER_CACHE !== undefined; + return Deno.env.get('PURGE_REMOTE_BUILDER_CACHE') !== undefined; } public static get lfsCacheFolderFull() { diff --git a/src/model/cloud-runner/services/cloud-runner-guid.ts b/src/model/cloud-runner/services/cloud-runner-guid.ts index 1ee760136..a24a76964 100644 --- a/src/model/cloud-runner/services/cloud-runner-guid.ts +++ b/src/model/cloud-runner/services/cloud-runner-guid.ts @@ -1,9 +1,9 @@ -import { customAlphabet } from 'nanoid'; -import CloudRunnerConstants from './cloud-runner-constants'; +import { nanoid } from '../../../dependencies.ts'; +import CloudRunnerConstants from './cloud-runner-constants.ts'; class CloudRunnerNamespace { static generateGuid(runNumber: string | number, platform: string) { - const nanoid = customAlphabet(CloudRunnerConstants.alphabet, 4); + const nanoid = nanoid.customAlphabet(CloudRunnerConstants.alphabet, 4); return `${runNumber}-${platform.toLowerCase().replace('standalone', '')}-${nanoid()}`; } diff --git a/src/model/cloud-runner/services/cloud-runner-logger.ts b/src/model/cloud-runner/services/cloud-runner-logger.ts index 53158843b..92c1382f7 100644 --- a/src/model/cloud-runner/services/cloud-runner-logger.ts +++ b/src/model/cloud-runner/services/cloud-runner-logger.ts @@ -1,4 +1,4 @@ -import * as core from '@actions/core'; +import { core } from '../../../dependencies.ts'; class CloudRunnerLogger { private static timestamp: number; @@ -10,24 +10,24 @@ class CloudRunnerLogger { } public static log(message: string) { - core.info(message); + log.info(message); } public static logWarning(message: string) { - core.warning(message); + log.warning(message); } public static logLine(message: string) { - core.info(`${message}\n`); + log.info(`${message}\n`); } public static error(message: string) { - core.error(message); + log.error(message); } public static logWithTime(message: string) { const newTimestamp = this.createTimestamp(); - core.info( + log.info( `${message} (Since previous: ${this.calculateTimeDiff( newTimestamp, this.timestamp, diff --git a/src/model/cloud-runner/services/cloud-runner-query-override.ts b/src/model/cloud-runner/services/cloud-runner-query-override.ts index 750f2bd5d..28fb3a2cd 100644 --- a/src/model/cloud-runner/services/cloud-runner-query-override.ts +++ b/src/model/cloud-runner/services/cloud-runner-query-override.ts @@ -1,5 +1,5 @@ -import Input from '../../input'; -import { GenericInputReader } from '../../input-readers/generic-input-reader'; +import Input from '../../input.ts'; +import { GenericInputReader } from '../../input-readers/generic-input-reader.ts'; const formatFunction = (value, arguments_) => { for (const element of arguments_) { @@ -32,7 +32,7 @@ class CloudRunnerQueryOverride { if (Input.readInputFromOverrideList() !== '') { const doesInclude = Input.readInputFromOverrideList().split(',').includes(query) || - Input.readInputFromOverrideList().split(',').includes(Input.ToEnvVarFormat(query)); + Input.readInputFromOverrideList().split(',').includes(Input.toEnvVarFormat(query)); return doesInclude ? true : false; } else { diff --git a/src/model/cloud-runner/services/cloud-runner-system.ts b/src/model/cloud-runner/services/cloud-runner-system.ts index 0545e607f..1ced6c9de 100644 --- a/src/model/cloud-runner/services/cloud-runner-system.ts +++ b/src/model/cloud-runner/services/cloud-runner-system.ts @@ -1,5 +1,5 @@ -import { exec } from 'child_process'; -import { RemoteClientLogger } from '../remote-client/remote-client-logger'; +import { RemoteClientLogger } from '../remote-client/remote-client-logger.ts'; +import { spawn } from 'https://deno.land/std@0.152.0/node/child_process.ts'; export class CloudRunnerSystem { public static async Run(command: string, suppressError = false, suppressLogs = false) { @@ -11,7 +11,9 @@ export class CloudRunnerSystem { return await new Promise((promise, throwError) => { let output = ''; - const child = exec(command, (error, stdout, stderr) => { + + // Todo - find Deno variant of child_process or rewrite this method to use execSync + const child = spawn(command, (error, stdout, stderr) => { if (!suppressError && error) { RemoteClientLogger.log(error.toString()); throwError(error); diff --git a/src/model/cloud-runner/services/depdency-override-service.ts b/src/model/cloud-runner/services/depdency-override-service.ts index 83d06277b..35fbfc06d 100644 --- a/src/model/cloud-runner/services/depdency-override-service.ts +++ b/src/model/cloud-runner/services/depdency-override-service.ts @@ -1,5 +1,5 @@ -import Input from '../../input'; -import { CloudRunnerSystem } from './cloud-runner-system'; +import Input from '../../input.ts'; +import { CloudRunnerSystem } from './cloud-runner-system.ts'; class DependencyOverrideService { public static async CheckHealth() { diff --git a/src/model/cloud-runner/services/follow-log-stream-service.ts b/src/model/cloud-runner/services/follow-log-stream-service.ts index 59efc9b87..357590fa3 100644 --- a/src/model/cloud-runner/services/follow-log-stream-service.ts +++ b/src/model/cloud-runner/services/follow-log-stream-service.ts @@ -1,7 +1,7 @@ -import CloudRunnerLogger from './cloud-runner-logger'; -import * as core from '@actions/core'; -import CloudRunner from '../cloud-runner'; -import { CloudRunnerStatics } from '../cloud-runner-statics'; +import CloudRunnerLogger from './cloud-runner-logger.ts'; +import { core } from '../../../dependencies.ts'; +import CloudRunner from '../cloud-runner.ts'; +import { CloudRunnerStatics } from '../cloud-runner-statics.ts'; export class FollowLogStreamService { public static handleIteration(message, shouldReadLogs, shouldCleanup, output) { @@ -9,19 +9,19 @@ export class FollowLogStreamService { CloudRunnerLogger.log('End of log transmission received'); shouldReadLogs = false; } else if (message.includes('Rebuilding Library because the asset database could not be found!')) { - core.warning('LIBRARY NOT FOUND!'); + log.warning('LIBRARY NOT FOUND!'); core.setOutput('library-found', 'false'); } else if (message.includes('Build succeeded')) { core.setOutput('build-result', 'success'); } else if (message.includes('Build fail')) { core.setOutput('build-result', 'failed'); - core.setFailed('unity build failed'); - core.error('BUILD FAILED!'); + log.error('build-result: failed'); + Deno.exit(1); } else if (CloudRunner.buildParameters.cloudRunnerIntegrationTests && message.includes(': Listening for Jobs')) { core.setOutput('cloud runner stop watching', 'true'); shouldReadLogs = false; shouldCleanup = false; - core.warning('cloud runner stop watching'); + log.warning('cloud runner stop watching'); } message = `[${CloudRunnerStatics.logPrefix}] ${message}`; if (CloudRunner.buildParameters.cloudRunnerIntegrationTests) { diff --git a/src/model/cloud-runner/services/lfs-hashing.ts b/src/model/cloud-runner/services/lfs-hashing.ts index 9d43130ad..856427699 100644 --- a/src/model/cloud-runner/services/lfs-hashing.ts +++ b/src/model/cloud-runner/services/lfs-hashing.ts @@ -1,10 +1,8 @@ -import path from 'path'; -import { CloudRunnerFolders } from './cloud-runner-folders'; -import { CloudRunnerSystem } from './cloud-runner-system'; -import fs from 'fs'; -import { assert } from 'console'; -import { Cli } from '../../cli/cli'; -import { CliFunction } from '../../cli/cli-functions-repository'; +import { CloudRunnerFolders } from './cloud-runner-folders.ts'; +import { CloudRunnerSystem } from './cloud-runner-system.ts'; +import { fsSync as fs, assert, path } from '../../../dependencies.ts'; +import { Cli } from '../../cli/cli.ts'; +import { CliFunction } from '../../cli/cli-functions-repository.ts'; export class LfsHashing { public static async createLFSHashFiles() { @@ -30,10 +28,11 @@ export class LfsHashing { } public static async hashAllFiles(folder: string) { const startPath = process.cwd(); + process.chdir(folder); - const result = await (await CloudRunnerSystem.Run(`find -type f -exec md5sum "{}" + | sort | md5sum`)) - .replace(/\n/g, '') - .split(` `)[0]; + const checksums = await CloudRunnerSystem.Run(`find -type f -exec md5sum "{}" + | sort | md5sum`); + const result = checksums.replace(/\n/g, '').split(` `)[0]; + process.chdir(startPath); return result; diff --git a/src/model/cloud-runner/services/task-parameter-serializer.ts b/src/model/cloud-runner/services/task-parameter-serializer.ts index 9bdf4170f..55fb0a7b3 100644 --- a/src/model/cloud-runner/services/task-parameter-serializer.ts +++ b/src/model/cloud-runner/services/task-parameter-serializer.ts @@ -1,9 +1,9 @@ -import { CloudRunner, Input } from '../..'; -import ImageEnvironmentFactory from '../../image-environment-factory'; -import CloudRunnerEnvironmentVariable from './cloud-runner-environment-variable'; -import { CloudRunnerBuildCommandProcessor } from './cloud-runner-build-command-process'; -import CloudRunnerSecret from './cloud-runner-secret'; -import CloudRunnerQueryOverride from './cloud-runner-query-override'; +import { CloudRunner, Input } from '../../index.ts'; +import ImageEnvironmentFactory from '../../image-environment-factory.ts'; +import CloudRunnerEnvironmentVariable from './cloud-runner-environment-variable.ts'; +import { CloudRunnerBuildCommandProcessor } from './cloud-runner-build-command-process.ts'; +import CloudRunnerSecret from './cloud-runner-secret.ts'; +import CloudRunnerQueryOverride from './cloud-runner-query-override.ts'; export class TaskParameterSerializer { public static readBuildEnvironmentVariables(): CloudRunnerEnvironmentVariable[] { @@ -38,7 +38,7 @@ export class TaskParameterSerializer { (x) => x.value !== undefined && x.name !== '0' && x.value !== '' && x.name !== 'prototype' && x.name !== 'length', ); array = array.map((x) => { - x.name = Input.ToEnvVarFormat(x.name); + x.name = Input.toEnvVarFormat(x.name); x.value = `${x.value}`; return x; diff --git a/src/model/cloud-runner/workflows/build-automation-workflow.ts b/src/model/cloud-runner/workflows/build-automation-workflow.ts index ac02ef9a7..089997f91 100644 --- a/src/model/cloud-runner/workflows/build-automation-workflow.ts +++ b/src/model/cloud-runner/workflows/build-automation-workflow.ts @@ -1,12 +1,11 @@ -import CloudRunnerLogger from '../services/cloud-runner-logger'; -import { CloudRunnerFolders } from '../services/cloud-runner-folders'; -import { CloudRunnerStepState } from '../cloud-runner-step-state'; -import { CustomWorkflow } from './custom-workflow'; -import { WorkflowInterface } from './workflow-interface'; -import * as core from '@actions/core'; -import { CloudRunnerBuildCommandProcessor } from '../services/cloud-runner-build-command-process'; -import path from 'path'; -import CloudRunner from '../cloud-runner'; +import CloudRunnerLogger from '../services/cloud-runner-logger.ts'; +import { CloudRunnerFolders } from '../services/cloud-runner-folders.ts'; +import { CloudRunnerStepState } from '../cloud-runner-step-state.ts'; +import { CustomWorkflow } from './custom-workflow.ts'; +import { WorkflowInterface } from './workflow-interface.ts'; +import { core, path } from '../../../dependencies.ts'; +import { CloudRunnerBuildCommandProcessor } from '../services/cloud-runner-build-command-process.ts'; +import CloudRunner from '../cloud-runner.ts'; export class BuildAutomationWorkflow implements WorkflowInterface { async run(cloudRunnerStepState: CloudRunnerStepState) { diff --git a/src/model/cloud-runner/workflows/custom-workflow.ts b/src/model/cloud-runner/workflows/custom-workflow.ts index 52bbf44d1..056e2d9c9 100644 --- a/src/model/cloud-runner/workflows/custom-workflow.ts +++ b/src/model/cloud-runner/workflows/custom-workflow.ts @@ -1,8 +1,8 @@ -import CloudRunnerLogger from '../services/cloud-runner-logger'; -import CloudRunnerSecret from '../services/cloud-runner-secret'; -import { CloudRunnerFolders } from '../services/cloud-runner-folders'; -import YAML from 'yaml'; -import { CloudRunner, Input } from '../..'; +import CloudRunnerLogger from '../services/cloud-runner-logger.ts'; +import CloudRunnerSecret from '../services/cloud-runner-secret.ts'; +import { CloudRunnerFolders } from '../services/cloud-runner-folders.ts'; +import { yaml } from '../../../dependencies.ts'; +import { CloudRunner, Input } from '../../index.ts'; export class CustomWorkflow { public static async runCustomJob(buildSteps) { @@ -12,7 +12,7 @@ export class CustomWorkflow { CloudRunnerLogger.log(`Parsing build steps: ${buildSteps}`); } try { - buildSteps = YAML.parse(buildSteps); + buildSteps = yaml.parse(buildSteps); } catch (error) { CloudRunnerLogger.log(`failed to parse a custom job "${buildSteps}"`); throw error; @@ -22,7 +22,7 @@ export class CustomWorkflow { const stepSecrets: CloudRunnerSecret[] = step.secrets.map((x) => { const secret: CloudRunnerSecret = { ParameterKey: x.name, - EnvironmentVariable: Input.ToEnvVarFormat(x.name), + EnvironmentVariable: Input.toEnvVarFormat(x.name), ParameterValue: x.value, }; diff --git a/src/model/cloud-runner/workflows/workflow-composition-root.ts b/src/model/cloud-runner/workflows/workflow-composition-root.ts index cb0522003..47fcbe8ea 100644 --- a/src/model/cloud-runner/workflows/workflow-composition-root.ts +++ b/src/model/cloud-runner/workflows/workflow-composition-root.ts @@ -1,8 +1,8 @@ -import { CloudRunnerStepState } from '../cloud-runner-step-state'; -import { CustomWorkflow } from './custom-workflow'; -import { WorkflowInterface } from './workflow-interface'; -import { BuildAutomationWorkflow } from './build-automation-workflow'; -import CloudRunner from '../cloud-runner'; +import { CloudRunnerStepState } from '../cloud-runner-step-state.ts'; +import { CustomWorkflow } from './custom-workflow.ts'; +import { WorkflowInterface } from './workflow-interface.ts'; +import { BuildAutomationWorkflow } from './build-automation-workflow.ts'; +import CloudRunner from '../cloud-runner.ts'; export class WorkflowCompositionRoot implements WorkflowInterface { async run(cloudRunnerStepState: CloudRunnerStepState) { diff --git a/src/model/cloud-runner/workflows/workflow-interface.ts b/src/model/cloud-runner/workflows/workflow-interface.ts index 3e8763edc..68bda864b 100644 --- a/src/model/cloud-runner/workflows/workflow-interface.ts +++ b/src/model/cloud-runner/workflows/workflow-interface.ts @@ -1,8 +1,5 @@ -import { CloudRunnerStepState } from '../cloud-runner-step-state'; +import { CloudRunnerStepState } from '../cloud-runner-step-state.ts'; export interface WorkflowInterface { - run( - // eslint-disable-next-line no-unused-vars - cloudRunnerStepState: CloudRunnerStepState, - ); + run(cloudRunnerStepState: CloudRunnerStepState); } diff --git a/src/model/docker.test.ts b/src/model/docker.test.ts index 290d19bf1..f9ed66f24 100644 --- a/src/model/docker.test.ts +++ b/src/model/docker.test.ts @@ -1,5 +1,5 @@ -import Action from './action'; -import Docker from './docker'; +import Action from './action.ts'; +import Docker from './docker.ts'; describe('Docker', () => { it.skip('runs', async () => { diff --git a/src/model/docker.ts b/src/model/docker.ts index 497b3f051..2e64574d1 100644 --- a/src/model/docker.ts +++ b/src/model/docker.ts @@ -1,69 +1,100 @@ -import { exec } from '@actions/exec'; -import ImageEnvironmentFactory from './image-environment-factory'; -import { existsSync, mkdirSync } from 'fs'; -import path from 'path'; +import ImageEnvironmentFactory from './image-environment-factory.ts'; +import { path, fsSync as fs } from '../dependencies.ts'; +import System from './system/system.ts'; class Docker { - static async run(image, parameters, silent = false) { - let runCommand = ''; - switch (process.platform) { + static async run(image, parameters) { + log.warning('running docker process for', process.platform); + let command = ''; + switch (Deno.build.os) { + case 'windows': { + // Todo: check if docker daemon is set for Windows or Linux containers. + command = await this.getWindowsCommand(image, parameters); + break; + } case 'linux': - runCommand = this.getLinuxCommand(image, parameters); + case 'darwin': { + command = await this.getLinuxCommand(image, parameters); break; - case 'win32': - runCommand = this.getWindowsCommand(image, parameters); + } + } + + try { + const test = await System.run(command, { attach: true }); + log.warning('test', test); + } catch (error) { + if (error.message.includes('docker: image operating system "windows" cannot be used on this platform')) { + throw new Error(String.dedent` + Docker daemon is not set to run Windows containers. + + To enable the Hyper-V container backend run: + Enable-WindowsOptionalFeature -Online -FeatureName $("Microsoft-Hyper-V", "Containers") -All + + To switch the docker daemon to run Windows containers run: + & $Env:ProgramFiles\\Docker\\Docker\\DockerCli.exe -SwitchDaemon . + + For more information see: + https://docs.microsoft.com/en-us/virtualization/windowscontainers/quick-start/set-up-environment?tabs=dockerce#prerequisites + `); + } + + throw error; } - await exec(runCommand, undefined, { silent }); } - static getLinuxCommand(image, parameters): string { + static async getLinuxCommand(image, parameters): string { const { workspace, actionFolder, runnerTempPath, sshAgent, gitPrivateToken } = parameters; const githubHome = path.join(runnerTempPath, '_github_home'); - if (!existsSync(githubHome)) mkdirSync(githubHome); + await fs.ensureDir(githubHome); const githubWorkflow = path.join(runnerTempPath, '_github_workflow'); - if (!existsSync(githubWorkflow)) mkdirSync(githubWorkflow); + await fs.ensureDir(githubWorkflow); - return `docker run \ - --workdir /github/workspace \ - --rm \ - ${ImageEnvironmentFactory.getEnvVarString(parameters)} \ - --env UNITY_SERIAL \ - --env GITHUB_WORKSPACE=/github/workspace \ - ${gitPrivateToken ? `--env GIT_PRIVATE_TOKEN="${gitPrivateToken}"` : ''} \ - ${sshAgent ? '--env SSH_AUTH_SOCK=/ssh-agent' : ''} \ - --volume "${githubHome}":"/root:z" \ - --volume "${githubWorkflow}":"/github/workflow:z" \ - --volume "${workspace}":"/github/workspace:z" \ - --volume "${actionFolder}/default-build-script:/UnityBuilderAction:z" \ - --volume "${actionFolder}/platforms/ubuntu/steps:/steps:z" \ - --volume "${actionFolder}/platforms/ubuntu/entrypoint.sh:/entrypoint.sh:z" \ - ${sshAgent ? `--volume ${sshAgent}:/ssh-agent` : ''} \ - ${sshAgent ? '--volume /home/runner/.ssh/known_hosts:/root/.ssh/known_hosts:ro' : ''} \ - ${image} \ - /bin/bash -c /entrypoint.sh`; + return String.dedent` + docker run \ + --rm \ + --workdir /github/workspace \ + ${ImageEnvironmentFactory.getEnvVarString(parameters)} \ + --env UNITY_SERIAL \ + --env GITHUB_WORKSPACE=/github/workspace \ + ${gitPrivateToken ? `--env GIT_PRIVATE_TOKEN="${gitPrivateToken}"` : ''} \ + ${sshAgent ? '--env SSH_AUTH_SOCK=/ssh-agent' : ''} \ + --volume "${githubHome}":"/root:z" \ + --volume "${githubWorkflow}":"/github/workflow:z" \ + --volume "${workspace}":"/github/workspace:z" \ + --volume "${actionFolder}/default-build-script:/UnityBuilderAction:z" \ + --volume "${actionFolder}/platforms/ubuntu/steps:/steps:z" \ + --volume "${actionFolder}/platforms/ubuntu/entrypoint.sh:/entrypoint.sh:z" \ + ${sshAgent ? `--volume ${sshAgent}:/ssh-agent` : ''} \ + ${sshAgent ? '--volume /home/runner/.ssh/known_hosts:/root/.ssh/known_hosts:ro' : ''} \ + ${image} \ + /bin/bash -c /entrypoint.sh + `; } - static getWindowsCommand(image: any, parameters: any): string { - const { workspace, actionFolder, unitySerial, gitPrivateToken } = parameters; + static async getWindowsCommand(image: any, parameters: any): string { + const { workspace, actionFolder, unitySerial, gitPrivateToken, cliStoragePath } = parameters; - return `docker run \ - --workdir /github/workspace \ - --rm \ - ${ImageEnvironmentFactory.getEnvVarString(parameters)} \ - --env UNITY_SERIAL="${unitySerial}" \ - --env GITHUB_WORKSPACE=c:/github/workspace \ - ${gitPrivateToken ? `--env GIT_PRIVATE_TOKEN="${gitPrivateToken}"` : ''} \ - --volume "${workspace}":"c:/github/workspace" \ - --volume "c:/regkeys":"c:/regkeys" \ - --volume "C:/Program Files (x86)/Microsoft Visual Studio":"C:/Program Files (x86)/Microsoft Visual Studio" \ - --volume "C:/Program Files (x86)/Windows Kits":"C:/Program Files (x86)/Windows Kits" \ - --volume "C:/ProgramData/Microsoft/VisualStudio":"C:/ProgramData/Microsoft/VisualStudio" \ - --volume "${actionFolder}/default-build-script":"c:/UnityBuilderAction" \ - --volume "${actionFolder}/platforms/windows":"c:/steps" \ - --volume "${actionFolder}/BlankProject":"c:/BlankProject" \ - ${image} \ - powershell c:/steps/entrypoint.ps1`; + // Note: the equals sign (=) is needed in Powershell. + return String.dedent` + docker run \ + --rm \ + --workdir="c:/github/workspace" \ + ${ImageEnvironmentFactory.getEnvVarString(parameters)} \ + --env UNITY_SERIAL="${unitySerial}" \ + --env GITHUB_WORKSPACE=c:/github/workspace \ + --env GIT_PRIVATE_TOKEN="${gitPrivateToken}" \ + --volume="${workspace}":"c:/github/workspace" \ + --volume="${cliStoragePath}/registry-keys":"c:/registry-keys" \ + --volume="C:/Program Files (x86)/Microsoft Visual Studio":"C:/Program Files (x86)/Microsoft Visual Studio" \ + --volume="C:/Program Files (x86)/Windows Kits":"C:/Program Files (x86)/Windows Kits" \ + --volume="C:/ProgramData/Microsoft/VisualStudio":"C:/ProgramData/Microsoft/VisualStudio" \ + --volume="${actionFolder}/default-build-script":"c:/UnityBuilderAction" \ + --volume="${actionFolder}/platforms/windows":"c:/steps" \ + --volume="${actionFolder}/BlankProject":"c:/BlankProject" \ + ${image} \ + powershell c:/steps/entrypoint.ps1 + `; } } diff --git a/src/model/engine/engine.ts b/src/model/engine/engine.ts new file mode 100644 index 000000000..1d082d8b3 --- /dev/null +++ b/src/model/engine/engine.ts @@ -0,0 +1,3 @@ +export class Engine { + public static unity = 'unity'; +} diff --git a/src/model/error/command-execution-error.test.ts b/src/model/error/command-execution-error.test.ts index 6e8ac9f04..bb420daed 100644 --- a/src/model/error/command-execution-error.test.ts +++ b/src/model/error/command-execution-error.test.ts @@ -1,4 +1,4 @@ -import CommandExecutionError from './command-execution-error'; +import CommandExecutionError from './command-execution-error.ts'; describe('CommandExecutionError', () => { it('instantiates', () => { diff --git a/src/model/error/not-implemented-exception.test.ts b/src/model/error/not-implemented-exception.test.ts index 7c9c053d4..64e64b471 100644 --- a/src/model/error/not-implemented-exception.test.ts +++ b/src/model/error/not-implemented-exception.test.ts @@ -1,4 +1,4 @@ -import NotImplementedException from './not-implemented-exception'; +import NotImplementedException from './not-implemented-exception.ts'; describe('NotImplementedException', () => { it('instantiates', () => { diff --git a/src/model/error/validation-error.test.ts b/src/model/error/validation-error.test.ts index 365f0fcf4..a51bef9b3 100644 --- a/src/model/error/validation-error.test.ts +++ b/src/model/error/validation-error.test.ts @@ -1,4 +1,4 @@ -import ValidationError from './validation-error'; +import ValidationError from './validation-error.ts'; describe('ValidationError', () => { it('instantiates', () => { diff --git a/src/model/image-environment-factory.ts b/src/model/image-environment-factory.ts index 15513fda3..22736ed94 100644 --- a/src/model/image-environment-factory.ts +++ b/src/model/image-environment-factory.ts @@ -1,5 +1,5 @@ -import BuildParameters from './build-parameters'; -import { ReadLicense } from './input-readers/test-license-reader'; +import Parameters from './parameters.ts'; +import { ReadLicense } from './input-readers/test-license-reader.ts'; class Parameter { public name; @@ -19,20 +19,30 @@ class ImageEnvironmentFactory { continue; } - string += `--env ${p.name}="${p.value}" `; + if (Deno.build.os === 'windows') { + // The ampersand (&) character is not allowed. The & operator is reserved for future use; wrap an ampersand in + // double quotation marks ("&") to pass it as part of a string. + const escapedValue = typeof p.value !== 'string' ? p.value : p.value?.replace(/&/, '\\"&\\"'); + string += `--env ${p.name}='${escapedValue}' `; + } else { + string += `--env ${p.name}="${p.value}" `; + } } return string; } - public static getEnvironmentVariables(parameters: BuildParameters) { + public static getEnvironmentVariables(parameters: Parameters) { + // Todo - replace with simple for of loop, mapping parameters to this specific format + // All parameters should be straight forward at this point in the process. + // We can convert between camelCase and UPPER_SNAKE_CASE relatively easily. const environmentVariables: Parameter[] = [ - { name: 'UNITY_LICENSE', value: process.env.UNITY_LICENSE || ReadLicense() }, - { name: 'UNITY_LICENSE_FILE', value: process.env.UNITY_LICENSE_FILE }, - { name: 'UNITY_EMAIL', value: process.env.UNITY_EMAIL }, - { name: 'UNITY_PASSWORD', value: process.env.UNITY_PASSWORD }, + { name: 'UNITY_LICENSE', value: parameters.unityLicense || ReadLicense(parameters) }, + { name: 'UNITY_LICENSE_FILE', value: parameters.unityLicenseFile }, + { name: 'UNITY_EMAIL', value: parameters.unityEmail }, + { name: 'UNITY_PASSWORD', value: parameters.unityPassword }, { name: 'UNITY_SERIAL', value: parameters.unitySerial }, { name: 'UNITY_VERSION', value: parameters.editorVersion }, - { name: 'USYM_UPLOAD_AUTH_TOKEN', value: process.env.USYM_UPLOAD_AUTH_TOKEN }, + { name: 'USYM_UPLOAD_AUTH_TOKEN', value: parameters.uploadAuthToken }, { name: 'PROJECT_PATH', value: parameters.projectPath }, { name: 'BUILD_TARGET', value: parameters.targetPlatform }, { name: 'BUILD_NAME', value: parameters.buildName }, @@ -50,21 +60,21 @@ class ImageEnvironmentFactory { { name: 'ANDROID_SDK_MANAGER_PARAMETERS', value: parameters.androidSdkManagerParameters }, { name: 'CUSTOM_PARAMETERS', value: parameters.customParameters }, { name: 'CHOWN_FILES_TO', value: parameters.chownFilesTo }, - { name: 'GITHUB_REF', value: process.env.GITHUB_REF }, - { name: 'GITHUB_SHA', value: process.env.GITHUB_SHA }, - { name: 'GITHUB_REPOSITORY', value: process.env.GITHUB_REPOSITORY }, - { name: 'GITHUB_ACTOR', value: process.env.GITHUB_ACTOR }, - { name: 'GITHUB_WORKFLOW', value: process.env.GITHUB_WORKFLOW }, - { name: 'GITHUB_HEAD_REF', value: process.env.GITHUB_HEAD_REF }, - { name: 'GITHUB_BASE_REF', value: process.env.GITHUB_BASE_REF }, - { name: 'GITHUB_EVENT_NAME', value: process.env.GITHUB_EVENT_NAME }, + { name: 'GITHUB_REF', value: Deno.env.get('GITHUB_REF') }, + { name: 'GITHUB_SHA', value: Deno.env.get('GITHUB_SHA') }, + { name: 'GITHUB_REPOSITORY', value: Deno.env.get('GITHUB_REPOSITORY') }, + { name: 'GITHUB_ACTOR', value: Deno.env.get('GITHUB_ACTOR') }, + { name: 'GITHUB_WORKFLOW', value: Deno.env.get('GITHUB_WORKFLOW') }, + { name: 'GITHUB_HEAD_REF', value: Deno.env.get('GITHUB_HEAD_REF') }, + { name: 'GITHUB_BASE_REF', value: Deno.env.get('GITHUB_BASE_REF') }, + { name: 'GITHUB_EVENT_NAME', value: Deno.env.get('GITHUB_EVENT_NAME') }, { name: 'GITHUB_WORKSPACE', value: '/github/workspace' }, - { name: 'GITHUB_ACTION', value: process.env.GITHUB_ACTION }, - { name: 'GITHUB_EVENT_PATH', value: process.env.GITHUB_EVENT_PATH }, - { name: 'RUNNER_OS', value: process.env.RUNNER_OS }, - { name: 'RUNNER_TOOL_CACHE', value: process.env.RUNNER_TOOL_CACHE }, - { name: 'RUNNER_TEMP', value: process.env.RUNNER_TEMP }, - { name: 'RUNNER_WORKSPACE', value: process.env.RUNNER_WORKSPACE }, + { name: 'GITHUB_ACTION', value: Deno.env.get('GITHUB_ACTION') }, + { name: 'GITHUB_EVENT_PATH', value: Deno.env.get('GITHUB_EVENT_PATH') }, + { name: 'RUNNER_OS', value: Deno.env.get('RUNNER_OS') }, + { name: 'RUNNER_TOOL_CACHE', value: Deno.env.get('RUNNER_TOOL_CACHE') }, + { name: 'RUNNER_TEMP', value: Deno.env.get('RUNNER_TEMP') }, + { name: 'RUNNER_WORKSPACE', value: Deno.env.get('RUNNER_WORKSPACE') }, ]; if (parameters.sshAgent) environmentVariables.push({ name: 'SSH_AUTH_SOCK', value: '/ssh-agent' }); diff --git a/src/model/image-tag.test.ts b/src/model/image-tag.test.ts index f7ac2e2b3..f0d1e270a 100644 --- a/src/model/image-tag.test.ts +++ b/src/model/image-tag.test.ts @@ -1,4 +1,4 @@ -import ImageTag from './image-tag'; +import ImageTag from './image-tag.ts'; describe('ImageTag', () => { const some = { diff --git a/src/model/image-tag.ts b/src/model/image-tag.ts index 576e8d6e2..309c2e9af 100644 --- a/src/model/image-tag.ts +++ b/src/model/image-tag.ts @@ -1,6 +1,6 @@ -import Platform from './platform'; +import UnityTargetPlatform from './unity/unity-target-platform.ts'; -import BuildParameters from './build-parameters'; +import Parameters from './parameters.ts'; class ImageTag { public repository: string; @@ -13,7 +13,7 @@ class ImageTag { public imageRollingVersion: number; public imagePlatformPrefix: string; - constructor(imageProperties: Partial) { + constructor(imageProperties: Partial) { const { editorVersion = '2019.2.11f1', targetPlatform, customImage, cloudRunnerBuilderPlatform } = imageProperties; if (!ImageTag.versionPattern.test(editorVersion)) { @@ -74,14 +74,14 @@ class ImageTag { const { generic, webgl, mac, windows, windowsIl2cpp, wsaPlayer, linux, linuxIl2cpp, android, ios, tvos, facebook } = ImageTag.targetPlatformSuffixes; - const [major, minor] = version.split('.').map((digit) => Number(digit)); + const [major, minor] = version.split('.').map(Number); // @see: https://docs.unity3d.com/ScriptReference/BuildTarget.html switch (platform) { - case Platform.types.StandaloneOSX: + case UnityTargetPlatform.StandaloneOSX: return mac; - case Platform.types.StandaloneWindows: - case Platform.types.StandaloneWindows64: + case UnityTargetPlatform.StandaloneWindows: + case UnityTargetPlatform.StandaloneWindows64: // Can only build windows-il2cpp on a windows based system if (process.platform === 'win32') { // Unity versions before 2019.3 do not support il2cpp @@ -94,7 +94,7 @@ class ImageTag { } return windows; - case Platform.types.StandaloneLinux64: { + case UnityTargetPlatform.StandaloneLinux64: { // Unity versions before 2019.3 do not support il2cpp if (major >= 2020 || (major === 2019 && minor >= 3)) { return linuxIl2cpp; @@ -102,45 +102,45 @@ class ImageTag { return linux; } - case Platform.types.iOS: + case UnityTargetPlatform.iOS: return ios; - case Platform.types.Android: + case UnityTargetPlatform.Android: return android; - case Platform.types.WebGL: + case UnityTargetPlatform.WebGL: return webgl; - case Platform.types.WSAPlayer: + case UnityTargetPlatform.WSAPlayer: if (process.platform !== 'win32') { throw new Error(`WSAPlayer can only be built on a windows base OS`); } return wsaPlayer; - case Platform.types.PS4: + case UnityTargetPlatform.PS4: return windows; - case Platform.types.XboxOne: + case UnityTargetPlatform.XboxOne: return windows; - case Platform.types.tvOS: + case UnityTargetPlatform.tvOS: if (process.platform !== 'win32') { throw new Error(`tvOS can only be built on a windows base OS`); } return tvos; - case Platform.types.Switch: + case UnityTargetPlatform.Switch: return windows; // Unsupported - case Platform.types.Lumin: + case UnityTargetPlatform.Lumin: return windows; - case Platform.types.BJM: + case UnityTargetPlatform.BJM: return windows; - case Platform.types.Stadia: + case UnityTargetPlatform.Stadia: return windows; - case Platform.types.Facebook: + case UnityTargetPlatform.Facebook: return facebook; - case Platform.types.NoTarget: + case UnityTargetPlatform.NoTarget: return generic; // Test specific - case Platform.types.Test: + case UnityTargetPlatform.Test: return generic; default: throw new Error(` diff --git a/src/model/index.test.ts b/src/model/index.test.ts index e172367bd..f9317bb14 100644 --- a/src/model/index.test.ts +++ b/src/model/index.test.ts @@ -1,4 +1,4 @@ -import * as Index from '.'; +import * as Index from './index.ts'; describe('Index', () => { test.each(['Action', 'BuildParameters', 'Cache', 'Docker', 'ImageTag', 'Input', 'Platform', 'Project', 'Unity'])( diff --git a/src/model/index.ts b/src/model/index.ts index dfd5b2f17..7c00b0f8a 100644 --- a/src/model/index.ts +++ b/src/model/index.ts @@ -1,27 +1,27 @@ -import Action from './action'; -import BuildParameters from './build-parameters'; -import Cache from './cache'; -import Docker from './docker'; -import Input from './input'; -import ImageTag from './image-tag'; -import Output from './output'; -import Platform from './platform'; -import Project from './project'; -import Unity from './unity'; -import Versioning from './versioning'; -import CloudRunner from './cloud-runner/cloud-runner'; +import Action from './action.ts'; +import Parameters from './parameters.ts'; +import Cache from './cache.ts'; +import Docker from './docker.ts'; +import Input from './input.ts'; +import ImageTag from './image-tag.ts'; +import Output from './output.ts'; +import UnityTargetPlatform from './unity/unity-target-platform.ts'; +import Project from './project.ts'; +import Unity from './unity/unity.ts'; +import BuildVersionGenerator from '../middleware/build-versioning/build-version-generator.ts'; +import CloudRunner from './cloud-runner/cloud-runner.ts'; export { Action, - BuildParameters, + Parameters, Cache, Docker, Input, ImageTag, Output, - Platform, + UnityTargetPlatform, Project, Unity, - Versioning, - CloudRunner as CloudRunner, + BuildVersionGenerator, + CloudRunner, }; diff --git a/src/model/input-readers/action-yaml.ts b/src/model/input-readers/action-yaml.ts index 8c46bb661..349ea02a8 100644 --- a/src/model/input-readers/action-yaml.ts +++ b/src/model/input-readers/action-yaml.ts @@ -1,6 +1,4 @@ -import fs from 'fs'; -import path from 'path'; -import YAML from 'yaml'; +import { fsSync as fs, path, yaml, __dirname } from '../../dependencies.ts'; export class ActionYamlReader { private actionYamlParsed: any; @@ -9,7 +7,7 @@ export class ActionYamlReader { if (!fs.existsSync(filename)) { filename = path.join(__dirname, `..`, filename); } - this.actionYamlParsed = YAML.parse(fs.readFileSync(filename).toString()); + this.actionYamlParsed = yaml.parse(Deno.readTextFileSync(filename)); } public GetActionYamlValue(key: string) { return this.actionYamlParsed.inputs[key]?.description || 'No description found in action.yml'; diff --git a/src/model/input-readers/generic-input-reader.ts b/src/model/input-readers/generic-input-reader.ts index 85a42fea4..4428fde90 100644 --- a/src/model/input-readers/generic-input-reader.ts +++ b/src/model/input-readers/generic-input-reader.ts @@ -1,5 +1,5 @@ -import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system'; -import Input from '../input'; +import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system.ts'; +import Input from '../input.ts'; export class GenericInputReader { public static async Run(command) { diff --git a/src/model/input-readers/git-repo.test.ts b/src/model/input-readers/git-repo.test.ts index 1e96db601..39ca27c1f 100644 --- a/src/model/input-readers/git-repo.test.ts +++ b/src/model/input-readers/git-repo.test.ts @@ -1,4 +1,4 @@ -import { GitRepoReader } from './git-repo'; +import { GitRepoReader } from './git-repo.ts'; describe(`git repo tests`, () => { it(`Branch value parsed from CLI to not contain illegal characters`, async () => { diff --git a/src/model/input-readers/git-repo.ts b/src/model/input-readers/git-repo.ts index 3372089e2..7f9be99a4 100644 --- a/src/model/input-readers/git-repo.ts +++ b/src/model/input-readers/git-repo.ts @@ -1,16 +1,18 @@ -import { assert } from 'console'; -import fs from 'fs'; -import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system'; -import CloudRunnerLogger from '../cloud-runner/services/cloud-runner-logger'; -import Input from '../input'; +import { assert } from 'https://deno.land/std@0.142.0/testing/asserts.ts'; +import { fsSync as fs } from '../../dependencies.ts'; +import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system.ts'; +import CloudRunnerLogger from '../cloud-runner/services/cloud-runner-logger.ts'; +import Input from '../input.ts'; +// Todo - DENO - return assertions export class GitRepoReader { public static async GetRemote() { if (Input.cloudRunnerCluster === 'local') { return ''; } assert(fs.existsSync(`.git`)); - const value = (await CloudRunnerSystem.Run(`git remote -v`, false, true)).replace(/ /g, ``); + const upstream = await CloudRunnerSystem.Run(`git remote -v`, false, true); + const value = upstream.replace(/ /g, ''); CloudRunnerLogger.log(`value ${value}`); assert(value.includes('github.com')); @@ -18,14 +20,11 @@ export class GitRepoReader { } public static async GetBranch() { - if (Input.cloudRunnerCluster === 'local') { - return ''; - } + if (Input.cloudRunnerCluster === 'local') return ''; + assert(fs.existsSync(`.git`)); + const currentBranch = await CloudRunnerSystem.Run(`git branch --show-current`, false, true); - return (await CloudRunnerSystem.Run(`git branch --show-current`, false, true)) - .split('\n')[0] - .replace(/ /g, ``) - .replace('/head', ''); + return currentBranch.split('\n')[0].replace(/ /g, ``).replace('/head', ''); } } diff --git a/src/model/input-readers/github-cli.test.ts b/src/model/input-readers/github-cli.test.ts index 424af0596..562fb3129 100644 --- a/src/model/input-readers/github-cli.test.ts +++ b/src/model/input-readers/github-cli.test.ts @@ -1,5 +1,5 @@ -import { GithubCliReader } from './github-cli'; -import * as core from '@actions/core'; +import { GithubCliReader } from './github-cli.ts'; +import { core } from '../../dependencies.ts'; describe(`github cli`, () => { // Todo - We can not assume that everyone has the GitHub cli installed locally. @@ -7,6 +7,6 @@ describe(`github cli`, () => { const token = await GithubCliReader.GetGitHubAuthToken(); // Todo - use expect(result).toStrictEqual(something) - core.info(token); + log.info(token); }); }); diff --git a/src/model/input-readers/github-cli.ts b/src/model/input-readers/github-cli.ts index 7ad6ef7a4..9968c7790 100644 --- a/src/model/input-readers/github-cli.ts +++ b/src/model/input-readers/github-cli.ts @@ -1,26 +1,23 @@ -import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system'; -import * as core from '@actions/core'; -import Input from '../input'; +import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system.ts'; +import Input from '../input.ts'; export class GithubCliReader { static async GetGitHubAuthToken() { - if (Input.cloudRunnerCluster === 'local') { - return ''; - } + if (Input.cloudRunnerCluster === 'local') return ''; + try { const authStatus = await CloudRunnerSystem.Run(`gh auth status`, true, true); if (authStatus.includes('You are not logged') || authStatus === '') { return ''; } - return (await CloudRunnerSystem.Run(`gh auth status -t`, false, true)) - .split(`Token: `)[1] - .replace(/ /g, '') - .replace(/\n/g, ''); - } catch (error: any) { - core.info(error || 'Failed to get github auth token from gh cli'); + const status = await CloudRunnerSystem.Run(`gh auth status -t`, false, true); - return ''; + return status.split(`Token: `)[1].replace(/ /g, '').replace(/\n/g, ''); + } catch (error) { + log.info(error || 'Failed to get github auth token from gh cli'); } + + return ''; } } diff --git a/src/model/input-readers/test-license-reader.ts b/src/model/input-readers/test-license-reader.ts index 5d072743c..12c758a1a 100644 --- a/src/model/input-readers/test-license-reader.ts +++ b/src/model/input-readers/test-license-reader.ts @@ -1,13 +1,12 @@ -import path from 'path'; -import fs from 'fs'; -import YAML from 'yaml'; -import Input from '../input'; +import { fsSync as fs, path, yaml, __dirname } from '../../dependencies.ts'; +import Input from '../input.ts'; -export function ReadLicense() { - if (Input.cloudRunnerCluster === 'local') { +export function ReadLicense(parameters) { + if (parameters.cloudRunnerCluster === 'local') { return ''; } + const pipelineFile = path.join(__dirname, `.github`, `workflows`, `cloud-runner-k8s-pipeline.yml`); - return fs.existsSync(pipelineFile) ? YAML.parse(fs.readFileSync(pipelineFile, 'utf8')).env.UNITY_LICENSE : ''; + return fs.existsSync(pipelineFile) ? yaml.parse(Deno.readTextFileSync(pipelineFile, 'utf8')).env.UNITY_LICENSE : ''; } diff --git a/src/model/input.test.ts b/src/model/input.test.ts index 7cf2fc54a..931b15edb 100644 --- a/src/model/input.test.ts +++ b/src/model/input.test.ts @@ -1,22 +1,22 @@ -import * as core from '@actions/core'; +import { core } from '../dependencies.ts'; -import Input from './input'; -import Platform from './platform'; +import Input from './input.ts'; +import UnityTargetPlatform from './unity/unity-target-platform.ts'; afterEach(() => { jest.restoreAllMocks(); }); describe('Input', () => { - describe('unityVersion', () => { + describe('engineVersion', () => { it('returns the default value', () => { - expect(Input.unityVersion).toStrictEqual('auto'); + expect(Input.engineVersion).toStrictEqual('auto'); }); it('takes input from the users workflow', () => { const mockValue = '2020.4.99f9'; const spy = jest.spyOn(core, 'getInput').mockReturnValue(mockValue); - expect(Input.unityVersion).toStrictEqual(mockValue); + expect(Input.engineVersion).toStrictEqual(mockValue); expect(spy).toHaveBeenCalledTimes(1); }); }); @@ -35,7 +35,7 @@ describe('Input', () => { describe('targetPlatform', () => { it('returns the default value', () => { - expect(Input.targetPlatform).toStrictEqual(Platform.default); + expect(Input.targetPlatform).toStrictEqual(UnityTargetPlatform.default); }); it('takes input from the users workflow', () => { diff --git a/src/model/input.ts b/src/model/input.ts index d3bb81160..969e2852a 100644 --- a/src/model/input.ts +++ b/src/model/input.ts @@ -1,29 +1,43 @@ -import fs from 'fs'; -import path from 'path'; -import { Cli } from './cli/cli'; -import CloudRunnerQueryOverride from './cloud-runner/services/cloud-runner-query-override'; -import Platform from './platform'; - -const core = require('@actions/core'); +import { fsSync as fs, path, core } from '../dependencies.ts'; +import { Cli } from './cli/cli.ts'; +import CloudRunnerQueryOverride from './cloud-runner/services/cloud-runner-query-override.ts'; +import { CliArguments } from '../core/cli/cli-arguments.ts'; /** - * Input variables specified in workflows using "with" prop. + * Input variables specified directly on the commandline. * + * Todo - check if the following statement is still correct: * Note that input is always passed as a string, even booleans. * * Todo: rename to UserInput and remove anything that is not direct input from the user / ci workflow */ class Input { + private readonly arguments: CliArguments; + + constructor(argumentsFromCli: CliArguments) { + this.arguments = argumentsFromCli; + + return this; + } + + // Todo - read something from environment instead and make that into a parameter, then use that. public static githubInputEnabled: boolean = true; - public static getInput(query) { + // Todo - Note that this is now invoked both statically and dynamically - which is a temporary mess. + public get(query: string) { + if (this && this.arguments) { + return this.arguments.get(query); + } + + // Legacy (static) + log.warn(`Querying static`); if (Input.githubInputEnabled) { const coreInput = core.getInput(query); if (coreInput && coreInput !== '') { return coreInput; } } - const alternativeQuery = Input.ToEnvVarFormat(query); + const alternativeQuery = Input.toEnvVarFormat(query); // Query input sources if (Cli.query(query, alternativeQuery)) { @@ -45,24 +59,26 @@ class Input { return; } - static get region(): string { - return Input.getInput('region') || 'eu-west-2'; + public get region(): string { + return this.get('region'); } - static get githubRepo() { - return Input.getInput('GITHUB_REPOSITORY') || Input.getInput('GITHUB_REPO') || undefined; + public get githubRepo() { + return this.get('GITHUB_REPOSITORY') || this.get('GITHUB_REPO') || undefined; } - static get branch() { - if (Input.getInput(`GITHUB_REF`)) { - return Input.getInput(`GITHUB_REF`).replace('refs/', '').replace(`head/`, '').replace(`heads/`, ''); - } else if (Input.getInput('branch')) { - return Input.getInput('branch'); + + public get branch() { + if (this.get(`GITHUB_REF`)) { + return this.get(`GITHUB_REF`).replace('refs/', '').replace(`head/`, '').replace(`heads/`, ''); + } else if (this.get('branch')) { + return this.get('branch').replace('/head', ''); } else { return ''; } } - static get cloudRunnerBuilderPlatform() { - const input = Input.getInput('cloudRunnerBuilderPlatform'); + + public get cloudRunnerBuilderPlatform() { + const input = this.get('cloudRunnerBuilderPlatform'); if (input) { return input; } @@ -73,216 +89,181 @@ class Input { return; } - static get gitSha() { - if (Input.getInput(`GITHUB_SHA`)) { - return Input.getInput(`GITHUB_SHA`); - } else if (Input.getInput(`GitSHA`)) { - return Input.getInput(`GitSHA`); + public get gitSha() { + if (this.get(`GITHUB_SHA`)) { + return this.get(`GITHUB_SHA`); + } else if (this.get(`GitSHA`)) { + return this.get(`GitSHA`); } } - static get runNumber() { - return Input.getInput('GITHUB_RUN_NUMBER') || '0'; + public get runNumber() { + return this.get('GITHUB_RUN_NUMBER') || '0'; } - static get targetPlatform() { - return Input.getInput('targetPlatform') || Platform.default; + public get unitySerial() { + return this.get('unitySerial') || ''; } - static get unityVersion() { - return Input.getInput('unityVersion') || 'auto'; - } + public get projectPath() { + let input = this.get('projectPath'); - static get customImage() { - return Input.getInput('customImage') || ''; - } + // Todo - remove hardcoded test project reference + const isTestProject = + fs.existsSync(path.join('test-project', 'ProjectSettings', 'ProjectVersion.txt')) && + !fs.existsSync(path.join('ProjectSettings', 'ProjectVersion.txt')); + if (!input && isTestProject) input = 'test-project'; - static get projectPath() { - const input = Input.getInput('projectPath'); - const rawProjectPath = input - ? input - : fs.existsSync(path.join('test-project', 'ProjectSettings', 'ProjectVersion.txt')) && - !fs.existsSync(path.join('ProjectSettings', 'ProjectVersion.txt')) - ? 'test-project' - : '.'; + if (!input) input = '.'; - return rawProjectPath.replace(/\/$/, ''); + return input.replace(/\/$/, ''); } - static get buildName() { - return Input.getInput('buildName') || this.targetPlatform; + public get buildName() { + return this.get('buildName'); } - static get buildsPath() { - return Input.getInput('buildsPath') || 'build'; + public get buildMethod() { + return this.get('buildMethod') || ''; // Processed in docker file } - static get buildMethod() { - return Input.getInput('buildMethod') || ''; // Processed in docker file + public get customParameters() { + return this.get('customParameters') || ''; } - static get customParameters() { - return Input.getInput('customParameters') || ''; + public get androidVersionCode() { + return this.get('androidVersionCode'); } - static get versioningStrategy() { - return Input.getInput('versioning') || 'Semantic'; - } - - static get specifiedVersion() { - return Input.getInput('version') || ''; - } - - static get androidVersionCode() { - return Input.getInput('androidVersionCode'); - } - - static get androidAppBundle() { - const input = Input.getInput('androidAppBundle') || false; + public get androidAppBundle() { + const input = this.get('androidAppBundle') || false; return input === 'true'; } - static get androidKeystoreName() { - return Input.getInput('androidKeystoreName') || ''; + public get androidKeystoreName() { + return this.get('androidKeystoreName') || ''; } - static get androidKeystoreBase64() { - return Input.getInput('androidKeystoreBase64') || ''; + public get androidKeystoreBase64() { + return this.get('androidKeystoreBase64') || ''; } - static get androidKeystorePass() { - return Input.getInput('androidKeystorePass') || ''; + public get androidKeystorePass() { + return this.get('androidKeystorePass') || ''; } - static get androidKeyaliasName() { - return Input.getInput('androidKeyaliasName') || ''; + public get androidKeyaliasName() { + return this.get('androidKeyaliasName') || ''; } - static get androidKeyaliasPass() { - return Input.getInput('androidKeyaliasPass') || ''; + public get androidKeyaliasPass() { + return this.get('androidKeyaliasPass') || ''; } - static get androidTargetSdkVersion() { - return Input.getInput('androidTargetSdkVersion') || ''; + public get androidTargetSdkVersion() { + return this.get('androidTargetSdkVersion') || ''; } - static get sshAgent() { - return Input.getInput('sshAgent') || ''; + public get sshAgent() { + return this.get('sshAgent') || ''; } - static get gitPrivateToken() { - return core.getInput('gitPrivateToken') || false; + public get gitPrivateToken() { + return this.get('gitPrivateToken') || ''; } - static get customJob() { - return Input.getInput('customJob') || ''; + public get customJob() { + return this.get('customJob') || ''; } - static customJobHooks() { - return Input.getInput('customJobHooks') || ''; + public customJobHooks() { + return this.get('customJobHooks') || ''; } - static cachePushOverrideCommand() { - return Input.getInput('cachePushOverrideCommand') || ''; + public cachePushOverrideCommand() { + return this.get('cachePushOverrideCommand') || ''; } - static cachePullOverrideCommand() { - return Input.getInput('cachePullOverrideCommand') || ''; + public cachePullOverrideCommand() { + return this.get('cachePullOverrideCommand') || ''; } - static readInputFromOverrideList() { - return Input.getInput('readInputFromOverrideList') || ''; + public readInputFromOverrideList() { + return this.get('readInputFromOverrideList') || ''; } - static readInputOverrideCommand() { - return Input.getInput('readInputOverrideCommand') || ''; + public readInputOverrideCommand() { + return this.get('readInputOverrideCommand') || ''; } - static get cloudRunnerBranch() { - return Input.getInput('cloudRunnerBranch') || 'cloud-runner-develop'; + public get cloudRunnerBranch() { + return this.get('cloudRunnerBranch') || 'cloud-runner-develop'; } - static get chownFilesTo() { - return Input.getInput('chownFilesTo') || ''; + public get chownFilesTo() { + return this.get('chownFilesTo') || ''; } - static get allowDirtyBuild() { - const input = Input.getInput('allowDirtyBuild') || false; - - return input === 'true'; + public get postBuildSteps() { + return this.get('postBuildSteps') || ''; } - static get postBuildSteps() { - return Input.getInput('postBuildSteps') || ''; + public get preBuildSteps() { + return this.get('preBuildSteps') || ''; } - static get preBuildSteps() { - return Input.getInput('preBuildSteps') || ''; + public get awsBaseStackName() { + return this.get('awsBaseStackName') || 'game-ci'; } - static get awsBaseStackName() { - return Input.getInput('awsBaseStackName') || 'game-ci'; + public get cloudRunnerCpu() { + return this.get('cloudRunnerCpu'); } - static get cloudRunnerCluster() { - if (Cli.isCliMode) { - return Input.getInput('cloudRunnerCluster') || 'aws'; - } - - return Input.getInput('cloudRunnerCluster') || 'local'; + public get cloudRunnerMemory() { + return this.get('cloudRunnerMemory'); } - static get cloudRunnerCpu() { - return Input.getInput('cloudRunnerCpu'); + public get kubeConfig() { + return this.get('kubeConfig') || ''; } - static get cloudRunnerMemory() { - return Input.getInput('cloudRunnerMemory'); + public get kubeVolume() { + return this.get('kubeVolume') || ''; } - static get kubeConfig() { - return Input.getInput('kubeConfig') || ''; + public get kubeVolumeSize() { + return this.get('kubeVolumeSize') || '5Gi'; } - static get kubeVolume() { - return Input.getInput('kubeVolume') || ''; + public get kubeStorageClass(): string { + return this.get('kubeStorageClass') || ''; } - static get kubeVolumeSize() { - return Input.getInput('kubeVolumeSize') || '5Gi'; + public get checkDependencyHealthOverride(): string { + return this.get('checkDependencyHealthOverride') || ''; } - static get kubeStorageClass(): string { - return Input.getInput('kubeStorageClass') || ''; + public get startDependenciesOverride(): string { + return this.get('startDependenciesOverride') || ''; } - static get checkDependencyHealthOverride(): string { - return Input.getInput('checkDependencyHealthOverride') || ''; + public get cacheKey(): string { + return this.get('cacheKey') || Input.branch; } - static get startDependenciesOverride(): string { - return Input.getInput('startDependenciesOverride') || ''; + public get cloudRunnerTests(): boolean { + return this.get(`cloudRunnerTests`) || false; } - static get cacheKey(): string { - return Input.getInput('cacheKey') || Input.branch; - } - - static get cloudRunnerTests(): boolean { - return Input.getInput(`cloudRunnerTests`) || false; - } - - public static ToEnvVarFormat(input: string) { - if (input.toUpperCase() === input) { - return input; - } + /** + * @deprecated Use Parameter.toEnvFormat + */ + public static toEnvVarFormat(input: string) { + if (input.toUpperCase() === input) return input; - return input - .replace(/([A-Z])/g, ' $1') - .trim() - .toUpperCase() - .replace(/ /g, '_'); + return input.replace(/([\da-z])([A-Z])/g, '$1_$2').toUpperCase(); } } diff --git a/src/model/mac-builder.ts b/src/model/mac-builder.ts index a95cf7cb5..57360656b 100644 --- a/src/model/mac-builder.ts +++ b/src/model/mac-builder.ts @@ -1,10 +1,10 @@ -import { exec } from '@actions/exec'; -import { BuildParameters } from '.'; +import { Parameters } from './parameters.ts'; +import System from './system/system.ts'; class MacBuilder { - public static async run(actionFolder, workspace, buildParameters: BuildParameters, silent = false) { - await exec('bash', [`${actionFolder}/platforms/mac/entrypoint.sh`], { - silent, + public static async run(actionFolder) { + log.warning('running the process'); + await System.run(`bash ${actionFolder}/platforms/mac/entrypoint.sh`, { ignoreReturnCode: true, }); } diff --git a/src/model/output.test.ts b/src/model/output.test.ts index c4fcb1141..6fb618b5e 100644 --- a/src/model/output.test.ts +++ b/src/model/output.test.ts @@ -1,4 +1,4 @@ -import Output from './output'; +import Output from './output.ts'; describe('Output', () => { describe('setBuildVersion', () => { diff --git a/src/model/output.ts b/src/model/output.ts index 1e482f6e8..57e35998b 100644 --- a/src/model/output.ts +++ b/src/model/output.ts @@ -1,4 +1,4 @@ -const core = require('@actions/core'); +import { core } from '../dependencies.ts'; class Output { static async setBuildVersion(buildVersion) { diff --git a/src/model/parameter-options.ts b/src/model/parameter-options.ts new file mode 100644 index 000000000..c8735414f --- /dev/null +++ b/src/model/parameter-options.ts @@ -0,0 +1,18 @@ +type ParameterOption = { + cli: boolean; + env: boolean; + cfg: boolean; + defaultValue: T; +}; + +// Todo - use CLI lib to define this instead. +export class ParameterOptions { + static get region(): ParameterOption { + return { + cli: true, + env: true, + cfg: true, + defaultValue: 'eu-west-2', + }; + } +} diff --git a/src/model/parameter.ts b/src/model/parameter.ts new file mode 100644 index 000000000..b8c12878a --- /dev/null +++ b/src/model/parameter.ts @@ -0,0 +1,14 @@ +export class Parameter { + public static toUpperSnakeCase(input: string) { + if (input.toUpperCase() === input) return input; + + return input.replace(/([\da-z])([A-Z])/g, '$1_$2').toUpperCase(); + } + + public static toCamelCase(input: string) { + return input + .split('_') + .map((word) => `${word[0].toUppercase()}${word.slice(1).toLowerCase()}`) + .join(''); + } +} diff --git a/src/model/parameters.ts b/src/model/parameters.ts new file mode 100644 index 000000000..03423f64e --- /dev/null +++ b/src/model/parameters.ts @@ -0,0 +1,169 @@ +import Input from './input.ts'; +import BuildVersionGenerator from '../middleware/build-versioning/build-version-generator.ts'; +import { GitRepoReader } from './input-readers/git-repo.ts'; +import { CommandInterface } from '../command/command-interface.ts'; +import { Environment } from '../core/env/environment.ts'; +import { Parameter } from './parameter.ts'; + +class Parameters { + private command: CommandInterface; + public editorVersion!: string; + public customImage!: string; + public unitySerial!: string; + public runnerTempPath: string | undefined; + public targetPlatform!: string; + public projectPath!: string; + public buildName!: string; + public buildPath!: string; + public buildFile!: string; + public buildMethod!: string; + public buildVersion!: string; + public androidVersionCode!: string; + public androidKeystoreName!: string; + public androidKeystoreBase64!: string; + public androidKeystorePass!: string; + public androidKeyaliasName!: string; + public androidKeyaliasPass!: string; + public androidTargetSdkVersion!: string; + public androidSdkManagerParameters!: string; + public customParameters!: string; + public sshAgent!: string; + public cloudRunnerCluster!: string; + public awsBaseStackName!: string; + public gitPrivateToken!: string; + public awsStackName!: string; + public kubeConfig!: string; + public cloudRunnerMemory!: string; + public cloudRunnerCpu!: string; + public kubeVolumeSize!: string; + public kubeVolume!: string; + public kubeStorageClass!: string; + public chownFilesTo!: string; + public customJobHooks!: string; + public cachePushOverrideCommand!: string; + public cachePullOverrideCommand!: string; + public readInputFromOverrideList!: string; + public readInputOverrideCommand!: string; + public checkDependencyHealthOverride!: string; + public startDependenciesOverride!: string; + public cacheKey!: string; + public postBuildSteps!: string; + public preBuildSteps!: string; + public customJob!: string; + public runNumber!: string; + public branch!: string; + public githubRepo!: string; + public gitSha!: string; + public logId!: string; + public buildGuid!: string; + public cloudRunnerBranch!: string; + public cloudRunnerIntegrationTests!: boolean; + public cloudRunnerBuilderPlatform!: string | undefined; + public isCliMode!: boolean; + + private defaults: Partial = { + region: 'eu-west-2', + }; + + private readonly input: Input; + private readonly env: Environment; + + constructor(input: Input, env: Environment) { + this.input = input; + this.env = env; + + // Todo - ~/.gameci should hold a config with default settings, like cloud region = 'eu-west-2' + // this.config = config; + } + + public get(query, useDefault = true) { + const defaultValue = useDefault ? this.default(query) : undefined; + const value = this.input.get(query) || this.env.get(Parameter.toUpperSnakeCase(query)) || defaultValue; + + if (log.isVeryVerbose) log.debug('Argument:', query, '=', value); + + return value; + } + + private default(query) { + return this.defaults[query]; + } + + public async parse(): Promise { + const branch = (await BuildVersionGenerator.getCurrentBranch()) || (await GitRepoReader.GetBranch()); + + const parameters = { + branch, + unitySerial: this.getUnitySerial(), + editorVersion: engineVersion, + runnerTempPath: this.env.get('RUNNER_TEMP'), + buildName, + buildPath, + buildFile, + buildMethod: this.input.buildMethod, + buildVersion, + androidVersionCode, + androidKeystoreName: this.input.androidKeystoreName, + androidKeystoreBase64: this.input.androidKeystoreBase64, + androidKeystorePass: this.input.androidKeystorePass, + androidKeyaliasName: this.input.androidKeyaliasName, + androidKeyaliasPass: this.input.androidKeyaliasPass, + androidTargetSdkVersion, + androidSdkManagerParameters, + customParameters: this.get('customParameters'), + sshAgent: this.input.sshAgent, + gitPrivateToken: this.get('gitPrivateToken'), + chownFilesTo: this.input.chownFilesTo, + customJob: this.input.customJob, + }; + + const commandParameterOverrides = await this.command.parseParameters(this.input, parameters); + + // Todo - Maybe return an instance instead + return { + ...parameters, + ...commandParameterOverrides, + }; + } + + private getUnitySerial() { + let unitySerial = this.get('unitySerial'); + + if (!unitySerial && this.env.getOS() === 'windows') { + const unityLicense = this.get('unityLicense'); + + unitySerial = this.getSerialFromLicense(unityLicense); + } + + return unitySerial; + } + + private getSerialFromLicense(license) { + if (!license) { + throw new Error(String.dedent` + Missing Unity License File and no Unity Serial was found. If this is a personal license, + make sure to follow the activation steps and set the UNITY_LICENSE variable or enter + a Unity serial number inside the UNITY_SERIAL variable. + `); + } + + const startKey = ``; + const startIndex = license.indexOf(startKey) + startKey.length; + if (startIndex < 0) { + throw new Error(`License File was corrupted, unable to locate serial`); + } + const endIndex = license.indexOf(endKey, startIndex); + + // Slice off the first 4 characters as they are garbage values + return Buffer.from(license.slice(startIndex, endIndex), 'base64').toString('binary').slice(4); + } + + registerCommand(command: CommandInterface) { + this.command = command; + + return this; + } +} + +export default Parameters; diff --git a/src/model/platform-setup.ts b/src/model/platform-setup.ts index 7ef9d9381..cd9bdfe8c 100644 --- a/src/model/platform-setup.ts +++ b/src/model/platform-setup.ts @@ -1,16 +1,15 @@ -import { BuildParameters } from '.'; -import { SetupMac, SetupWindows } from './platform-setup/'; -import ValidateWindows from './platform-validation/validate-windows'; +import { Parameters } from './index.ts'; +import { SetupMac, SetupWindows } from '../logic/unity/platform-setup/index.ts'; +import ValidateWindows from '../logic/unity/platform-validation/validate-windows.ts'; class PlatformSetup { - static async setup(buildParameters: BuildParameters, actionFolder: string) { + static async setup(parameters: Parameters, actionFolder: string) { switch (process.platform) { case 'win32': - ValidateWindows.validate(buildParameters); - SetupWindows.setup(buildParameters); + await SetupWindows.setup(parameters); break; case 'darwin': - await SetupMac.setup(buildParameters, actionFolder); + await SetupMac.setup(parameters, actionFolder); break; // Add other baseOS's here diff --git a/src/model/platform-setup/index.ts b/src/model/platform-setup/index.ts deleted file mode 100644 index cc2ab97b1..000000000 --- a/src/model/platform-setup/index.ts +++ /dev/null @@ -1,4 +0,0 @@ -import SetupWindows from './setup-windows'; -import SetupMac from './setup-mac'; - -export { SetupWindows, SetupMac }; diff --git a/src/model/platform-setup/setup-mac.ts b/src/model/platform-setup/setup-mac.ts deleted file mode 100644 index beb551954..000000000 --- a/src/model/platform-setup/setup-mac.ts +++ /dev/null @@ -1,75 +0,0 @@ -import { BuildParameters } from '..'; -import { getUnityChangeset } from 'unity-changeset'; -import { exec } from '@actions/exec'; -import fs from 'fs'; - -class SetupMac { - static unityHubPath = `"/Applications/Unity Hub.app/Contents/MacOS/Unity Hub"`; - - public static async setup(buildParameters: BuildParameters, actionFolder: string) { - const unityEditorPath = `/Applications/Unity/Hub/Editor/${buildParameters.editorVersion}/Unity.app/Contents/MacOS/Unity`; - - // Only install unity if the editor doesn't already exist - if (!fs.existsSync(unityEditorPath)) { - await SetupMac.installUnityHub(); - await SetupMac.installUnity(buildParameters); - } - - await SetupMac.setEnvironmentVariables(buildParameters, actionFolder); - } - - private static async installUnityHub(silent = false) { - const command = 'brew install unity-hub'; - if (!fs.existsSync(this.unityHubPath)) { - // Ignoring return code because the log seems to overflow the internal buffer which triggers - // a false error - const errorCode = await exec(command, undefined, { silent, ignoreReturnCode: true }); - if (errorCode) { - throw new Error(`There was an error installing the Unity Editor. See logs above for details.`); - } - } - } - - private static async installUnity(buildParameters: BuildParameters, silent = false) { - const unityChangeset = await getUnityChangeset(buildParameters.editorVersion); - const command = `${this.unityHubPath} -- --headless install \ - --version ${buildParameters.editorVersion} \ - --changeset ${unityChangeset.changeset} \ - --module mac-il2cpp \ - --childModules`; - - // Ignoring return code because the log seems to overflow the internal buffer which triggers - // a false error - const errorCode = await exec(command, undefined, { silent, ignoreReturnCode: true }); - if (errorCode) { - throw new Error(`There was an error installing the Unity Editor. See logs above for details.`); - } - } - - private static async setEnvironmentVariables(buildParameters: BuildParameters, actionFolder: string) { - // Need to set environment variables from here because we execute - // the scripts on the host for mac - process.env.ACTION_FOLDER = actionFolder; - process.env.UNITY_VERSION = buildParameters.editorVersion; - process.env.UNITY_SERIAL = buildParameters.unitySerial; - process.env.PROJECT_PATH = buildParameters.projectPath; - process.env.BUILD_TARGET = buildParameters.targetPlatform; - process.env.BUILD_NAME = buildParameters.buildName; - process.env.BUILD_PATH = buildParameters.buildPath; - process.env.BUILD_FILE = buildParameters.buildFile; - process.env.BUILD_METHOD = buildParameters.buildMethod; - process.env.VERSION = buildParameters.buildVersion; - process.env.ANDROID_VERSION_CODE = buildParameters.androidVersionCode; - process.env.ANDROID_KEYSTORE_NAME = buildParameters.androidKeystoreName; - process.env.ANDROID_KEYSTORE_BASE64 = buildParameters.androidKeystoreBase64; - process.env.ANDROID_KEYSTORE_PASS = buildParameters.androidKeystorePass; - process.env.ANDROID_KEYALIAS_NAME = buildParameters.androidKeyaliasName; - process.env.ANDROID_KEYALIAS_PASS = buildParameters.androidKeyaliasPass; - process.env.ANDROID_TARGET_SDK_VERSION = buildParameters.androidTargetSdkVersion; - process.env.ANDROID_SDK_MANAGER_PARAMETERS = buildParameters.androidSdkManagerParameters; - process.env.CUSTOM_PARAMETERS = buildParameters.customParameters; - process.env.CHOWN_FILES_TO = buildParameters.chownFilesTo; - } -} - -export default SetupMac; diff --git a/src/model/platform-setup/setup-windows.ts b/src/model/platform-setup/setup-windows.ts deleted file mode 100644 index 2ba7ab365..000000000 --- a/src/model/platform-setup/setup-windows.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { exec } from '@actions/exec'; -import fs from 'fs'; -import { BuildParameters } from '..'; - -class SetupWindows { - public static async setup(buildParameters: BuildParameters) { - const { targetPlatform } = buildParameters; - - await SetupWindows.setupWindowsRun(targetPlatform); - } - - private static async setupWindowsRun(targetPlatform, silent = false) { - if (!fs.existsSync('c:/regkeys')) { - fs.mkdirSync('c:/regkeys'); - } - - // These all need the Windows 10 SDK - switch (targetPlatform) { - case 'StandaloneWindows': - case 'StandaloneWindows64': - case 'WSAPlayer': - await this.generateWinSDKRegKeys(silent); - break; - } - } - - private static async generateWinSDKRegKeys(silent = false) { - // Export registry keys that point to the Windows 10 SDK - const exportWinSDKRegKeysCommand = - 'reg export "HKLM\\SOFTWARE\\WOW6432Node\\Microsoft\\Microsoft SDKs\\Windows\\v10.0" c:/regkeys/winsdk.reg /y'; - await exec(exportWinSDKRegKeysCommand, undefined, { silent }); - } -} - -export default SetupWindows; diff --git a/src/model/platform.test.ts b/src/model/platform.test.ts deleted file mode 100644 index 7fb820b60..000000000 --- a/src/model/platform.test.ts +++ /dev/null @@ -1,37 +0,0 @@ -import Platform from './platform'; - -describe('Platform', () => { - describe('default', () => { - it('does not throw', () => { - expect(() => Platform.default).not.toThrow(); - }); - - it('returns a string', () => { - expect(typeof Platform.default).toStrictEqual('string'); - }); - - it('returns a platform', () => { - expect(Object.values(Platform.types)).toContain(Platform.default); - }); - }); - - describe('isWindows', () => { - it('returns true for windows', () => { - expect(Platform.isWindows(Platform.types.StandaloneWindows64)).toStrictEqual(true); - }); - - it('returns false for MacOS', () => { - expect(Platform.isWindows(Platform.types.StandaloneOSX)).toStrictEqual(false); - }); - }); - - describe('isAndroid', () => { - it('returns true for Android', () => { - expect(Platform.isAndroid(Platform.types.Android)).toStrictEqual(true); - }); - - it('returns false for Windows', () => { - expect(Platform.isAndroid(Platform.types.StandaloneWindows64)).toStrictEqual(false); - }); - }); -}); diff --git a/src/model/platform.ts b/src/model/platform.ts deleted file mode 100644 index 2d8fe865d..000000000 --- a/src/model/platform.ts +++ /dev/null @@ -1,53 +0,0 @@ -class Platform { - static get default() { - return Platform.types.StandaloneWindows64; - } - - static get types() { - return { - StandaloneOSX: 'StandaloneOSX', - StandaloneWindows: 'StandaloneWindows', - StandaloneWindows64: 'StandaloneWindows64', - StandaloneLinux64: 'StandaloneLinux64', - iOS: 'iOS', - Android: 'Android', - WebGL: 'WebGL', - WSAPlayer: 'WSAPlayer', - PS4: 'PS4', - XboxOne: 'XboxOne', - tvOS: 'tvOS', - Switch: 'Switch', - - // Unsupported - Lumin: 'Lumin', - BJM: 'BJM', - Stadia: 'Stadia', - Facebook: 'Facebook', - NoTarget: 'NoTarget', - - // Test specific - Test: 'Test', - }; - } - - static isWindows(platform) { - switch (platform) { - case Platform.types.StandaloneWindows: - case Platform.types.StandaloneWindows64: - return true; - default: - return false; - } - } - - static isAndroid(platform) { - switch (platform) { - case Platform.types.Android: - return true; - default: - return false; - } - } -} - -export default Platform; diff --git a/src/model/project.test.ts b/src/model/project.test.ts index b900929d0..5da7915ea 100644 --- a/src/model/project.test.ts +++ b/src/model/project.test.ts @@ -1,4 +1,4 @@ -import Project from './project'; +import Project from './project.ts'; jest.mock('./input'); diff --git a/src/model/project.ts b/src/model/project.ts index 718331205..b22f50bc6 100644 --- a/src/model/project.ts +++ b/src/model/project.ts @@ -1,6 +1,6 @@ -import Input from './input'; -import Unity from './unity'; -import Action from './action'; +import Input from './input.ts'; +import Unity from './unity/unity.ts'; +import Action from './action.ts'; class Project { static get relativePath() { diff --git a/src/model/system.ts b/src/model/system.ts deleted file mode 100644 index 9f1cb6761..000000000 --- a/src/model/system.ts +++ /dev/null @@ -1,66 +0,0 @@ -import * as core from '@actions/core'; -import { exec } from '@actions/exec'; - -class System { - static async run(command, arguments_: any = [], options = {}, shouldLog = true) { - let result = ''; - let error = ''; - let debug = ''; - - const listeners = { - stdout: (dataBuffer) => { - result += dataBuffer.toString(); - }, - stderr: (dataBuffer) => { - error += dataBuffer.toString(); - }, - debug: (dataString) => { - debug += dataString.toString(); - }, - }; - - const showOutput = () => { - if (debug !== '' && shouldLog) { - core.debug(debug); - } - - if (result !== '' && shouldLog) { - core.info(result); - } - - if (error !== '' && shouldLog) { - core.warning(error); - } - }; - - const throwContextualError = (message) => { - let commandAsString = command; - if (Array.isArray(arguments_)) { - commandAsString += ` ${arguments_.join(' ')}`; - } else if (typeof arguments_ === 'string') { - commandAsString += ` ${arguments_}`; - } - - throw new Error(`Failed to run "${commandAsString}".\n ${message}`); - }; - - try { - if (command.trim() === '') { - throw new Error(`Failed to execute empty command`); - } - - const exitCode = await exec(command, arguments_, { silent: true, listeners, ...options }); - showOutput(); - if (exitCode !== 0) { - throwContextualError(`Command returned non-zero exit code.\nError: ${error}`); - } - } catch (inCommandError) { - showOutput(); - throwContextualError(`In-command error caught: ${inCommandError}`); - } - - return result; - } -} - -export default System; diff --git a/src/model/system.integration.test.ts b/src/model/system/system.integration.test.ts similarity index 90% rename from src/model/system.integration.test.ts rename to src/model/system/system.integration.test.ts index 4d452fd15..7b7d46e6c 100644 --- a/src/model/system.integration.test.ts +++ b/src/model/system/system.integration.test.ts @@ -1,5 +1,5 @@ -import * as core from '@actions/core'; -import System from './system'; +import { core } from '../../dependencies.ts'; +import System from './system.ts'; jest.spyOn(core, 'debug').mockImplementation(() => {}); jest.spyOn(core, 'info').mockImplementation(() => {}); @@ -15,7 +15,7 @@ describe('System', () => { * To ensure our integration with '@actions/exec' works as expected we run some specific tests in CI only */ describe('integration', () => { - if (!process.env.CI) { + if (!Deno.env.get('CI')) { it("doesn't run locally", () => { expect(true).toBe(true); }); @@ -37,7 +37,7 @@ describe('System', () => { System.run('sh', undefined, { input: Buffer.from('git tag --list --merged HEAD | grep v[0-9]* | wc -l'), // eslint-disable-next-line github/no-then - }).then((result) => Number(result)), + }).then(Number), ).resolves.not.toBeNaN(); }); } diff --git a/src/model/system.test.ts b/src/model/system/system.test.ts similarity index 88% rename from src/model/system.test.ts rename to src/model/system/system.test.ts index ed4c43f59..338a121b7 100644 --- a/src/model/system.test.ts +++ b/src/model/system/system.test.ts @@ -1,12 +1,11 @@ -import * as core from '@actions/core'; -import * as exec from '@actions/exec'; -import System from './system'; +import { core } from '../../dependencies.ts'; +import System from './system.ts'; jest.spyOn(core, 'debug').mockImplementation(() => {}); const info = jest.spyOn(core, 'info').mockImplementation(() => {}); jest.spyOn(core, 'warning').mockImplementation(() => {}); jest.spyOn(core, 'error').mockImplementation(() => {}); -const execSpy = jest.spyOn(exec, 'exec').mockImplementation(async () => 0); +const execSpy = jest.spyOn(System, 'run').mockImplementation(async () => 0); afterEach(() => jest.clearAllMocks()); diff --git a/src/model/system/system.ts b/src/model/system/system.ts new file mode 100644 index 000000000..fcf5d75a8 --- /dev/null +++ b/src/model/system/system.ts @@ -0,0 +1,123 @@ +export interface RunOptions { + cwd: string; + attach: boolean; +} + +class System { + /** + * Run any command as if you're typing in shell. + * Make sure it's Windows/MacOS/Ubuntu compatible or has alternative commands. + * + * Intended to always be silent and capture the output, unless attach is passed. + * + * @returns {string} output of the command on success or failure + * + * @throws {Error} if anything was output to stderr. + */ + static async run(command: string, options: RunOptions = {}): Promise { + const isWindows = Deno.build.os === 'windows'; + const shellMethod = isWindows ? System.powershellRun : System.shellRun; + + if (log.isVeryVerbose) log.debug(`The following command is run using ${shellMethod.name}`); + + return shellMethod(command, options); + } + + static async shellRun(rawCommand: string, options: RunOptions = {}): Promise { + const { attach, cwd } = options; + + let command = rawCommand; + if (cwd) command = `cd ${cwd} ; ${command}`; + + return attach ? System.runAndAttach('sh', ['-c', command]) : System.runAndCapture('sh', ['-c', command]); + } + + static async powershellRun(rawCommand: string, options: RunOptions = {}): Promise { + const { attach, cwd } = options; + + let command = rawCommand; + if (cwd) command = `cd ${cwd} ; ${command}`; + + return attach ? System.runAndAttach('powershell', [command]) : System.runAndCapture('powershell', [command]); + } + + /** + * Internal cross-platform run, that spawns a new process and captures its output. + * + * If any error is written to stderr, this method will throw them. + * ❌ new Error(stdoutErrors) + * + * In case of no errors, this will return an object similar to these examples + * ✔️ { status: { success: true, code: 0 }, output: 'output from the command' } + * ⚠️ { status: { success: false, code: 1~255 }, output: 'output from the command' } + * + * Example usage: + * System.newRun(sh, ['-c', 'echo something']) + * System.newRun(powershell, ['echo something']) + * + * @deprecated not really deprecated, but please use System.run instead because this method will be made private. + */ + public static async runAndCapture(command, args: string | string[] = []): Promise { + if (!Array.isArray(args)) args = [args]; + + const argsString = args.join(' '); + const process = Deno.run({ + cmd: [command, ...args], + stdout: 'piped', + stderr: 'piped', + }); + + const status = await process.status(); + const outputBuffer = await process.output(); + const errorBuffer = await process.stderrOutput(); + + process.close(); + + const output = new TextDecoder().decode(outputBuffer).replace(/[\n\r]+$/, ''); + const error = new TextDecoder().decode(errorBuffer).replace(/[\n\r]+$/, ''); + + const result = { status, output }; + + // Log command output if verbose is enabled + if (log.isVeryVerbose) { + const symbol = status.success ? '✅' : '❗'; + const truncatedOutput = output.length >= 30 ? `${output.slice(0, 27)}...` : output; + log.debug('Command:', command, argsString, symbol, { + status, + output: log.isMaxVerbose ? output : truncatedOutput, + }); + } + + if (error) { + // Make sure we don't swallow any output + const errorMessage = output ? `${error}\n\n---\n\nOutput before the error:\n${output}` : error; + + // Throw instead or returning when any output was written to stdout + throw new Error(errorMessage); + } + + return result; + } + + /** + * Output stdout and stderr to the terminal and attach to the process. + * + * Note that the return signature is slightly different from runAndCapture, because we don't have stderrOutput. + * + * Todo - it would be nice to pipe the output to both stdout and capture it in the result object, but this doesn't seem possible yet. + */ + private static async runAndAttach(command, args: string | string[] = []): Promise { + if (!Array.isArray(args)) args = [args]; + + const process = Deno.run({ cmd: [command, ...args] }); + const status = await process.status(); + + process.close(); + + if (!status.success) throw new Error(`Command failed with code ${status.code}`); + + return { status, output: 'runAndAttach has access to the output stream' }; + } +} + +export default System; diff --git a/src/model/unity.ts b/src/model/unity.ts deleted file mode 100644 index 72f8e1eda..000000000 --- a/src/model/unity.ts +++ /dev/null @@ -1,7 +0,0 @@ -class Unity { - static get libraryFolder() { - return 'Library'; - } -} - -export default Unity; diff --git a/src/model/unity/unity-target-platform.test.ts b/src/model/unity/unity-target-platform.test.ts new file mode 100644 index 000000000..3880bf968 --- /dev/null +++ b/src/model/unity/unity-target-platform.test.ts @@ -0,0 +1,37 @@ +import UnityTargetPlatform from './unity-target-platform.ts'; + +describe('UnityTargetPlatform', () => { + describe('default', () => { + it('does not throw', () => { + expect(() => UnityTargetPlatform.default).not.toThrow(); + }); + + it('returns a string', () => { + expect(typeof UnityTargetPlatform.default).toStrictEqual('string'); + }); + + it('returns a platform', () => { + expect(Object.values(UnityTargetPlatform.types)).toContain(UnityTargetPlatform.default); + }); + }); + + describe('isWindows', () => { + it('returns true for windows', () => { + expect(UnityTargetPlatform.isWindows(UnityTargetPlatform.StandaloneWindows64)).toStrictEqual(true); + }); + + it('returns false for MacOS', () => { + expect(UnityTargetPlatform.isWindows(UnityTargetPlatform.StandaloneOSX)).toStrictEqual(false); + }); + }); + + describe('isAndroid', () => { + it('returns true for Android', () => { + expect(UnityTargetPlatform.isAndroid(UnityTargetPlatform.Android)).toStrictEqual(true); + }); + + it('returns false for Windows', () => { + expect(UnityTargetPlatform.isAndroid(UnityTargetPlatform.StandaloneWindows64)).toStrictEqual(false); + }); + }); +}); diff --git a/src/model/unity/unity-target-platform.ts b/src/model/unity/unity-target-platform.ts new file mode 100644 index 000000000..477971354 --- /dev/null +++ b/src/model/unity/unity-target-platform.ts @@ -0,0 +1,48 @@ +class UnityTargetPlatform { + public static readonly Android = 'Android'; + public static readonly iOS = 'iOS'; + public static readonly StandaloneLinux64 = 'StandaloneLinux64'; + public static readonly StandaloneOSX = 'StandaloneOSX'; + public static readonly StandaloneWindows = 'StandaloneWindows'; + public static readonly StandaloneWindows64 = 'StandaloneWindows64'; + public static readonly Switch = 'Switch'; + public static readonly tvOS = 'tvOS'; + public static readonly WebGL = 'WebGL'; + public static readonly WSAPlayer = 'WSAPlayer'; + public static readonly XboxOne = 'XboxOne'; + + // Unsupported + public static readonly Lumin = 'Lumin'; + public static readonly BJM = 'BJM'; + public static readonly Stadia = 'Stadia'; + public static readonly Facebook = 'Facebook'; + public static readonly NoTarget = 'NoTarget'; + + // Test specific + public static readonly Test = 'Test'; + + static get default() { + return UnityTargetPlatform.StandaloneWindows64; + } + + static isWindows(platform) { + switch (platform) { + case UnityTargetPlatform.StandaloneWindows: + case UnityTargetPlatform.StandaloneWindows64: + return true; + default: + return false; + } + } + + static isAndroid(platform) { + switch (platform) { + case UnityTargetPlatform.Android: + return true; + default: + return false; + } + } +} + +export default UnityTargetPlatform; diff --git a/src/model/unity/unity-target-platforms.ts b/src/model/unity/unity-target-platforms.ts new file mode 100644 index 000000000..8da1bdbab --- /dev/null +++ b/src/model/unity/unity-target-platforms.ts @@ -0,0 +1,27 @@ +import { UnityTargetPlatform } from '../index.ts'; + +export class UnityTargetPlatforms { + public static readonly all = [ + UnityTargetPlatform.Android, + UnityTargetPlatform.iOS, + UnityTargetPlatform.StandaloneLinux64, + UnityTargetPlatform.StandaloneOSX, + UnityTargetPlatform.StandaloneWindows, + UnityTargetPlatform.StandaloneWindows64, + UnityTargetPlatform.Switch, + UnityTargetPlatform.tvOS, + UnityTargetPlatform.WebGL, + UnityTargetPlatform.WSAPlayer, + UnityTargetPlatform.XboxOne, + + // Unsupported + UnityTargetPlatform.Lumin, + UnityTargetPlatform.BJM, + UnityTargetPlatform.Stadia, + UnityTargetPlatform.Facebook, + UnityTargetPlatform.NoTarget, + + // Test specific + UnityTargetPlatform.Test, + ]; +} diff --git a/src/model/unity.test.ts b/src/model/unity/unity.test.ts similarity index 89% rename from src/model/unity.test.ts rename to src/model/unity/unity.test.ts index 05cd7d028..bbfe2b72e 100644 --- a/src/model/unity.test.ts +++ b/src/model/unity/unity.test.ts @@ -1,4 +1,4 @@ -import Unity from './unity'; +import Unity from './unity.ts'; describe('Unity', () => { describe('libraryFolder', () => { diff --git a/src/model/unity/unity.ts b/src/model/unity/unity.ts new file mode 100644 index 000000000..8c0ea823e --- /dev/null +++ b/src/model/unity/unity.ts @@ -0,0 +1,21 @@ +import UnityTargetPlatform from './unity-target-platform.ts'; + +class Unity { + static get libraryFolder() { + return 'Library'; + } + + static determineBuildFileName(buildName, platform, androidAppBundle) { + if (UnityTargetPlatform.isWindows(platform)) { + return `${buildName}.exe`; + } + + if (UnityTargetPlatform.isAndroid(platform)) { + return androidAppBundle ? `${buildName}.aab` : `${buildName}.apk`; + } + + return buildName; + } +} + +export default Unity; diff --git a/src/model/versioning/versioning-strategies.ts b/src/model/versioning/versioning-strategies.ts new file mode 100644 index 000000000..f16e4cc58 --- /dev/null +++ b/src/model/versioning/versioning-strategies.ts @@ -0,0 +1,7 @@ +import { VersioningStrategy } from './versioning-strategy.ts'; + +export class VersioningStrategies { + public static get all() { + return [VersioningStrategy.None, VersioningStrategy.Semantic, VersioningStrategy.Tag, VersioningStrategy.Custom]; + } +} diff --git a/src/model/versioning/versioning-strategy.ts b/src/model/versioning/versioning-strategy.ts new file mode 100644 index 000000000..bd0023e6b --- /dev/null +++ b/src/model/versioning/versioning-strategy.ts @@ -0,0 +1,6 @@ +export class VersioningStrategy { + public static None: 'None'; + public static Semantic: 'Semantic'; + public static Tag: 'Tag'; + public static Custom: 'Custom'; +} diff --git a/src/model/versioning.test.ts b/src/model/versioning/versioning.test.ts similarity index 52% rename from src/model/versioning.test.ts rename to src/model/versioning/versioning.test.ts index 0451c74fa..cfb5acd4d 100644 --- a/src/model/versioning.test.ts +++ b/src/model/versioning/versioning.test.ts @@ -1,8 +1,8 @@ -import * as core from '@actions/core'; -import NotImplementedException from './error/not-implemented-exception'; -import System from './system'; -import Versioning from './versioning'; -import { validVersionTagInputs, invalidVersionTagInputs } from './__data__/versions'; +import { core } from '../../dependencies.ts'; +import NotImplementedException from '../error/not-implemented-exception.ts'; +import System from '../system/system.ts'; +import BuildVersionGenerator from '../../middleware/build-versioning/build-version-generator.ts'; +import { validVersionTagInputs, invalidVersionTagInputs } from '../__data__/versions.ts'; afterEach(() => { jest.restoreAllMocks(); @@ -11,27 +11,27 @@ afterEach(() => { describe('Versioning', () => { describe('strategies', () => { it('returns an object', () => { - expect(typeof Versioning.strategies).toStrictEqual('object'); + expect(typeof BuildVersionGenerator.strategies).toStrictEqual('object'); }); it('has items', () => { - expect(Object.values(Versioning.strategies).length).toBeGreaterThan(2); + expect(Object.values(BuildVersionGenerator.strategies).length).toBeGreaterThan(2); }); it('has an opt out option', () => { - expect(Versioning.strategies).toHaveProperty('None'); + expect(BuildVersionGenerator.strategies).toHaveProperty('None'); }); it('has the semantic option', () => { - expect(Versioning.strategies).toHaveProperty('Semantic'); + expect(BuildVersionGenerator.strategies).toHaveProperty('Semantic'); }); it('has a strategy for tags', () => { - expect(Versioning.strategies).toHaveProperty('Tag'); + expect(BuildVersionGenerator.strategies).toHaveProperty('Tag'); }); it('has an option that allows custom input', () => { - expect(Versioning.strategies).toHaveProperty('Custom'); + expect(BuildVersionGenerator.strategies).toHaveProperty('Custom'); }); }); @@ -39,7 +39,7 @@ describe('Versioning', () => { // eslint-disable-next-line unicorn/consistent-function-scoping const matchInputUsingGrep = async (input) => { const output = await System.run('sh', undefined, { - input: Buffer.from(`echo '${input}' | grep -E '${Versioning.grepCompatibleInputVersionRegex}'`), + input: Buffer.from(`echo '${input}' | grep -E '${BuildVersionGenerator.grepCompatibleInputVersionRegex}'`), silent: true, }); @@ -56,35 +56,35 @@ describe('Versioning', () => { }); describe('branch', () => { - it('returns headRef when set', () => { - const headReference = jest.spyOn(Versioning, 'headRef', 'get').mockReturnValue('feature-branch-1'); + it('returns headRef when set', async () => { + const headReference = jest.spyOn(BuildVersionGenerator, 'headRef', 'get').mockReturnValue('feature-branch-1'); - expect(Versioning.branch).toStrictEqual('feature-branch-1'); + await expect(BuildVersionGenerator.getCurrentBranch).resolves.toStrictEqual('feature-branch-1'); expect(headReference).toHaveBeenCalledTimes(1); }); it('returns part of Ref when set', () => { - jest.spyOn(Versioning, 'headRef', 'get').mockImplementation(); - const reference = jest.spyOn(Versioning, 'ref', 'get').mockReturnValue('refs/heads/feature-branch-2'); + jest.spyOn(BuildVersionGenerator, 'headRef', 'get').mockImplementation(); + const reference = jest.spyOn(BuildVersionGenerator, 'ref', 'get').mockReturnValue('refs/heads/feature-branch-2'); - expect(Versioning.branch).toStrictEqual('feature-branch-2'); + await expect(BuildVersionGenerator.getCurrentBranch).resolves.toStrictEqual('feature-branch-2'); expect(reference).toHaveBeenCalledTimes(2); }); it('prefers headRef over ref when set', () => { - const headReference = jest.spyOn(Versioning, 'headRef', 'get').mockReturnValue('feature-branch-1'); - const reference = jest.spyOn(Versioning, 'ref', 'get').mockReturnValue('refs/heads/feature-2'); + const headReference = jest.spyOn(BuildVersionGenerator, 'headRef', 'get').mockReturnValue('feature-branch-1'); + const reference = jest.spyOn(BuildVersionGenerator, 'ref', 'get').mockReturnValue('refs/heads/feature-2'); - expect(Versioning.branch).toStrictEqual('feature-branch-1'); + await expect(BuildVersionGenerator.getCurrentBranch).resolves.toStrictEqual('feature-branch-1'); expect(headReference).toHaveBeenCalledTimes(1); expect(reference).toHaveBeenCalledTimes(0); }); - it('returns undefined when headRef and ref are not set', () => { - const headReference = jest.spyOn(Versioning, 'headRef', 'get').mockImplementation(); - const reference = jest.spyOn(Versioning, 'ref', 'get').mockImplementation(); + it('returns undefined when headRef and ref are not set', async () => { + const headReference = jest.spyOn(BuildVersionGenerator, 'headRef', 'get').mockImplementation(); + const reference = jest.spyOn(BuildVersionGenerator, 'ref', 'get').mockImplementation(); - expect(Versioning.branch).not.toBeDefined(); + await expect(BuildVersionGenerator.getCurrentBranch).resolves.not.toBeDefined(); expect(headReference).toHaveBeenCalledTimes(1); expect(reference).toHaveBeenCalledTimes(1); @@ -93,23 +93,23 @@ describe('Versioning', () => { describe('headRef', () => { it('does not throw', () => { - expect(() => Versioning.headRef).not.toThrow(); + expect(() => BuildVersionGenerator.headRef).not.toThrow(); }); }); describe('ref', () => { it('does not throw', () => { - expect(() => Versioning.ref).not.toThrow(); + expect(() => BuildVersionGenerator.ref).not.toThrow(); }); }); describe('isDirtyAllowed', () => { it('does not throw', () => { - expect(() => Versioning.isDirtyAllowed).not.toThrow(); + expect(() => BuildVersionGenerator.isDirtyAllowed).not.toThrow(); }); it('returns false by default', () => { - expect(Versioning.isDirtyAllowed).toStrictEqual(false); + expect(BuildVersionGenerator.isDirtyAllowed).toStrictEqual(false); }); }); @@ -117,17 +117,17 @@ describe('Versioning', () => { it('calls git diff', async () => { // allowDirtyBuild: true jest.spyOn(core, 'getInput').mockReturnValue('true'); - jest.spyOn(Versioning, 'isShallow').mockResolvedValue(true); - jest.spyOn(Versioning, 'isDirty').mockResolvedValue(false); - jest.spyOn(Versioning, 'fetch').mockImplementation(); - jest.spyOn(Versioning, 'hasAnyVersionTags').mockResolvedValue(true); + jest.spyOn(BuildVersionGenerator, 'isShallow').mockResolvedValue(true); + jest.spyOn(BuildVersionGenerator, 'isDirty').mockResolvedValue(false); + jest.spyOn(BuildVersionGenerator, 'fetch').mockImplementation(); + jest.spyOn(BuildVersionGenerator, 'hasAnyVersionTags').mockResolvedValue(true); jest - .spyOn(Versioning, 'parseSemanticVersion') + .spyOn(BuildVersionGenerator, 'parseSemanticVersion') .mockResolvedValue({ match: '', tag: 'mocktag', commits: 'abcdef', hash: '75822BCAF' }); - const logDiffSpy = jest.spyOn(Versioning, 'logDiff'); + const logDiffSpy = jest.spyOn(BuildVersionGenerator, 'logDiff'); const gitSpy = jest.spyOn(System, 'run').mockImplementation(); - await Versioning.generateSemanticVersion(); + await BuildVersionGenerator.generateSemanticVersion(); expect(logDiffSpy).toHaveBeenCalledTimes(1); expect(gitSpy).toHaveBeenCalledTimes(1); @@ -140,39 +140,39 @@ describe('Versioning', () => { describe('descriptionRegex1', () => { it('is a valid regex', () => { - expect(Versioning.descriptionRegex1).toBeInstanceOf(RegExp); + expect(BuildVersionGenerator.descriptionRegex1).toBeInstanceOf(RegExp); }); test.each(['v1.1-1-g12345678', 'v0.1-2-g12345678', 'v0.0-500-gA9B6C3D0-dirty'])( 'is happy with valid %s', (description) => { - expect(Versioning.descriptionRegex1.test(description)).toBeTruthy(); + expect(BuildVersionGenerator.descriptionRegex1.test(description)).toBeTruthy(); }, ); test.each(['1.1-1-g12345678', '0.1-2-g12345678', '0.0-500-gA9B6C3D0-dirty'])( 'accepts valid semantic versions without v-prefix %s', (description) => { - expect(Versioning.descriptionRegex1.test(description)).toBeTruthy(); + expect(BuildVersionGenerator.descriptionRegex1.test(description)).toBeTruthy(); }, ); test.each(['v0', 'v0.1', 'v0.1.2', 'v0.1-2', 'v0.1-2-g'])('does not like %s', (description) => { - expect(Versioning.descriptionRegex1.test(description)).toBeFalsy(); + expect(BuildVersionGenerator.descriptionRegex1.test(description)).toBeFalsy(); // Also, never expect without the v to work for any of these cases. - expect(Versioning.descriptionRegex1.test(description?.slice(1))).toBeFalsy(); + expect(BuildVersionGenerator.descriptionRegex1.test(description?.slice(1))).toBeFalsy(); }); }); describe('determineBuildVersion', () => { test.each(['somethingRandom'])('throws for invalid strategy %s', async (strategy) => { - await expect(Versioning.determineBuildVersion(strategy, '')).rejects.toThrowErrorMatchingSnapshot(); + await expect(BuildVersionGenerator.determineBuildVersion(strategy, '')).rejects.toThrowErrorMatchingSnapshot(); }); describe('opt out strategy', () => { it("returns 'none'", async () => { - await expect(Versioning.determineBuildVersion('None', 'v1.0')).resolves.toMatchInlineSnapshot(`"none"`); + await expect(BuildVersionGenerator.determineBuildVersion('None', 'v1.0')).resolves.toMatchInlineSnapshot(`"none"`); }); }); @@ -180,25 +180,25 @@ describe('Versioning', () => { test.each(['v0.1', '1', 'CamelCase', 'dashed-version'])( 'returns the inputVersion for %s', async (inputVersion) => { - await expect(Versioning.determineBuildVersion('Custom', inputVersion)).resolves.toStrictEqual(inputVersion); + await expect(BuildVersionGenerator.determineBuildVersion('Custom', inputVersion)).resolves.toStrictEqual(inputVersion); }, ); }); describe('semantic strategy', () => { it('refers to generateSemanticVersion', async () => { - const generateSemanticVersion = jest.spyOn(Versioning, 'generateSemanticVersion').mockResolvedValue('1.3.37'); + const generateSemanticVersion = jest.spyOn(BuildVersionGenerator, 'generateSemanticVersion').mockResolvedValue('1.3.37'); - await expect(Versioning.determineBuildVersion('Semantic', '')).resolves.toStrictEqual('1.3.37'); + await expect(BuildVersionGenerator.determineBuildVersion('Semantic', '')).resolves.toStrictEqual('1.3.37'); expect(generateSemanticVersion).toHaveBeenCalledTimes(1); }); }); describe('tag strategy', () => { it('refers to generateTagVersion', async () => { - const generateTagVersion = jest.spyOn(Versioning, 'generateTagVersion').mockResolvedValue('0.1'); + const generateTagVersion = jest.spyOn(BuildVersionGenerator, 'generateTagVersion').mockResolvedValue('0.1'); - await expect(Versioning.determineBuildVersion('Tag', '')).resolves.toStrictEqual('0.1'); + await expect(BuildVersionGenerator.determineBuildVersion('Tag', '')).resolves.toStrictEqual('0.1'); expect(generateTagVersion).toHaveBeenCalledTimes(1); }); }); @@ -207,24 +207,24 @@ describe('Versioning', () => { it('throws a not implemented exception', async () => { const strategy = 'Test'; // @ts-ignore - jest.spyOn(Versioning, 'strategies', 'get').mockReturnValue({ [strategy]: strategy }); - await expect(Versioning.determineBuildVersion(strategy, '')).rejects.toThrowError(NotImplementedException); + jest.spyOn(BuildVersionGenerator, 'strategies', 'get').mockReturnValue({ [strategy]: strategy }); + await expect(BuildVersionGenerator.determineBuildVersion(strategy, '')).rejects.toThrowError(NotImplementedException); }); }); }); describe('generateTagVersion', () => { it('removes the v', async () => { - jest.spyOn(Versioning, 'getTag').mockResolvedValue('v1.3.37'); - await expect(Versioning.generateTagVersion()).resolves.toStrictEqual('1.3.37'); + jest.spyOn(BuildVersionGenerator, 'getTag').mockResolvedValue('v1.3.37'); + await expect(BuildVersionGenerator.generateTagVersion()).resolves.toStrictEqual('1.3.37'); }); }); describe('parseSemanticVersion', () => { it('returns the named parts', async () => { - jest.spyOn(Versioning, 'getVersionDescription').mockResolvedValue('v0.1-2-g12345678'); + jest.spyOn(BuildVersionGenerator, 'getVersionDescription').mockResolvedValue('v0.1-2-g12345678'); - await expect(Versioning.parseSemanticVersion()).resolves.toMatchObject({ + await expect(BuildVersionGenerator.parseSemanticVersion()).resolves.toMatchObject({ tag: '0.1', commits: '2', hash: '12345678', @@ -232,9 +232,9 @@ describe('Versioning', () => { }); it('throws when no match could be made', async () => { - jest.spyOn(Versioning, 'getVersionDescription').mockResolvedValue('no-match-can-be-made'); + jest.spyOn(BuildVersionGenerator, 'getVersionDescription').mockResolvedValue('no-match-can-be-made'); - await expect(Versioning.parseSemanticVersion()).toMatchObject({}); + await expect(BuildVersionGenerator.parseSemanticVersion()).toMatchObject({}); }); }); @@ -242,7 +242,7 @@ describe('Versioning', () => { it('returns the commands output', async () => { const runOutput = 'someValue'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.getVersionDescription()).resolves.toStrictEqual(runOutput); + await expect(BuildVersionGenerator.getVersionDescription()).resolves.toStrictEqual(runOutput); }); }); @@ -250,13 +250,13 @@ describe('Versioning', () => { it('returns true when the repo is shallow', async () => { const runOutput = 'true\n'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.isShallow()).resolves.toStrictEqual(true); + await expect(BuildVersionGenerator.isShallow()).resolves.toStrictEqual(true); }); it('returns false when the repo is not shallow', async () => { const runOutput = 'false\n'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.isShallow()).resolves.toStrictEqual(false); + await expect(BuildVersionGenerator.isShallow()).resolves.toStrictEqual(false); }); }); @@ -264,14 +264,14 @@ describe('Versioning', () => { it('awaits the command', async () => { jest.spyOn(core, 'warning').mockImplementation(() => {}); jest.spyOn(System, 'run').mockImplementation(); - await expect(Versioning.fetch()).resolves.not.toThrow(); + await expect(BuildVersionGenerator.fetch()).resolves.not.toThrow(); }); it('falls back to the second strategy when the first fails', async () => { jest.spyOn(core, 'warning').mockImplementation(() => {}); const gitFetch = jest.spyOn(System, 'run').mockImplementation(); - await expect(Versioning.fetch()).resolves.not.toThrow(); + await expect(BuildVersionGenerator.fetch()).resolves.not.toThrow(); expect(gitFetch).toHaveBeenCalledTimes(1); }); }); @@ -280,35 +280,35 @@ describe('Versioning', () => { it('returns a proper version from description', async () => { jest.spyOn(System, 'run').mockImplementation(); jest.spyOn(core, 'info').mockImplementation(() => {}); - jest.spyOn(Versioning, 'isDirty').mockResolvedValue(false); - jest.spyOn(Versioning, 'hasAnyVersionTags').mockResolvedValue(true); - jest.spyOn(Versioning, 'getTotalNumberOfCommits').mockResolvedValue(2); - jest.spyOn(Versioning, 'parseSemanticVersion').mockResolvedValue({ + jest.spyOn(BuildVersionGenerator, 'isDirty').mockResolvedValue(false); + jest.spyOn(BuildVersionGenerator, 'hasAnyVersionTags').mockResolvedValue(true); + jest.spyOn(BuildVersionGenerator, 'getTotalNumberOfCommits').mockResolvedValue(2); + jest.spyOn(BuildVersionGenerator, 'parseSemanticVersion').mockResolvedValue({ match: '0.1-2-g1b345678', tag: '0.1', commits: '2', hash: '1b345678', }); - await expect(Versioning.generateSemanticVersion()).resolves.toStrictEqual('0.1.2'); + await expect(BuildVersionGenerator.generateSemanticVersion()).resolves.toStrictEqual('0.1.2'); }); it('throws when dirty', async () => { jest.spyOn(System, 'run').mockImplementation(); jest.spyOn(core, 'info').mockImplementation(() => {}); - jest.spyOn(Versioning, 'isDirty').mockResolvedValue(true); - await expect(Versioning.generateSemanticVersion()).rejects.toThrowError(); + jest.spyOn(BuildVersionGenerator, 'isDirty').mockResolvedValue(true); + await expect(BuildVersionGenerator.generateSemanticVersion()).rejects.toThrowError(); }); it('falls back to commits only, when no tags are present', async () => { const commits = Math.round(Math.random() * 10); jest.spyOn(System, 'run').mockImplementation(); jest.spyOn(core, 'info').mockImplementation(() => {}); - jest.spyOn(Versioning, 'isDirty').mockResolvedValue(false); - jest.spyOn(Versioning, 'hasAnyVersionTags').mockResolvedValue(false); - jest.spyOn(Versioning, 'getTotalNumberOfCommits').mockResolvedValue(commits); + jest.spyOn(BuildVersionGenerator, 'isDirty').mockResolvedValue(false); + jest.spyOn(BuildVersionGenerator, 'hasAnyVersionTags').mockResolvedValue(false); + jest.spyOn(BuildVersionGenerator, 'getTotalNumberOfCommits').mockResolvedValue(commits); - await expect(Versioning.generateSemanticVersion()).resolves.toStrictEqual(`0.0.${commits}`); + await expect(BuildVersionGenerator.generateSemanticVersion()).resolves.toStrictEqual(`0.0.${commits}`); }); }); @@ -316,13 +316,13 @@ describe('Versioning', () => { it('returns true when there are files listed', async () => { const runOutput = 'file.ext\nfile2.ext'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.isDirty()).resolves.toStrictEqual(true); + await expect(BuildVersionGenerator.isDirty()).resolves.toStrictEqual(true); }); it('returns false when there is no output', async () => { const runOutput = ''; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.isDirty()).resolves.toStrictEqual(false); + await expect(BuildVersionGenerator.isDirty()).resolves.toStrictEqual(false); }); }); @@ -330,7 +330,7 @@ describe('Versioning', () => { it('returns the commands output', async () => { const runOutput = 'v1.0'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.getTag()).resolves.toStrictEqual(runOutput); + await expect(BuildVersionGenerator.getTag()).resolves.toStrictEqual(runOutput); }); }); @@ -338,20 +338,20 @@ describe('Versioning', () => { it('returns false when the command returns 0', async () => { const runOutput = '0'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.hasAnyVersionTags()).resolves.toStrictEqual(false); + await expect(BuildVersionGenerator.hasAnyVersionTags()).resolves.toStrictEqual(false); }); it('returns true when the command returns >= 0', async () => { const runOutput = '9'; jest.spyOn(System, 'run').mockResolvedValue(runOutput); - await expect(Versioning.hasAnyVersionTags()).resolves.toStrictEqual(true); + await expect(BuildVersionGenerator.hasAnyVersionTags()).resolves.toStrictEqual(true); }); }); describe('getTotalNumberOfCommits', () => { it('returns a number from the command', async () => { jest.spyOn(System, 'run').mockResolvedValue('9'); - await expect(Versioning.getTotalNumberOfCommits()).resolves.toStrictEqual(9); + await expect(BuildVersionGenerator.getTotalNumberOfCommits()).resolves.toStrictEqual(9); }); }); }); diff --git a/src/module/actions/core.ts b/src/module/actions/core.ts new file mode 100644 index 000000000..5470bad5a --- /dev/null +++ b/src/module/actions/core.ts @@ -0,0 +1,31 @@ +/* eslint-disable no-console */ + +export const core = { + info: console.log, + + warning: console.warn, + + error: (error) => { + console.error(error, error.stack); + }, + + setOutput: (key, value) => { + console.log(`(mock) Output "${key}" is set to "${value}"`); + }, + + // Adapted from: https://github.com/actions/toolkit/blob/9b7bcb1567c9b7f134eb3c2d6bbf409a5106a956/packages/core/src/core.ts#L128 + getInput: (name, options) => { + const variable = `INPUT_${name.replace(/ /g, '_').toUpperCase()}`; + const value: string = Deno.env.get(variable) || ''; + + if (options?.required && !value) { + throw new Error(`Input required and not supplied: ${name}`); + } + + if (options?.trimWhitespace === false) { + return value; + } + + return value.trim(); + }, +}; diff --git a/src/module/actions/index.ts b/src/module/actions/index.ts new file mode 100644 index 000000000..d5ec94afc --- /dev/null +++ b/src/module/actions/index.ts @@ -0,0 +1,5 @@ +/** + * The actions/core package from GitHub errors out. + * This substitutes the parts we use in a Deno-compatible way. + */ +export { core } from './core.ts'; diff --git a/src/module/dedent.ts b/src/module/dedent.ts new file mode 100644 index 000000000..bc7c8add1 --- /dev/null +++ b/src/module/dedent.ts @@ -0,0 +1,54 @@ +/* eslint-disable unicorn/no-array-reduce,unicorn/explicit-length-check,padding-line-between-statements,github/array-foreach,unicorn/no-array-for-each */ + +// Source: https://github.com/tamino-martinius/node-ts-dedent/blob/master/src/index.ts +export function dedent(template: TemplateStringsArray | string, ...values: unknown[]): string { + let strings = Array.from(typeof template === 'string' ? [template] : template); + + // 1. Remove trailing whitespace. + strings[strings.length - 1] = strings[strings.length - 1].replace(/\r?\n([\t ]*)$/, ''); + + // 2. Find all line breaks to determine the highest common indentation level. + const indentLengths = strings.reduce((arr, str) => { + const matches = str.match(/\n([\t ]+|(?!\s).)/g); + if (matches) { + return arr.concat(matches.map((match) => match.match(/[\t ]/g)?.length ?? 0)); + } + return arr; + }, []); + + // 3. Remove the common indentation from all strings. + if (indentLengths.length) { + const pattern = new RegExp(`\n[\t ]{${Math.min(...indentLengths)}}`, 'g'); + + strings = strings.map((str) => str.replace(pattern, '\n')); + } + + // 4. Remove leading whitespace. + strings[0] = strings[0].replace(/^\r?\n/, ''); + + // 5. Perform interpolation. + let string = strings[0]; + + values.forEach((value, i) => { + // 5.1 Read current indentation level + const indentations = string.match(/(?:^|\n)( *)$/); + const indentation = indentations ? indentations[1] : ''; + let indentedValue = value; + + // 5.2 Add indentation to values with multiline strings + if (typeof value === 'string' && value.includes('\n')) { + indentedValue = String(value) + .split('\n') + .map((str, i) => { + return i === 0 ? str : `${indentation}${str}`; + }) + .join('\n'); + } + + string += indentedValue + strings[i + 1]; + }); + + return string; +} + +export default dedent; diff --git a/src/module/wait-until.ts b/src/module/wait-until.ts new file mode 100644 index 000000000..6dbac9fc0 --- /dev/null +++ b/src/module/wait-until.ts @@ -0,0 +1,281 @@ +/* eslint-disable github/no-then,lines-around-comment */ +// Source: https://github.com/devlato/async-wait-until/blob/master/src/index.ts + +/** + * This module implements a function that waits for a given predicate to be truthy. + * Relies on Promises and supports async/await. + * @packageDocumentation + * @module async-wait-until + */ + +/** + * @type Error JavaScript's generic Error type + * @public + */ + +/** + * Timeout error, which is thrown when timeout passes but the predicate + * doesn't resolve with a truthy value + * @public + * @class + * @exception + * @category Exceptions + */ +export class TimeoutError extends Error { + /** + * Creates a TimeoutError instance + * @public + * @param timeoutInMs Expected timeout, in milliseconds + */ + constructor(timeoutInMs?: number) { + super(timeoutInMs != null ? `Timed out after waiting for ${timeoutInMs} ms` : 'Timed out'); + + Object.setPrototypeOf(this, TimeoutError.prototype); + } +} + +/** + * A utility function for cross-platform type-safe scheduling + * @private + * @returns Returns a proper scheduler instance depending on the current environment + * @throws Error + * @category Utilities + */ +const getScheduler = (): Scheduler => ({ + schedule: (fn, interval) => { + let scheduledTimer: number | undefined; + + const cleanUp = (timer: number | undefined) => { + if (timer != null) { + clearTimeout(timer as number); + } + + scheduledTimer = undefined; + }; + + const iteration = () => { + cleanUp(scheduledTimer); + fn(); + }; + + scheduledTimer = setTimeout(iteration, interval); + + return { + cancel: () => cleanUp(scheduledTimer), + }; + }, +}); + +/** + * Delays the execution by the given interval, in milliseconds + * @private + * @param scheduler A scheduler instance + * @param interval An interval to wait for before resolving the Promise, in milliseconds + * @returns A Promise that gets resolved once the given interval passes + * @throws Error + * @category Utilities + */ +const delay = (scheduler: Scheduler, interval: number): Promise => + new Promise((resolve, reject) => { + try { + scheduler.schedule(resolve, interval); + } catch (error) { + reject(error); + } + }); + +/** + * Platform-specific scheduler + * @private + * @category Defaults + */ +const SCHEDULER: Scheduler = getScheduler(); + +/** + * Default interval between attempts, in milliseconds + * @public + * @category Defaults + */ +export const DEFAULT_INTERVAL_BETWEEN_ATTEMPTS_IN_MS = 50; + +/** + * Default timeout, in milliseconds + * @public + * @category Defaults + */ +export const DEFAULT_TIMEOUT_IN_MS = 5000; + +/** + * Timeout that represents infinite wait time + * @public + * @category Defaults + */ +export const WAIT_FOREVER = Number.POSITIVE_INFINITY; + +/** + * Waits for predicate to be truthy and resolves a Promise + * @public + * @param predicate A predicate function that checks the condition, it should return either a truthy value or a falsy value + * @param options Options object (or *(deprecated)*: a maximum wait interval, *5000 ms* by default) + * @param intervalBetweenAttempts *(deprecated)* Interval to wait for between attempts, optional, *50 ms* by default + * @returns A promise to return the given predicate's result, once it resolves with a truthy value + * @template T Result type for the truthy value returned by the predicate + * @throws [[TimeoutError]] An exception thrown when the specified timeout interval passes but the predicate doesn't return a truthy value + * @throws Error + * @see [[TruthyValue]] + * @see [[FalsyValue]] + * @see [[Options]] + */ +export const waitUntil = ( + predicate: Predicate, + options?: number | Options, + intervalBetweenAttempts?: number, +): Promise => { + const timerTimeout = (typeof options === 'number' ? options : options?.timeout) ?? DEFAULT_TIMEOUT_IN_MS; + const timerIntervalBetweenAttempts = + (typeof options === 'number' ? intervalBetweenAttempts : options?.intervalBetweenAttempts) ?? + DEFAULT_INTERVAL_BETWEEN_ATTEMPTS_IN_MS; + + const runPredicate = (): Promise>> => + new Promise((resolve, reject) => { + try { + resolve(predicate()); + } catch (error) { + reject(error); + } + }); + + let isTimedOut = false; + + const predicatePromise = (): Promise => + new Promise((resolve, reject) => { + const iteration = () => { + if (isTimedOut) { + return; + } + + runPredicate() + .then((result) => { + if (result) { + resolve(result); + + return; + } + + delay(SCHEDULER, timerIntervalBetweenAttempts).then(iteration).catch(reject); + }) + .catch(reject); + }; + + iteration(); + }); + + const timeoutPromise = + timerTimeout !== WAIT_FOREVER + ? () => + delay(SCHEDULER, timerTimeout).then(() => { + isTimedOut = true; + throw new TimeoutError(timerTimeout); + }) + : undefined; + + return timeoutPromise != null ? Promise.race([predicatePromise(), timeoutPromise()]) : predicatePromise(); +}; + +/** + * The predicate type + * @private + * @template T Returned value type, either a truthy value or a falsy value + * @throws Error + * @category Common Types + * @see [[TruthyValue]] + * @see [[FalsyValue]] + */ +export type Predicate = () => T | Promise; + +/** + * A type that represents a falsy value + * @private + * @category Common Types + */ +export type FalsyValue = null | undefined | false | '' | 0 | void; + +/** + * A type that represents a truthy value + * @private + * @category Common Types + */ +export type TruthyValue = + | Record + | unknown[] + | symbol + | ((...args: unknown[]) => unknown) + | Exclude + | Exclude + | true; + +/** + * A type that represents a Predicate's return value + * @private + * @category Common Types + */ +export type PredicateReturnValue = TruthyValue | FalsyValue; + +/** + * Options that allow to specify timeout or time interval between consecutive attempts + * @public + * @category Common Types + */ +export type Options = { + // Maximum wait interval, *5000 ms* by default + timeout?: number; + + // Interval to wait for between attempts, optional, *50 ms* by default + intervalBetweenAttempts?: number; +}; + +/** + * A function that schedules a given callback to run in given number of milliseconds + * @private + * @param callback A callback to execute + * @param interval A time interval to wait before executing the callback + * @returns An instance of ScheduleCanceler that allows to cancel the scheduled callback's execution + * @template T The callback params' type + * @throws Error + * @category Common Types + */ +type ScheduleFn = (callback: (...args: T[]) => void, interval: number) => ScheduleCanceler; + +/** + * A function that cancels the previously scheduled callback's execution + * @private + * @throws Error + * @category Common Types + */ +type CancelScheduledFn = () => void; + +/** + * A stateful abstraction over Node.js & web browser timers that cancels the scheduled task + * @private + * @category Common Types + */ +type ScheduleCanceler = { + /** + * @property A function that cancels the previously scheduled callback's execution + */ + cancel: CancelScheduledFn; +}; + +/** + * A stateful abstraction over Node.js & web browser timers that schedules a task + * @private + * @category Common Types + */ +type Scheduler = { + /** + * @property A function that schedules a given callback to run in given number of milliseconds + */ + schedule: ScheduleFn; +}; + +export default waitUntil; diff --git a/test-project/Assets/LFS_Test_File.jpg.meta b/test-project/Assets/LFS_Test_File.jpg.meta new file mode 100644 index 000000000..d70787508 --- /dev/null +++ b/test-project/Assets/LFS_Test_File.jpg.meta @@ -0,0 +1,92 @@ +fileFormatVersion: 2 +guid: 1fa26fa0ad51a6c4aa77c419d0dfb797 +TextureImporter: + internalIDToNameTable: [] + externalObjects: {} + serializedVersion: 11 + mipmaps: + mipMapMode: 0 + enableMipMap: 0 + sRGBTexture: 1 + linearTexture: 0 + fadeOut: 0 + borderMipMap: 0 + mipMapsPreserveCoverage: 0 + alphaTestReferenceValue: 0.5 + mipMapFadeDistanceStart: 1 + mipMapFadeDistanceEnd: 3 + bumpmap: + convertToNormalMap: 0 + externalNormalMap: 0 + heightScale: 0.25 + normalMapFilter: 0 + isReadable: 0 + streamingMipmaps: 0 + streamingMipmapsPriority: 0 + grayScaleToAlpha: 0 + generateCubemap: 6 + cubemapConvolution: 0 + seamlessCubemap: 0 + textureFormat: 1 + maxTextureSize: 2048 + textureSettings: + serializedVersion: 2 + filterMode: 1 + aniso: 1 + mipBias: 0 + wrapU: 1 + wrapV: 1 + wrapW: 1 + nPOTScale: 0 + lightmap: 0 + compressionQuality: 50 + spriteMode: 1 + spriteExtrude: 1 + spriteMeshType: 1 + alignment: 0 + spritePivot: {x: 0.5, y: 0.5} + spritePixelsToUnits: 100 + spriteBorder: {x: 0, y: 0, z: 0, w: 0} + spriteGenerateFallbackPhysicsShape: 1 + alphaUsage: 1 + alphaIsTransparency: 1 + spriteTessellationDetail: -1 + textureType: 8 + textureShape: 1 + singleChannelComponent: 0 + maxTextureSizeSet: 0 + compressionQualitySet: 0 + textureFormatSet: 0 + applyGammaDecoding: 0 + platformSettings: + - serializedVersion: 3 + buildTarget: DefaultTexturePlatform + maxTextureSize: 2048 + resizeAlgorithm: 0 + textureFormat: -1 + textureCompression: 1 + compressionQuality: 50 + crunchedCompression: 0 + allowsAlphaSplitting: 0 + overridden: 0 + androidETC2FallbackOverride: 0 + forceMaximumCompressionQuality_BC6H_BC7: 0 + spriteSheet: + serializedVersion: 2 + sprites: [] + outline: [] + physicsShape: [] + bones: [] + spriteID: 5e97eb03825dee720800000000000000 + internalID: 0 + vertices: [] + indices: + edges: [] + weights: [] + secondaryTextures: [] + spritePackingTag: + pSDRemoveMatte: 0 + pSDShowRemoveMatteOption: 0 + userData: + assetBundleName: + assetBundleVariant: diff --git a/test-project/Packages/manifest.json b/test-project/Packages/manifest.json index 18a1e415e..9ae0c8007 100644 --- a/test-project/Packages/manifest.json +++ b/test-project/Packages/manifest.json @@ -1,3 +1,5 @@ { - "dependencies": {} + "dependencies": { + "com.unity.ide.visualstudio": "2.0.15" + } } diff --git a/test-project/Packages/packages-lock.json b/test-project/Packages/packages-lock.json new file mode 100644 index 000000000..4520f3d57 --- /dev/null +++ b/test-project/Packages/packages-lock.json @@ -0,0 +1,43 @@ +{ + "dependencies": { + "com.unity.ext.nunit": { + "version": "1.0.6", + "depth": 2, + "source": "registry", + "dependencies": {}, + "url": "https://packages.unity.com" + }, + "com.unity.ide.visualstudio": { + "version": "2.0.15", + "depth": 0, + "source": "registry", + "dependencies": { + "com.unity.test-framework": "1.1.9" + }, + "url": "https://packages.unity.com" + }, + "com.unity.test-framework": { + "version": "1.1.31", + "depth": 1, + "source": "registry", + "dependencies": { + "com.unity.ext.nunit": "1.0.6", + "com.unity.modules.imgui": "1.0.0", + "com.unity.modules.jsonserialize": "1.0.0" + }, + "url": "https://packages.unity.com" + }, + "com.unity.modules.imgui": { + "version": "1.0.0", + "depth": 2, + "source": "builtin", + "dependencies": {} + }, + "com.unity.modules.jsonserialize": { + "version": "1.0.0", + "depth": 2, + "source": "builtin", + "dependencies": {} + } + } +} diff --git a/test-project/ProjectSettings/PackageManagerSettings.asset b/test-project/ProjectSettings/PackageManagerSettings.asset new file mode 100644 index 000000000..6920e3adb --- /dev/null +++ b/test-project/ProjectSettings/PackageManagerSettings.asset @@ -0,0 +1,38 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!114 &1 +MonoBehaviour: + m_ObjectHideFlags: 61 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 13964, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_ScopedRegistriesSettingsExpanded: 1 + oneTimeWarningShown: 0 + m_Registries: + - m_Id: main + m_Name: + m_Url: https://packages.unity.com + m_Scopes: [] + m_IsDefault: 1 + m_UserSelectedRegistryName: + m_UserAddingNewScopedRegistry: 0 + m_RegistryInfoDraft: + m_ErrorMessage: + m_Original: + m_Id: + m_Name: + m_Url: + m_Scopes: [] + m_IsDefault: 0 + m_Modified: 0 + m_Name: + m_Url: + m_Scopes: + - + m_SelectedScopeIndex: 0 diff --git a/test-project/ProjectSettings/ProjectVersion.txt b/test-project/ProjectSettings/ProjectVersion.txt index 87079c135..4c191298a 100644 --- a/test-project/ProjectSettings/ProjectVersion.txt +++ b/test-project/ProjectSettings/ProjectVersion.txt @@ -1,2 +1,2 @@ -m_EditorVersion: 2019.2.11f1 -m_EditorVersionWithRevision: 2019.2.11f1 (5f859a4cfee5) +m_EditorVersion: 2019.4.40f1 +m_EditorVersionWithRevision: 2019.4.40f1 (ffc62b691db5) diff --git a/yarn.lock b/yarn.lock index b6ef43470..e4877d5b6 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2,50 +2,6 @@ # yarn lockfile v1 -"@actions/core@^1.9.1": - version "1.9.1" - resolved "https://registry.yarnpkg.com/@actions/core/-/core-1.9.1.tgz#97c0201b1f9856df4f7c3a375cdcdb0c2a2f750b" - integrity sha512-5ad+U2YGrmmiw6du20AQW5XuWo7UKN2052FjSV7MX+Wfjf8sCqcsZe62NfgHys4QI4/Y+vQvLKYL8jWtA1ZBTA== - dependencies: - "@actions/http-client" "^2.0.1" - uuid "^8.3.2" - -"@actions/exec@^1.1.0": - version "1.1.0" - resolved "https://registry.yarnpkg.com/@actions/exec/-/exec-1.1.0.tgz#53441d968e56d2fec69ad3f15773d4d94e01162c" - integrity sha512-LImpN9AY0J1R1mEYJjVJfSZWU4zYOlEcwSTgPve1rFQqK5AwrEs6uWW5Rv70gbDIQIAUwI86z6B+9mPK4w9Sbg== - dependencies: - "@actions/io" "^1.0.1" - -"@actions/github@^5.0.0": - version "5.0.0" - resolved "https://registry.yarnpkg.com/@actions/github/-/github-5.0.0.tgz#1754127976c50bd88b2e905f10d204d76d1472f8" - integrity sha512-QvE9eAAfEsS+yOOk0cylLBIO/d6WyWIOvsxxzdrPFaud39G6BOkUwScXZn1iBzQzHyu9SBkkLSWlohDWdsasAQ== - dependencies: - "@actions/http-client" "^1.0.11" - "@octokit/core" "^3.4.0" - "@octokit/plugin-paginate-rest" "^2.13.3" - "@octokit/plugin-rest-endpoint-methods" "^5.1.1" - -"@actions/http-client@^1.0.11": - version "1.0.11" - resolved "https://registry.yarnpkg.com/@actions/http-client/-/http-client-1.0.11.tgz#c58b12e9aa8b159ee39e7dd6cbd0e91d905633c0" - integrity sha512-VRYHGQV1rqnROJqdMvGUbY/Kn8vriQe/F9HR2AlYHzmKuM/p3kjNuXhmdBfcVgsvRWTz5C5XW5xvndZrVBuAYg== - dependencies: - tunnel "0.0.6" - -"@actions/http-client@^2.0.1": - version "2.0.1" - resolved "https://registry.yarnpkg.com/@actions/http-client/-/http-client-2.0.1.tgz#873f4ca98fe32f6839462a6f046332677322f99c" - integrity sha512-PIXiMVtz6VvyaRsGY268qvj57hXQEpsYogYOu2nrQhlf+XCGmZstmuZBbAybUl1nQGnvS1k1eEsQ69ZoD7xlSw== - dependencies: - tunnel "^0.0.6" - -"@actions/io@^1.0.1": - version "1.1.0" - resolved "https://registry.npmjs.org/@actions/io/-/io-1.1.0.tgz" - integrity sha512-PspSX7Z9zh2Fyyuf3F6BsYeXcYHfc/VJ1vwy2vouas95efHVd42M6UfBFRs+jY0uiMDXhAoUtATn9g2r1MaWBQ== - "@ampproject/remapping@^2.1.0": version "2.1.2" resolved "https://registry.yarnpkg.com/@ampproject/remapping/-/remapping-2.1.2.tgz#4edca94973ded9630d20101cd8559cedb8d8bd34" @@ -72,38 +28,45 @@ dependencies: "@babel/highlight" "^7.16.7" -"@babel/compat-data@^7.13.15": - version "7.13.15" - resolved "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.13.15.tgz" - integrity sha512-ltnibHKR1VnrU4ymHyQ/CXtNXI6yZC0oJThyW78Hft8XndANwi+9H+UIklBDraIjFEJzw8wmcM427oDd9KS5wA== +"@babel/code-frame@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.18.6.tgz#3b25d38c89600baa2dcc219edfa88a74eb2c427a" + integrity sha512-TDCmlK5eOvH+eH7cdAFlNXeVJqWIQ7gW9tY1GJIpUtFb6CmjVyq2VM3u71bOyR8CRihcCgMUYoDNyLXao3+70Q== + dependencies: + "@babel/highlight" "^7.18.6" "@babel/compat-data@^7.16.4": version "7.17.0" resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.17.0.tgz#86850b8597ea6962089770952075dcaabb8dba34" integrity sha512-392byTlpGWXMv4FbyWw3sAZ/FrW/DrwqLGXpy0mbyNe9Taqv1mg9yON5/o0cnr8XYCkFTZbC1eV+c+LAROgrng== -"@babel/core@^7.1.0", "@babel/core@^7.12.16": - version "7.13.16" - resolved "https://registry.npmjs.org/@babel/core/-/core-7.13.16.tgz" - integrity sha512-sXHpixBiWWFti0AV2Zq7avpTasr6sIAu7Y396c608541qAU2ui4a193m0KSQmfPSKFZLnQ3cvlKDOm3XkuXm3Q== +"@babel/compat-data@^7.18.8": + version "7.18.8" + resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.18.8.tgz#2483f565faca607b8535590e84e7de323f27764d" + integrity sha512-HSmX4WZPPK3FUxYp7g2T6EyO8j96HlZJlxmKPSh6KAcqwyDrfx7hKjXpAW/0FhFfTJsR0Yt4lAjLI2coMptIHQ== + +"@babel/core@^7.11.6": + version "7.18.10" + resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.18.10.tgz#39ad504991d77f1f3da91be0b8b949a5bc466fb8" + integrity sha512-JQM6k6ENcBFKVtWvLavlvi/mPcpYZ3+R+2EySDEMSMbp7Mn4FexlbbJVrx2R7Ijhr01T8gyqrOaABWIOgxeUyw== dependencies: - "@babel/code-frame" "^7.12.13" - "@babel/generator" "^7.13.16" - "@babel/helper-compilation-targets" "^7.13.16" - "@babel/helper-module-transforms" "^7.13.14" - "@babel/helpers" "^7.13.16" - "@babel/parser" "^7.13.16" - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.15" - "@babel/types" "^7.13.16" + "@ampproject/remapping" "^2.1.0" + "@babel/code-frame" "^7.18.6" + "@babel/generator" "^7.18.10" + "@babel/helper-compilation-targets" "^7.18.9" + "@babel/helper-module-transforms" "^7.18.9" + "@babel/helpers" "^7.18.9" + "@babel/parser" "^7.18.10" + "@babel/template" "^7.18.10" + "@babel/traverse" "^7.18.10" + "@babel/types" "^7.18.10" convert-source-map "^1.7.0" debug "^4.1.0" gensync "^1.0.0-beta.2" - json5 "^2.1.2" + json5 "^2.2.1" semver "^6.3.0" - source-map "^0.5.0" -"@babel/core@^7.12.3", "@babel/core@^7.7.2", "@babel/core@^7.8.0": +"@babel/core@^7.12.3": version "7.17.5" resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.17.5.tgz#6cd2e836058c28f06a4ca8ee7ed955bbf37c8225" integrity sha512-/BBMw4EvjmyquN5O+t5eh0+YqB3XXJkYD2cjKpYtWOfFy4lQ4UozNSmxAcWT8r2XtZs0ewG+zrfsqeR15i1ajA== @@ -124,24 +87,6 @@ json5 "^2.1.2" semver "^6.3.0" -"@babel/eslint-parser@^7.12.16": - version "7.13.14" - resolved "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.13.14.tgz" - integrity sha512-I0HweR36D73Ibn/FfrRDMKlMqJHFwidIUgYdMpH+aXYuQC+waq59YaJ6t9e9N36axJ82v1jR041wwqDrDXEwRA== - dependencies: - eslint-scope "^5.1.0" - eslint-visitor-keys "^1.3.0" - semver "^6.3.0" - -"@babel/generator@^7.13.16", "@babel/generator@^7.13.9": - version "7.13.16" - resolved "https://registry.npmjs.org/@babel/generator/-/generator-7.13.16.tgz" - integrity sha512-grBBR75UnKOcUWMp8WoDxNsWCFl//XCK6HWTrBQKTr5SV9f5g0pNOjdyzi/DTBv12S9GnYPInIXQBTky7OXEMg== - dependencies: - "@babel/types" "^7.13.16" - jsesc "^2.5.1" - source-map "^0.5.0" - "@babel/generator@^7.17.3", "@babel/generator@^7.7.2": version "7.17.3" resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.17.3.tgz#a2c30b0c4f89858cb87050c3ffdfd36bdf443200" @@ -151,15 +96,14 @@ jsesc "^2.5.1" source-map "^0.5.0" -"@babel/helper-compilation-targets@^7.13.16": - version "7.13.16" - resolved "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.13.16.tgz" - integrity sha512-3gmkYIrpqsLlieFwjkGgLaSHmhnvlAYzZLlYVjlW+QwI+1zE17kGxuJGmIqDQdYp56XdmGeD+Bswx0UTyG18xA== +"@babel/generator@^7.18.10": + version "7.18.12" + resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.18.12.tgz#fa58daa303757bd6f5e4bbca91b342040463d9f4" + integrity sha512-dfQ8ebCN98SvyL7IxNMCUtZQSq5R7kxgN+r8qYTGDmmSion1hX2C0zq2yo1bsCDhXixokv1SAWTZUMYbO/V5zg== dependencies: - "@babel/compat-data" "^7.13.15" - "@babel/helper-validator-option" "^7.12.17" - browserslist "^4.14.5" - semver "^6.3.0" + "@babel/types" "^7.18.10" + "@jridgewell/gen-mapping" "^0.3.2" + jsesc "^2.5.1" "@babel/helper-compilation-targets@^7.16.7": version "7.16.7" @@ -171,6 +115,16 @@ browserslist "^4.17.5" semver "^6.3.0" +"@babel/helper-compilation-targets@^7.18.9": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.18.9.tgz#69e64f57b524cde3e5ff6cc5a9f4a387ee5563bf" + integrity sha512-tzLCyVmqUiFlcFoAPLA/gL9TeYrF61VLNtb+hvkuVaB5SUjW7jcfrglBIX1vUIoT7CLP3bBlIMeyEsIl2eFQNg== + dependencies: + "@babel/compat-data" "^7.18.8" + "@babel/helper-validator-option" "^7.18.6" + browserslist "^4.20.2" + semver "^6.3.0" + "@babel/helper-environment-visitor@^7.16.7": version "7.16.7" resolved "https://registry.yarnpkg.com/@babel/helper-environment-visitor/-/helper-environment-visitor-7.16.7.tgz#ff484094a839bde9d89cd63cba017d7aae80ecd7" @@ -178,14 +132,10 @@ dependencies: "@babel/types" "^7.16.7" -"@babel/helper-function-name@^7.12.13": - version "7.12.13" - resolved "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.12.13.tgz" - integrity sha512-TZvmPn0UOqmvi5G4vvw0qZTpVptGkB1GL61R6lKvrSdIxGm5Pky7Q3fpKiIkQCAtRCBUwB0PaThlx9vebCDSwA== - dependencies: - "@babel/helper-get-function-arity" "^7.12.13" - "@babel/template" "^7.12.13" - "@babel/types" "^7.12.13" +"@babel/helper-environment-visitor@^7.18.9": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/helper-environment-visitor/-/helper-environment-visitor-7.18.9.tgz#0c0cee9b35d2ca190478756865bb3528422f51be" + integrity sha512-3r/aACDJ3fhQ/EVgFy0hpj8oHyHpQc+LPtJoY9SzTThAsStm4Ptegq92vqKoE3vD706ZVFWITnMnxucw+S9Ipg== "@babel/helper-function-name@^7.16.7": version "7.16.7" @@ -196,12 +146,13 @@ "@babel/template" "^7.16.7" "@babel/types" "^7.16.7" -"@babel/helper-get-function-arity@^7.12.13": - version "7.12.13" - resolved "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.12.13.tgz" - integrity sha512-DjEVzQNz5LICkzN0REdpD5prGoidvbdYk1BVgRUOINaWJP2t6avB27X1guXK1kXNrX0WMfsrm1A/ZBthYuIMQg== +"@babel/helper-function-name@^7.18.9": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/helper-function-name/-/helper-function-name-7.18.9.tgz#940e6084a55dee867d33b4e487da2676365e86b0" + integrity sha512-fJgWlZt7nxGksJS9a0XdSaI4XvpExnNIgRP+rVefWh5U7BL8pPuir6SJUmFKRfjWQ51OtWSzwOxhaH/EBWWc0A== dependencies: - "@babel/types" "^7.12.13" + "@babel/template" "^7.18.6" + "@babel/types" "^7.18.9" "@babel/helper-get-function-arity@^7.16.7": version "7.16.7" @@ -217,19 +168,12 @@ dependencies: "@babel/types" "^7.16.7" -"@babel/helper-member-expression-to-functions@^7.13.12": - version "7.13.12" - resolved "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.13.12.tgz" - integrity sha512-48ql1CLL59aKbU94Y88Xgb2VFy7a95ykGRbJJaaVv+LX5U8wFpLfiGXJJGUozsmA1oEh/o5Bp60Voq7ACyA/Sw== - dependencies: - "@babel/types" "^7.13.12" - -"@babel/helper-module-imports@^7.13.12": - version "7.13.12" - resolved "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.13.12.tgz" - integrity sha512-4cVvR2/1B693IuOvSI20xqqa/+bl7lqAMR59R4iu39R9aOX8/JoYY1sFaNvUMyMBGnHdwvJgUrzNLoUZxXypxA== +"@babel/helper-hoist-variables@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.18.6.tgz#d4d2c8fb4baeaa5c68b99cc8245c56554f926678" + integrity sha512-UlJQPkFqFULIcyW5sbzgbkxn2FKRgwWiRexcuaR8RNJRy8+LLveqPjwZV/bwrLZCN0eUHD/x8D0heK1ozuoo6Q== dependencies: - "@babel/types" "^7.13.12" + "@babel/types" "^7.18.6" "@babel/helper-module-imports@^7.16.7": version "7.16.7" @@ -238,19 +182,12 @@ dependencies: "@babel/types" "^7.16.7" -"@babel/helper-module-transforms@^7.13.14": - version "7.13.14" - resolved "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.13.14.tgz" - integrity sha512-QuU/OJ0iAOSIatyVZmfqB0lbkVP0kDRiKj34xy+QNsnVZi/PA6BoSoreeqnxxa9EHFAIL0R9XOaAR/G9WlIy5g== +"@babel/helper-module-imports@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.18.6.tgz#1e3ebdbbd08aad1437b428c50204db13c5a3ca6e" + integrity sha512-0NFvs3VkuSYbFi1x2Vd6tKrywq+z/cLeYC/RJNFrIX/30Bf5aiGYbtvGXolEktzJH8o5E5KJ3tT+nkxuuZFVlA== dependencies: - "@babel/helper-module-imports" "^7.13.12" - "@babel/helper-replace-supers" "^7.13.12" - "@babel/helper-simple-access" "^7.13.12" - "@babel/helper-split-export-declaration" "^7.12.13" - "@babel/helper-validator-identifier" "^7.12.11" - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.13" - "@babel/types" "^7.13.14" + "@babel/types" "^7.18.6" "@babel/helper-module-transforms@^7.16.7": version "7.17.6" @@ -266,12 +203,19 @@ "@babel/traverse" "^7.17.3" "@babel/types" "^7.17.0" -"@babel/helper-optimise-call-expression@^7.12.13": - version "7.12.13" - resolved "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.12.13.tgz" - integrity sha512-BdWQhoVJkp6nVjB7nkFWcn43dkprYauqtk++Py2eaf/GRDFm5BxRqEIZCiHlZUGAVmtwKcsVL1dC68WmzeFmiA== +"@babel/helper-module-transforms@^7.18.9": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.18.9.tgz#5a1079c005135ed627442df31a42887e80fcb712" + integrity sha512-KYNqY0ICwfv19b31XzvmI/mfcylOzbLtowkw+mfvGPAQ3kfCnMLYbED3YecL5tPd8nAYFQFAd6JHp2LxZk/J1g== dependencies: - "@babel/types" "^7.12.13" + "@babel/helper-environment-visitor" "^7.18.9" + "@babel/helper-module-imports" "^7.18.6" + "@babel/helper-simple-access" "^7.18.6" + "@babel/helper-split-export-declaration" "^7.18.6" + "@babel/helper-validator-identifier" "^7.18.6" + "@babel/template" "^7.18.6" + "@babel/traverse" "^7.18.9" + "@babel/types" "^7.18.9" "@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.10.4", "@babel/helper-plugin-utils@^7.12.13", "@babel/helper-plugin-utils@^7.8.0": version "7.13.0" @@ -283,23 +227,6 @@ resolved "https://registry.yarnpkg.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.16.7.tgz#aa3a8ab4c3cceff8e65eb9e73d87dc4ff320b2f5" integrity sha512-Qg3Nk7ZxpgMrsox6HreY1ZNKdBq7K72tDSliA6dCl5f007jR4ne8iD5UzuNnCJH2xBf2BEEVGr+/OL6Gdp7RxA== -"@babel/helper-replace-supers@^7.13.12": - version "7.13.12" - resolved "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.13.12.tgz" - integrity sha512-Gz1eiX+4yDO8mT+heB94aLVNCL+rbuT2xy4YfyNqu8F+OI6vMvJK891qGBTqL9Uc8wxEvRW92Id6G7sDen3fFw== - dependencies: - "@babel/helper-member-expression-to-functions" "^7.13.12" - "@babel/helper-optimise-call-expression" "^7.12.13" - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.12" - -"@babel/helper-simple-access@^7.13.12": - version "7.13.12" - resolved "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.13.12.tgz" - integrity sha512-7FEjbrx5SL9cWvXioDbnlYTppcZGuCY6ow3/D5vMggb2Ywgu4dMrpTJX0JdQAIcRRUElOIxF3yEooa9gUb9ZbA== - dependencies: - "@babel/types" "^7.13.12" - "@babel/helper-simple-access@^7.16.7": version "7.16.7" resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.16.7.tgz#d656654b9ea08dbb9659b69d61063ccd343ff0f7" @@ -307,12 +234,12 @@ dependencies: "@babel/types" "^7.16.7" -"@babel/helper-split-export-declaration@^7.12.13": - version "7.12.13" - resolved "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.12.13.tgz" - integrity sha512-tCJDltF83htUtXx5NLcaDqRmknv652ZWCHyoTETf1CXYJdPC7nohZohjUgieXhv0hTJdRf2FjDueFehdNucpzg== +"@babel/helper-simple-access@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.18.6.tgz#d6d8f51f4ac2978068df934b569f08f29788c7ea" + integrity sha512-iNpIgTgyAvDQpDj76POqg+YEt8fPxx3yaNBg3S30dxNKm2SWfYhD0TGrK/Eu9wHpUW63VQU894TsTg+GLbUa1g== dependencies: - "@babel/types" "^7.12.13" + "@babel/types" "^7.18.6" "@babel/helper-split-export-declaration@^7.16.7": version "7.16.7" @@ -321,6 +248,18 @@ dependencies: "@babel/types" "^7.16.7" +"@babel/helper-split-export-declaration@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.18.6.tgz#7367949bc75b20c6d5a5d4a97bba2824ae8ef075" + integrity sha512-bde1etTx6ZyTmobl9LLMMQsaizFVZrquTEHOqKeQESMKo4PlObf+8+JA25ZsIpZhT/WEd39+vOdLXAFG/nELpA== + dependencies: + "@babel/types" "^7.18.6" + +"@babel/helper-string-parser@^7.18.10": + version "7.18.10" + resolved "https://registry.yarnpkg.com/@babel/helper-string-parser/-/helper-string-parser-7.18.10.tgz#181f22d28ebe1b3857fa575f5c290b1aaf659b56" + integrity sha512-XtIfWmeNY3i4t7t4D2t02q50HvqHybPqW2ki1kosnvWCwuCMeo81Jf0gwr85jy/neUdg5XDdeFE/80DXiO+njw== + "@babel/helper-validator-identifier@^7.12.11": version "7.12.11" resolved "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.12.11.tgz" @@ -331,24 +270,20 @@ resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.16.7.tgz#e8c602438c4a8195751243da9031d1607d247cad" integrity sha512-hsEnFemeiW4D08A5gUAZxLBTXpZ39P+a+DGDsHw1yxqyQ/jzFEnxf5uTEGp+3bzAbNOxU1paTgYS4ECU/IgfDw== -"@babel/helper-validator-option@^7.12.17": - version "7.12.17" - resolved "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.12.17.tgz" - integrity sha512-TopkMDmLzq8ngChwRlyjR6raKD6gMSae4JdYDB8bByKreQgG0RBTuKe9LRxW3wFtUnjxOPRKBDwEH6Mg5KeDfw== +"@babel/helper-validator-identifier@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.18.6.tgz#9c97e30d31b2b8c72a1d08984f2ca9b574d7a076" + integrity sha512-MmetCkz9ej86nJQV+sFCxoGGrUbU3q02kgLciwkrt9QqEB7cP39oKEY0PakknEO0Gu20SskMRi+AYZ3b1TpN9g== "@babel/helper-validator-option@^7.16.7": version "7.16.7" resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.16.7.tgz#b203ce62ce5fe153899b617c08957de860de4d23" integrity sha512-TRtenOuRUVo9oIQGPC5G9DgK4743cdxvtOw0weQNpZXaS16SCBi5MNjZF8vba3ETURjZpTbVn7Vvcf2eAwFozQ== -"@babel/helpers@^7.13.16": - version "7.13.16" - resolved "https://registry.npmjs.org/@babel/helpers/-/helpers-7.13.16.tgz" - integrity sha512-x5otxUaLpdWHl02P4L94wBU+2BJXBkvO+6d6uzQ+xD9/h2hTSAwA5O8QV8GqKx/l8i+VYmKKQg9e2QGTa2Wu3Q== - dependencies: - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.15" - "@babel/types" "^7.13.16" +"@babel/helper-validator-option@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.18.6.tgz#bf0d2b5a509b1f336099e4ff36e1a63aa5db4db8" + integrity sha512-XO7gESt5ouv/LRJdrVjkShckw6STTaB7l9BrpBaAHDeF5YZT+01PCwmR0SJHnkW6i8OwW/EVWRShfi4j2x+KQw== "@babel/helpers@^7.17.2": version "7.17.2" @@ -359,6 +294,15 @@ "@babel/traverse" "^7.17.0" "@babel/types" "^7.17.0" +"@babel/helpers@^7.18.9": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.18.9.tgz#4bef3b893f253a1eced04516824ede94dcfe7ff9" + integrity sha512-Jf5a+rbrLoR4eNdUmnFu8cN5eNJT6qdTdOg5IHIzq87WwyRw9PwguLFOWYgktN/60IP4fgDUawJvs7PjQIzELQ== + dependencies: + "@babel/template" "^7.18.6" + "@babel/traverse" "^7.18.9" + "@babel/types" "^7.18.9" + "@babel/highlight@^7.12.13": version "7.13.10" resolved "https://registry.npmjs.org/@babel/highlight/-/highlight-7.13.10.tgz" @@ -377,7 +321,16 @@ chalk "^2.0.0" js-tokens "^4.0.0" -"@babel/parser@^7.1.0", "@babel/parser@^7.12.13", "@babel/parser@^7.13.15", "@babel/parser@^7.13.16": +"@babel/highlight@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.18.6.tgz#81158601e93e2563795adcbfbdf5d64be3f2ecdf" + integrity sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g== + dependencies: + "@babel/helper-validator-identifier" "^7.18.6" + chalk "^2.0.0" + js-tokens "^4.0.0" + +"@babel/parser@^7.1.0", "@babel/parser@^7.12.13": version "7.13.16" resolved "https://registry.npmjs.org/@babel/parser/-/parser-7.13.16.tgz" integrity sha512-6bAg36mCwuqLO0hbR+z7PHuqWiCeP7Dzg73OpQwsAB1Eb8HnGEz5xYBzCfbu+YjoaJsJs+qheDxVAuqbt3ILEw== @@ -387,6 +340,11 @@ resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.17.3.tgz#b07702b982990bf6fdc1da5049a23fece4c5c3d0" integrity sha512-7yJPvPV+ESz2IUTPbOL+YkIGyCqOyNIzdguKQuJGnH7bg1WTIifuM21YqokFt/THWh1AkCRn9IgoykTRCBVpzA== +"@babel/parser@^7.18.10", "@babel/parser@^7.18.11": + version "7.18.11" + resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.18.11.tgz#68bb07ab3d380affa9a3f96728df07969645d2d9" + integrity sha512-9JKn5vN+hDt0Hdqn1PiJ2guflwP+B6Ga8qbDuoF0PzzVhrzsKIJo8yGqVk6CmMHiMei9w1C1Bp9IMJSIK+HPIQ== + "@babel/plugin-syntax-async-generators@^7.8.4": version "7.8.4" resolved "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz" @@ -478,14 +436,20 @@ dependencies: "@babel/helper-plugin-utils" "^7.16.7" -"@babel/template@^7.12.13", "@babel/template@^7.3.3": - version "7.12.13" - resolved "https://registry.npmjs.org/@babel/template/-/template-7.12.13.tgz" - integrity sha512-/7xxiGA57xMo/P2GVvdEumr8ONhFOhfgq2ihK3h1e6THqzTAkHbkXgB0xI9yeTfIUoH3+oAeHhqm/I43OTbbjA== +"@babel/runtime-corejs3@^7.10.2": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/runtime-corejs3/-/runtime-corejs3-7.18.9.tgz#7bacecd1cb2dd694eacd32a91fcf7021c20770ae" + integrity sha512-qZEWeccZCrHA2Au4/X05QW5CMdm4VjUDCrGq5gf1ZDcM4hRqreKrtwAn7yci9zfgAS9apvnsFXiGBHBAxZdK9A== dependencies: - "@babel/code-frame" "^7.12.13" - "@babel/parser" "^7.12.13" - "@babel/types" "^7.12.13" + core-js-pure "^3.20.2" + regenerator-runtime "^0.13.4" + +"@babel/runtime@^7.10.2", "@babel/runtime@^7.18.9": + version "7.18.9" + resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.18.9.tgz#b4fcfce55db3d2e5e080d2490f608a3b9f407f4a" + integrity sha512-lkqXDcvlFT5rvEjiu6+QYO+1GXrEHRo2LOtS7E4GtX5ESIZOgepqsZBVIj6Pv+a6zqsya9VCgiK1KAK4BvJDAw== + dependencies: + regenerator-runtime "^0.13.4" "@babel/template@^7.16.7": version "7.16.7" @@ -496,19 +460,23 @@ "@babel/parser" "^7.16.7" "@babel/types" "^7.16.7" -"@babel/traverse@^7.13.0", "@babel/traverse@^7.13.13", "@babel/traverse@^7.13.15": - version "7.13.15" - resolved "https://registry.npmjs.org/@babel/traverse/-/traverse-7.13.15.tgz" - integrity sha512-/mpZMNvj6bce59Qzl09fHEs8Bt8NnpEDQYleHUPZQ3wXUMvXi+HJPLars68oAbmp839fGoOkv2pSL2z9ajCIaQ== +"@babel/template@^7.18.10", "@babel/template@^7.18.6": + version "7.18.10" + resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.18.10.tgz#6f9134835970d1dbf0835c0d100c9f38de0c5e71" + integrity sha512-TI+rCtooWHr3QJ27kJxfjutghu44DLnasDMwpDqCXVTal9RLp3RSYNh4NdBrRP2cQAoG9A8juOQl6P6oZG4JxA== + dependencies: + "@babel/code-frame" "^7.18.6" + "@babel/parser" "^7.18.10" + "@babel/types" "^7.18.10" + +"@babel/template@^7.3.3": + version "7.12.13" + resolved "https://registry.npmjs.org/@babel/template/-/template-7.12.13.tgz" + integrity sha512-/7xxiGA57xMo/P2GVvdEumr8ONhFOhfgq2ihK3h1e6THqzTAkHbkXgB0xI9yeTfIUoH3+oAeHhqm/I43OTbbjA== dependencies: "@babel/code-frame" "^7.12.13" - "@babel/generator" "^7.13.9" - "@babel/helper-function-name" "^7.12.13" - "@babel/helper-split-export-declaration" "^7.12.13" - "@babel/parser" "^7.13.15" - "@babel/types" "^7.13.14" - debug "^4.1.0" - globals "^11.1.0" + "@babel/parser" "^7.12.13" + "@babel/types" "^7.12.13" "@babel/traverse@^7.17.0", "@babel/traverse@^7.17.3", "@babel/traverse@^7.7.2": version "7.17.3" @@ -526,7 +494,23 @@ debug "^4.1.0" globals "^11.1.0" -"@babel/types@^7.0.0", "@babel/types@^7.12.13", "@babel/types@^7.13.12", "@babel/types@^7.13.14", "@babel/types@^7.13.16", "@babel/types@^7.3.0", "@babel/types@^7.3.3": +"@babel/traverse@^7.18.10", "@babel/traverse@^7.18.9": + version "7.18.11" + resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.18.11.tgz#3d51f2afbd83ecf9912bcbb5c4d94e3d2ddaa16f" + integrity sha512-TG9PiM2R/cWCAy6BPJKeHzNbu4lPzOSZpeMfeNErskGpTJx6trEvFaVCbDvpcxwy49BKWmEPwiW8mrysNiDvIQ== + dependencies: + "@babel/code-frame" "^7.18.6" + "@babel/generator" "^7.18.10" + "@babel/helper-environment-visitor" "^7.18.9" + "@babel/helper-function-name" "^7.18.9" + "@babel/helper-hoist-variables" "^7.18.6" + "@babel/helper-split-export-declaration" "^7.18.6" + "@babel/parser" "^7.18.11" + "@babel/types" "^7.18.10" + debug "^4.1.0" + globals "^11.1.0" + +"@babel/types@^7.0.0", "@babel/types@^7.12.13", "@babel/types@^7.3.0", "@babel/types@^7.3.3": version "7.13.16" resolved "https://registry.npmjs.org/@babel/types/-/types-7.13.16.tgz" integrity sha512-7enM8Wxhrl1hB1+k6+xO6RmxpNkaveRWkdpyii8DkrLWRgr0l3x29/SEuhTIkP+ynHsU/Hpjn8Evd/axv/ll6Q== @@ -542,6 +526,15 @@ "@babel/helper-validator-identifier" "^7.16.7" to-fast-properties "^2.0.0" +"@babel/types@^7.18.10", "@babel/types@^7.18.6", "@babel/types@^7.18.9": + version "7.18.10" + resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.18.10.tgz#4908e81b6b339ca7c6b7a555a5fc29446f26dde6" + integrity sha512-MJvnbEiiNkpjo+LknnmRrqbY1GPUUggjv+wQVjetM/AONoupqRALB7I6jGqNUAZsKcRIEu2J6FRFvsczljjsaQ== + dependencies: + "@babel/helper-string-parser" "^7.18.10" + "@babel/helper-validator-identifier" "^7.18.6" + to-fast-properties "^2.0.0" + "@bcoe/v8-coverage@^0.2.3": version "0.2.3" resolved "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz" @@ -559,22 +552,40 @@ dependencies: "@cspotcode/source-map-consumer" "0.8.0" -"@eslint/eslintrc@^0.2.2": - version "0.2.2" - resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-0.2.2.tgz#d01fc791e2fc33e88a29d6f3dc7e93d0cd784b76" - integrity sha512-EfB5OHNYp1F4px/LI/FEnGylop7nOqkQ1LRzCM0KccA2U8tvV8w01KBv37LbO7nW4H+YhKyo2LcJhRwjjV17QQ== +"@eslint/eslintrc@^1.3.0": + version "1.3.0" + resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-1.3.0.tgz#29f92c30bb3e771e4a2048c95fa6855392dfac4f" + integrity sha512-UWW0TMTmk2d7hLcWD1/e2g5HDM/HQ3csaLSqXCfqwh4uNDuNqlaKWXmEsL4Cs41Z0KnILNvwbHAah3C2yt06kw== dependencies: ajv "^6.12.4" - debug "^4.1.1" - espree "^7.3.0" - globals "^12.1.0" - ignore "^4.0.6" + debug "^4.3.2" + espree "^9.3.2" + globals "^13.15.0" + ignore "^5.2.0" import-fresh "^3.2.1" - js-yaml "^3.13.1" - lodash "^4.17.19" - minimatch "^3.0.4" + js-yaml "^4.1.0" + minimatch "^3.1.2" strip-json-comments "^3.1.1" +"@humanwhocodes/config-array@^0.10.4": + version "0.10.4" + resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.10.4.tgz#01e7366e57d2ad104feea63e72248f22015c520c" + integrity sha512-mXAIHxZT3Vcpg83opl1wGlVZ9xydbfZO3r5YfRSH6Gpp2J/PfdBP0wbDa2sO6/qRbcalpoevVyW6A/fI6LfeMw== + dependencies: + "@humanwhocodes/object-schema" "^1.2.1" + debug "^4.1.1" + minimatch "^3.0.4" + +"@humanwhocodes/gitignore-to-minimatch@^1.0.2": + version "1.0.2" + resolved "https://registry.yarnpkg.com/@humanwhocodes/gitignore-to-minimatch/-/gitignore-to-minimatch-1.0.2.tgz#316b0a63b91c10e53f242efb4ace5c3b34e8728d" + integrity sha512-rSqmMJDdLFUsyxR6FMtD00nfQKKLFb1kv+qBbOVKqErvloEIJLo5bDTJTQNTYgeyp78JsA7u/NPi5jT1GR/MuA== + +"@humanwhocodes/object-schema@^1.2.1": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz#b520529ec21d8e5945a1851dfd1c32e94e39ff45" + integrity sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA== + "@istanbuljs/load-nyc-config@^1.0.0": version "1.1.0" resolved "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz" @@ -591,191 +602,218 @@ resolved "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz" integrity sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA== -"@jest/console@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/console/-/console-27.5.1.tgz#260fe7239602fe5130a94f1aa386eff54b014bba" - integrity sha512-kZ/tNpS3NXn0mlXXXPNuDZnb4c0oZ20r4K5eemM2k30ZC3G0T02nXUvyhf5YdbXWHPEJLc9qGLxEZ216MdL+Zg== +"@jest/console@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/console/-/console-28.1.3.tgz#2030606ec03a18c31803b8a36382762e447655df" + integrity sha512-QPAkP5EwKdK/bxIr6C1I4Vs0rm2nHiANzj/Z5X2JQkrZo6IqvC4ldZ9K95tF0HdidhA8Bo6egxSzUFPYKcEXLw== dependencies: - "@jest/types" "^27.5.1" + "@jest/types" "^28.1.3" "@types/node" "*" chalk "^4.0.0" - jest-message-util "^27.5.1" - jest-util "^27.5.1" + jest-message-util "^28.1.3" + jest-util "^28.1.3" slash "^3.0.0" -"@jest/core@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/core/-/core-27.5.1.tgz#267ac5f704e09dc52de2922cbf3af9edcd64b626" - integrity sha512-AK6/UTrvQD0Cd24NSqmIA6rKsu0tKIxfiCducZvqxYdmMisOYAsdItspT+fQDQYARPf8XgjAFZi0ogW2agH5nQ== +"@jest/core@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/core/-/core-28.1.3.tgz#0ebf2bd39840f1233cd5f2d1e6fc8b71bd5a1ac7" + integrity sha512-CIKBrlaKOzA7YG19BEqCw3SLIsEwjZkeJzf5bdooVnW4bH5cktqe3JX+G2YV1aK5vP8N9na1IGWFzYaTp6k6NA== dependencies: - "@jest/console" "^27.5.1" - "@jest/reporters" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/transform" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/console" "^28.1.3" + "@jest/reporters" "^28.1.3" + "@jest/test-result" "^28.1.3" + "@jest/transform" "^28.1.3" + "@jest/types" "^28.1.3" "@types/node" "*" ansi-escapes "^4.2.1" chalk "^4.0.0" - emittery "^0.8.1" + ci-info "^3.2.0" exit "^0.1.2" graceful-fs "^4.2.9" - jest-changed-files "^27.5.1" - jest-config "^27.5.1" - jest-haste-map "^27.5.1" - jest-message-util "^27.5.1" - jest-regex-util "^27.5.1" - jest-resolve "^27.5.1" - jest-resolve-dependencies "^27.5.1" - jest-runner "^27.5.1" - jest-runtime "^27.5.1" - jest-snapshot "^27.5.1" - jest-util "^27.5.1" - jest-validate "^27.5.1" - jest-watcher "^27.5.1" + jest-changed-files "^28.1.3" + jest-config "^28.1.3" + jest-haste-map "^28.1.3" + jest-message-util "^28.1.3" + jest-regex-util "^28.0.2" + jest-resolve "^28.1.3" + jest-resolve-dependencies "^28.1.3" + jest-runner "^28.1.3" + jest-runtime "^28.1.3" + jest-snapshot "^28.1.3" + jest-util "^28.1.3" + jest-validate "^28.1.3" + jest-watcher "^28.1.3" micromatch "^4.0.4" + pretty-format "^28.1.3" rimraf "^3.0.0" slash "^3.0.0" strip-ansi "^6.0.0" -"@jest/environment@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/environment/-/environment-27.5.1.tgz#d7425820511fe7158abbecc010140c3fd3be9c74" - integrity sha512-/WQjhPJe3/ghaol/4Bq480JKXV/Rfw8nQdN7f41fM8VDHLcxKXou6QyXAh3EFr9/bVG3x74z1NWDkP87EiY8gA== +"@jest/environment@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/environment/-/environment-28.1.3.tgz#abed43a6b040a4c24fdcb69eab1f97589b2d663e" + integrity sha512-1bf40cMFTEkKyEf585R9Iz1WayDjHoHqvts0XFYEqyKM3cFWDpeMoqKKTAF9LSYQModPUlh8FKptoM2YcMWAXA== dependencies: - "@jest/fake-timers" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/fake-timers" "^28.1.3" + "@jest/types" "^28.1.3" "@types/node" "*" - jest-mock "^27.5.1" + jest-mock "^28.1.3" + +"@jest/expect-utils@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/expect-utils/-/expect-utils-28.1.3.tgz#58561ce5db7cd253a7edddbc051fb39dda50f525" + integrity sha512-wvbi9LUrHJLn3NlDW6wF2hvIMtd4JUl2QNVrjq+IBSHirgfrR3o9RnVtxzdEGO2n9JyIWwHnLfby5KzqBGg2YA== + dependencies: + jest-get-type "^28.0.2" + +"@jest/expect@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/expect/-/expect-28.1.3.tgz#9ac57e1d4491baca550f6bdbd232487177ad6a72" + integrity sha512-lzc8CpUbSoE4dqT0U+g1qODQjBRHPpCPXissXD4mS9+sWQdmmpeJ9zSH1rS1HEkrsMN0fb7nKrJ9giAR1d3wBw== + dependencies: + expect "^28.1.3" + jest-snapshot "^28.1.3" -"@jest/fake-timers@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/fake-timers/-/fake-timers-27.5.1.tgz#76979745ce0579c8a94a4678af7a748eda8ada74" - integrity sha512-/aPowoolwa07k7/oM3aASneNeBGCmGQsc3ugN4u6s4C/+s5M64MFo/+djTdiwcbQlRfFElGuDXWzaWj6QgKObQ== +"@jest/fake-timers@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/fake-timers/-/fake-timers-28.1.3.tgz#230255b3ad0a3d4978f1d06f70685baea91c640e" + integrity sha512-D/wOkL2POHv52h+ok5Oj/1gOG9HSywdoPtFsRCUmlCILXNn5eIWmcnd3DIiWlJnpGvQtmajqBP95Ei0EimxfLw== dependencies: - "@jest/types" "^27.5.1" - "@sinonjs/fake-timers" "^8.0.1" + "@jest/types" "^28.1.3" + "@sinonjs/fake-timers" "^9.1.2" "@types/node" "*" - jest-message-util "^27.5.1" - jest-mock "^27.5.1" - jest-util "^27.5.1" + jest-message-util "^28.1.3" + jest-mock "^28.1.3" + jest-util "^28.1.3" -"@jest/globals@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/globals/-/globals-27.5.1.tgz#7ac06ce57ab966566c7963431cef458434601b2b" - integrity sha512-ZEJNB41OBQQgGzgyInAv0UUfDDj3upmHydjieSxFvTRuZElrx7tXg/uVQ5hYVEwiXs3+aMsAeEc9X7xiSKCm4Q== +"@jest/globals@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/globals/-/globals-28.1.3.tgz#a601d78ddc5fdef542728309894895b4a42dc333" + integrity sha512-XFU4P4phyryCXu1pbcqMO0GSQcYe1IsalYCDzRNyhetyeyxMcIxa11qPNDpVNLeretItNqEmYYQn1UYz/5x1NA== dependencies: - "@jest/environment" "^27.5.1" - "@jest/types" "^27.5.1" - expect "^27.5.1" + "@jest/environment" "^28.1.3" + "@jest/expect" "^28.1.3" + "@jest/types" "^28.1.3" -"@jest/reporters@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/reporters/-/reporters-27.5.1.tgz#ceda7be96170b03c923c37987b64015812ffec04" - integrity sha512-cPXh9hWIlVJMQkVk84aIvXuBB4uQQmFqZiacloFuGiP3ah1sbCxCosidXFDfqG8+6fO1oR2dTJTlsOy4VFmUfw== +"@jest/reporters@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/reporters/-/reporters-28.1.3.tgz#9adf6d265edafc5fc4a434cfb31e2df5a67a369a" + integrity sha512-JuAy7wkxQZVNU/V6g9xKzCGC5LVXx9FDcABKsSXp5MiKPEE2144a/vXTEDoyzjUpZKfVwp08Wqg5A4WfTMAzjg== dependencies: "@bcoe/v8-coverage" "^0.2.3" - "@jest/console" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/transform" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/console" "^28.1.3" + "@jest/test-result" "^28.1.3" + "@jest/transform" "^28.1.3" + "@jest/types" "^28.1.3" + "@jridgewell/trace-mapping" "^0.3.13" "@types/node" "*" chalk "^4.0.0" collect-v8-coverage "^1.0.0" exit "^0.1.2" - glob "^7.1.2" + glob "^7.1.3" graceful-fs "^4.2.9" istanbul-lib-coverage "^3.0.0" istanbul-lib-instrument "^5.1.0" istanbul-lib-report "^3.0.0" istanbul-lib-source-maps "^4.0.0" istanbul-reports "^3.1.3" - jest-haste-map "^27.5.1" - jest-resolve "^27.5.1" - jest-util "^27.5.1" - jest-worker "^27.5.1" + jest-message-util "^28.1.3" + jest-util "^28.1.3" + jest-worker "^28.1.3" slash "^3.0.0" - source-map "^0.6.0" string-length "^4.0.1" + strip-ansi "^6.0.0" terminal-link "^2.0.0" - v8-to-istanbul "^8.1.0" + v8-to-istanbul "^9.0.1" -"@jest/source-map@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/source-map/-/source-map-27.5.1.tgz#6608391e465add4205eae073b55e7f279e04e8cf" - integrity sha512-y9NIHUYF3PJRlHk98NdC/N1gl88BL08aQQgu4k4ZopQkCw9t9cV8mtl3TV8b/YCB8XaVTFrmUTAJvjsntDireg== +"@jest/schemas@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/schemas/-/schemas-28.1.3.tgz#ad8b86a66f11f33619e3d7e1dcddd7f2d40ff905" + integrity sha512-/l/VWsdt/aBXgjshLWOFyFt3IVdYypu5y2Wn2rOO1un6nkqIn8SLXzgIMYXFyYsRWDyF5EthmKJMIdJvk08grg== dependencies: + "@sinclair/typebox" "^0.24.1" + +"@jest/source-map@^28.1.2": + version "28.1.2" + resolved "https://registry.yarnpkg.com/@jest/source-map/-/source-map-28.1.2.tgz#7fe832b172b497d6663cdff6c13b0a920e139e24" + integrity sha512-cV8Lx3BeStJb8ipPHnqVw/IM2VCMWO3crWZzYodSIkxXnRcXJipCdx1JCK0K5MsJJouZQTH73mzf4vgxRaH9ww== + dependencies: + "@jridgewell/trace-mapping" "^0.3.13" callsites "^3.0.0" graceful-fs "^4.2.9" - source-map "^0.6.0" -"@jest/test-result@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/test-result/-/test-result-27.5.1.tgz#56a6585fa80f7cdab72b8c5fc2e871d03832f5bb" - integrity sha512-EW35l2RYFUcUQxFJz5Cv5MTOxlJIQs4I7gxzi2zVU7PJhOwfYq1MdC5nhSmYjX1gmMmLPvB3sIaC+BkcHRBfag== +"@jest/test-result@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/test-result/-/test-result-28.1.3.tgz#5eae945fd9f4b8fcfce74d239e6f725b6bf076c5" + integrity sha512-kZAkxnSE+FqE8YjW8gNuoVkkC9I7S1qmenl8sGcDOLropASP+BkcGKwhXoyqQuGOGeYY0y/ixjrd/iERpEXHNg== dependencies: - "@jest/console" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/console" "^28.1.3" + "@jest/types" "^28.1.3" "@types/istanbul-lib-coverage" "^2.0.0" collect-v8-coverage "^1.0.0" -"@jest/test-sequencer@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/test-sequencer/-/test-sequencer-27.5.1.tgz#4057e0e9cea4439e544c6353c6affe58d095745b" - integrity sha512-LCheJF7WB2+9JuCS7VB/EmGIdQuhtqjRNI9A43idHv3E4KltCTsPsLxvdaubFHSYwY/fNjMWjl6vNRhDiN7vpQ== +"@jest/test-sequencer@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/test-sequencer/-/test-sequencer-28.1.3.tgz#9d0c283d906ac599c74bde464bc0d7e6a82886c3" + integrity sha512-NIMPEqqa59MWnDi1kvXXpYbqsfQmSJsIbnd85mdVGkiDfQ9WQQTXOLsvISUfonmnBT+w85WEgneCigEEdHDFxw== dependencies: - "@jest/test-result" "^27.5.1" + "@jest/test-result" "^28.1.3" graceful-fs "^4.2.9" - jest-haste-map "^27.5.1" - jest-runtime "^27.5.1" + jest-haste-map "^28.1.3" + slash "^3.0.0" -"@jest/transform@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/transform/-/transform-27.5.1.tgz#6c3501dcc00c4c08915f292a600ece5ecfe1f409" - integrity sha512-ipON6WtYgl/1329g5AIJVbUuEh0wZVbdpGwC99Jw4LwuoBNS95MVphU6zOeD9pDkon+LLbFL7lOQRapbB8SCHw== +"@jest/transform@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/transform/-/transform-28.1.3.tgz#59d8098e50ab07950e0f2fc0fc7ec462371281b0" + integrity sha512-u5dT5di+oFI6hfcLOHGTAfmUxFRrjK+vnaP0kkVow9Md/M7V/MxqQMOz/VV25UZO8pzeA9PjfTpOu6BDuwSPQA== dependencies: - "@babel/core" "^7.1.0" - "@jest/types" "^27.5.1" + "@babel/core" "^7.11.6" + "@jest/types" "^28.1.3" + "@jridgewell/trace-mapping" "^0.3.13" babel-plugin-istanbul "^6.1.1" chalk "^4.0.0" convert-source-map "^1.4.0" fast-json-stable-stringify "^2.0.0" graceful-fs "^4.2.9" - jest-haste-map "^27.5.1" - jest-regex-util "^27.5.1" - jest-util "^27.5.1" + jest-haste-map "^28.1.3" + jest-regex-util "^28.0.2" + jest-util "^28.1.3" micromatch "^4.0.4" pirates "^4.0.4" slash "^3.0.0" - source-map "^0.6.1" - write-file-atomic "^3.0.0" + write-file-atomic "^4.0.1" -"@jest/types@^27.4.2": - version "27.4.2" - resolved "https://registry.yarnpkg.com/@jest/types/-/types-27.4.2.tgz#96536ebd34da6392c2b7c7737d693885b5dd44a5" - integrity sha512-j35yw0PMTPpZsUoOBiuHzr1zTYoad1cVIE0ajEjcrJONxxrko/IRGKkXx3os0Nsi4Hu3+5VmDbVfq5WhG/pWAg== +"@jest/types@^28.1.3": + version "28.1.3" + resolved "https://registry.yarnpkg.com/@jest/types/-/types-28.1.3.tgz#b05de80996ff12512bc5ceb1d208285a7d11748b" + integrity sha512-RyjiyMUZrKz/c+zlMFO1pm70DcIlST8AeWTkoUdZevew44wcNZQHsEVOiCVtgVnlFFD82FPaXycys58cf2muVQ== dependencies: + "@jest/schemas" "^28.1.3" "@types/istanbul-lib-coverage" "^2.0.0" "@types/istanbul-reports" "^3.0.0" "@types/node" "*" - "@types/yargs" "^16.0.0" + "@types/yargs" "^17.0.8" chalk "^4.0.0" -"@jest/types@^27.5.1": - version "27.5.1" - resolved "https://registry.yarnpkg.com/@jest/types/-/types-27.5.1.tgz#3c79ec4a8ba61c170bf937bcf9e98a9df175ec80" - integrity sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw== +"@jridgewell/gen-mapping@^0.3.2": + version "0.3.2" + resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.2.tgz#c1aedc61e853f2bb9f5dfe6d4442d3b565b253b9" + integrity sha512-mh65xKQAzI6iBcFzwv28KVWSmCkdRBWoOh+bYQGW3+6OZvbbN3TqMGo5hqYxQniRcH9F2VZIoJCm4pa3BPDK/A== dependencies: - "@types/istanbul-lib-coverage" "^2.0.0" - "@types/istanbul-reports" "^3.0.0" - "@types/node" "*" - "@types/yargs" "^16.0.0" - chalk "^4.0.0" + "@jridgewell/set-array" "^1.0.1" + "@jridgewell/sourcemap-codec" "^1.4.10" + "@jridgewell/trace-mapping" "^0.3.9" "@jridgewell/resolve-uri@^3.0.3": version "3.0.5" resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.0.5.tgz#68eb521368db76d040a6315cdb24bf2483037b9c" integrity sha512-VPeQ7+wH0itvQxnG+lIzWgkysKIr3L9sslimFW55rHMdGu/qCQ5z5h9zq4gI8uBtqkpHhsF4Z/OwExufUCThew== +"@jridgewell/set-array@^1.0.1": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.1.2.tgz#7c6cf998d6d20b914c0a55a91ae928ff25965e72" + integrity sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw== + "@jridgewell/sourcemap-codec@^1.4.10": version "1.4.11" resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.11.tgz#771a1d8d744eeb71b6adb35808e1a6c7b9b8c8ec" @@ -789,53 +827,13 @@ "@jridgewell/resolve-uri" "^3.0.3" "@jridgewell/sourcemap-codec" "^1.4.10" -"@kubernetes/client-node@0.10.2": - version "0.10.2" - resolved "https://registry.npmjs.org/@kubernetes/client-node/-/client-node-0.10.2.tgz" - integrity sha512-JvsmxbTwiMqsh9LyuXMzT5HjoENFbB3a/JroJsobuAzkxN162UqAOvg++/AA+ccIMWRR2Qln4FyaOJ0a4eKyXg== - dependencies: - "@types/js-yaml" "^3.12.1" - "@types/node" "^10.12.0" - "@types/request" "^2.47.1" - "@types/underscore" "^1.8.9" - "@types/ws" "^6.0.1" - isomorphic-ws "^4.0.1" - js-yaml "^3.13.1" - json-stream "^1.0.0" - jsonpath-plus "^0.19.0" - request "^2.88.0" - shelljs "^0.8.2" - tslib "^1.9.3" - underscore "^1.9.1" - ws "^6.1.0" - -"@kubernetes/client-node@^0.16.3": - version "0.16.3" - resolved "https://registry.yarnpkg.com/@kubernetes/client-node/-/client-node-0.16.3.tgz#a26a5abbd6e45603b4f75f0baff00e19853e5be7" - integrity sha512-L7IckuyuPfhd+/Urib8MRas9D6sfKEq8IaITYcaE6LlU+Y8MeD7MTbuW6Yb2WdeRuFN8HPSS47mxPnOUNYBXEg== - dependencies: - "@types/js-yaml" "^4.0.1" - "@types/node" "^10.12.0" - "@types/request" "^2.47.1" - "@types/stream-buffers" "^3.0.3" - "@types/tar" "^4.0.3" - "@types/underscore" "^1.8.9" - "@types/ws" "^6.0.1" - byline "^5.0.0" - execa "5.0.0" - isomorphic-ws "^4.0.1" - js-yaml "^4.1.0" - jsonpath-plus "^0.19.0" - openid-client "^4.1.1" - request "^2.88.0" - rfc4648 "^1.3.0" - shelljs "^0.8.5" - stream-buffers "^3.0.2" - tar "^6.1.11" - tmp-promise "^3.0.2" - tslib "^1.9.3" - underscore "^1.9.1" - ws "^7.3.1" +"@jridgewell/trace-mapping@^0.3.12", "@jridgewell/trace-mapping@^0.3.13", "@jridgewell/trace-mapping@^0.3.9": + version "0.3.14" + resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.14.tgz#b231a081d8f66796e475ad588a1ef473112701ed" + integrity sha512-bJWEfQ9lPTvm3SneWwRFVLzrh6nhjwqw7TUFFBEMzwvg7t7PCDenf2lDwqo4NQXzdpgBXyFgDWnQA+2vkruksQ== + dependencies: + "@jridgewell/resolve-uri" "^3.0.3" + "@jridgewell/sourcemap-codec" "^1.4.10" "@nodelib/fs.scandir@2.1.4": version "2.1.4" @@ -858,118 +856,10 @@ "@nodelib/fs.scandir" "2.1.4" fastq "^1.6.0" -"@octokit/auth-token@^2.4.4": - version "2.5.0" - resolved "https://registry.yarnpkg.com/@octokit/auth-token/-/auth-token-2.5.0.tgz#27c37ea26c205f28443402477ffd261311f21e36" - integrity sha512-r5FVUJCOLl19AxiuZD2VRZ/ORjp/4IN98Of6YJoJOkY75CIBuYfmiNHGrDwXr+aLGG55igl9QrxX3hbiXlLb+g== - dependencies: - "@octokit/types" "^6.0.3" - -"@octokit/core@^3.4.0", "@octokit/core@^3.5.1": - version "3.5.1" - resolved "https://registry.yarnpkg.com/@octokit/core/-/core-3.5.1.tgz#8601ceeb1ec0e1b1b8217b960a413ed8e947809b" - integrity sha512-omncwpLVxMP+GLpLPgeGJBF6IWJFjXDS5flY5VbppePYX9XehevbDykRH9PdCdvqt9TS5AOTiDide7h0qrkHjw== - dependencies: - "@octokit/auth-token" "^2.4.4" - "@octokit/graphql" "^4.5.8" - "@octokit/request" "^5.6.0" - "@octokit/request-error" "^2.0.5" - "@octokit/types" "^6.0.3" - before-after-hook "^2.2.0" - universal-user-agent "^6.0.0" - -"@octokit/endpoint@^6.0.1": - version "6.0.11" - resolved "https://registry.npmjs.org/@octokit/endpoint/-/endpoint-6.0.11.tgz" - integrity sha512-fUIPpx+pZyoLW4GCs3yMnlj2LfoXTWDUVPTC4V3MUEKZm48W+XYpeWSZCv+vYF1ZABUm2CqnDVf1sFtIYrj7KQ== - dependencies: - "@octokit/types" "^6.0.3" - is-plain-object "^5.0.0" - universal-user-agent "^6.0.0" - -"@octokit/graphql@^4.5.8": - version "4.8.0" - resolved "https://registry.yarnpkg.com/@octokit/graphql/-/graphql-4.8.0.tgz#664d9b11c0e12112cbf78e10f49a05959aa22cc3" - integrity sha512-0gv+qLSBLKF0z8TKaSKTsS39scVKF9dbMxJpj3U0vC7wjNWFuIpL/z76Qe2fiuCbDRcJSavkXsVtMS6/dtQQsg== - dependencies: - "@octokit/request" "^5.6.0" - "@octokit/types" "^6.0.3" - universal-user-agent "^6.0.0" - -"@octokit/openapi-types@^11.2.0": - version "11.2.0" - resolved "https://registry.yarnpkg.com/@octokit/openapi-types/-/openapi-types-11.2.0.tgz#b38d7fc3736d52a1e96b230c1ccd4a58a2f400a6" - integrity sha512-PBsVO+15KSlGmiI8QAzaqvsNlZlrDlyAJYcrXBCvVUxCp7VnXjkwPoFHgjEJXx3WF9BAwkA6nfCUA7i9sODzKA== - -"@octokit/openapi-types@^6.0.0": - version "6.0.0" - resolved "https://registry.npmjs.org/@octokit/openapi-types/-/openapi-types-6.0.0.tgz" - integrity sha512-CnDdK7ivHkBtJYzWzZm7gEkanA7gKH6a09Eguz7flHw//GacPJLmkHA3f3N++MJmlxD1Fl+mB7B32EEpSCwztQ== - -"@octokit/plugin-paginate-rest@^2.13.3": - version "2.17.0" - resolved "https://registry.yarnpkg.com/@octokit/plugin-paginate-rest/-/plugin-paginate-rest-2.17.0.tgz#32e9c7cab2a374421d3d0de239102287d791bce7" - integrity sha512-tzMbrbnam2Mt4AhuyCHvpRkS0oZ5MvwwcQPYGtMv4tUa5kkzG58SVB0fcsLulOZQeRnOgdkZWkRUiyBlh0Bkyw== - dependencies: - "@octokit/types" "^6.34.0" - -"@octokit/plugin-rest-endpoint-methods@^5.1.1": - version "5.13.0" - resolved "https://registry.yarnpkg.com/@octokit/plugin-rest-endpoint-methods/-/plugin-rest-endpoint-methods-5.13.0.tgz#8c46109021a3412233f6f50d28786f8e552427ba" - integrity sha512-uJjMTkN1KaOIgNtUPMtIXDOjx6dGYysdIFhgA52x4xSadQCz3b/zJexvITDVpANnfKPW/+E0xkOvLntqMYpviA== - dependencies: - "@octokit/types" "^6.34.0" - deprecation "^2.3.1" - -"@octokit/request-error@^2.0.5", "@octokit/request-error@^2.1.0": - version "2.1.0" - resolved "https://registry.yarnpkg.com/@octokit/request-error/-/request-error-2.1.0.tgz#9e150357831bfc788d13a4fd4b1913d60c74d677" - integrity sha512-1VIvgXxs9WHSjicsRwq8PlR2LR2x6DwsJAaFgzdi0JfJoGSO8mYI/cHJQ+9FbN21aa+DrgNLnwObmyeSC8Rmpg== - dependencies: - "@octokit/types" "^6.0.3" - deprecation "^2.0.0" - once "^1.4.0" - -"@octokit/request@^5.6.0": - version "5.6.2" - resolved "https://registry.yarnpkg.com/@octokit/request/-/request-5.6.2.tgz#1aa74d5da7b9e04ac60ef232edd9a7438dcf32d8" - integrity sha512-je66CvSEVf0jCpRISxkUcCa0UkxmFs6eGDRSbfJtAVwbLH5ceqF+YEyC8lj8ystKyZTy8adWr0qmkY52EfOeLA== - dependencies: - "@octokit/endpoint" "^6.0.1" - "@octokit/request-error" "^2.1.0" - "@octokit/types" "^6.16.1" - is-plain-object "^5.0.0" - node-fetch "^2.6.1" - universal-user-agent "^6.0.0" - -"@octokit/types@^6.0.3": - version "6.13.0" - resolved "https://registry.npmjs.org/@octokit/types/-/types-6.13.0.tgz" - integrity sha512-W2J9qlVIU11jMwKHUp5/rbVUeErqelCsO5vW5PKNb7wAXQVUz87Rc+imjlEvpvbH8yUb+KHmv8NEjVZdsdpyxA== - dependencies: - "@octokit/openapi-types" "^6.0.0" - -"@octokit/types@^6.16.1", "@octokit/types@^6.34.0": - version "6.34.0" - resolved "https://registry.yarnpkg.com/@octokit/types/-/types-6.34.0.tgz#c6021333334d1ecfb5d370a8798162ddf1ae8218" - integrity sha512-s1zLBjWhdEI2zwaoSgyOFoKSl109CUcVBCc7biPJ3aAf6LGLU6szDvi31JPU7bxfla2lqfhjbbg/5DdFNxOwHw== - dependencies: - "@octokit/openapi-types" "^11.2.0" - -"@panva/asn1.js@^1.0.0": - version "1.0.0" - resolved "https://registry.yarnpkg.com/@panva/asn1.js/-/asn1.js-1.0.0.tgz#dd55ae7b8129e02049f009408b97c61ccf9032f6" - integrity sha512-UdkG3mLEqXgnlKsWanWcgb6dOjUzJ+XC5f+aWw30qrtjxeNUSfKX1cd5FBzOaXQumoe9nIqeZUvrRJS03HCCtw== - -"@sindresorhus/is@^0.14.0": - version "0.14.0" - resolved "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz" - integrity sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ== - -"@sindresorhus/is@^4.0.0": - version "4.6.0" - resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-4.6.0.tgz#3c7c9c46e678feefe7a2e5bb609d3dbd665ffb3f" - integrity sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw== +"@sinclair/typebox@^0.24.1": + version "0.24.27" + resolved "https://registry.yarnpkg.com/@sinclair/typebox/-/typebox-0.24.27.tgz#d55643516a1546174e10da681a8aaa81e757452d" + integrity sha512-K7C7IlQ3zLePEZleUN21ceBA2aLcMnLHTLph8QWk1JK37L90obdpY+QGY8bXMKxf1ht1Z0MNewvXxWv0oGDYFg== "@sinonjs/commons@^1.7.0": version "1.8.3" @@ -978,32 +868,13 @@ dependencies: type-detect "4.0.8" -"@sinonjs/fake-timers@^8.0.1": - version "8.1.0" - resolved "https://registry.yarnpkg.com/@sinonjs/fake-timers/-/fake-timers-8.1.0.tgz#3fdc2b6cb58935b21bfb8d1625eb1300484316e7" - integrity sha512-OAPJUAtgeINhh/TAlUID4QTs53Njm7xzddaVlEs/SXwgtiD1tW22zAB/W1wdqfrpmikgaWQ9Fw6Ws+hsiRm5Vg== +"@sinonjs/fake-timers@^9.1.2": + version "9.1.2" + resolved "https://registry.yarnpkg.com/@sinonjs/fake-timers/-/fake-timers-9.1.2.tgz#4eaab737fab77332ab132d396a3c0d364bd0ea8c" + integrity sha512-BPS4ynJW/o92PUR4wgriz2Ud5gpST5vz6GQfMixEDK0Z8ZCUv2M7SkBLykH56T++Xs+8ln9zTGbOvNGIe02/jw== dependencies: "@sinonjs/commons" "^1.7.0" -"@szmarczak/http-timer@^1.1.2": - version "1.1.2" - resolved "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz" - integrity sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA== - dependencies: - defer-to-connect "^1.0.1" - -"@szmarczak/http-timer@^4.0.5": - version "4.0.6" - resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-4.0.6.tgz#b4a914bb62e7c272d4e5989fe4440f812ab1d807" - integrity sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w== - dependencies: - defer-to-connect "^2.0.0" - -"@tootallnate/once@1": - version "1.1.2" - resolved "https://registry.yarnpkg.com/@tootallnate/once/-/once-1.1.2.tgz#ccb91445360179a04e7fe6aff78c00ffc1eeaf82" - integrity sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw== - "@tsconfig/node10@^1.0.7": version "1.0.8" resolved "https://registry.yarnpkg.com/@tsconfig/node10/-/node10-1.0.8.tgz#c1e4e80d6f964fbecb3359c43bd48b40f7cadad9" @@ -1024,17 +895,6 @@ resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.2.tgz#423c77877d0569db20e1fc80885ac4118314010e" integrity sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA== -"@types/babel__core@^7.0.0": - version "7.1.14" - resolved "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.14.tgz" - integrity sha512-zGZJzzBUVDo/eV6KgbE0f0ZI7dInEYvo12Rb70uNQDshC3SkRMb67ja0GgRHZgAX3Za6rhaWlvbDO8rrGyAb1g== - dependencies: - "@babel/parser" "^7.1.0" - "@babel/types" "^7.0.0" - "@types/babel__generator" "*" - "@types/babel__template" "*" - "@types/babel__traverse" "*" - "@types/babel__core@^7.1.14": version "7.1.18" resolved "https://registry.yarnpkg.com/@types/babel__core/-/babel__core-7.1.18.tgz#1a29abcc411a9c05e2094c98f9a1b7da6cdf49f8" @@ -1061,49 +921,20 @@ "@babel/parser" "^7.1.0" "@babel/types" "^7.0.0" -"@types/babel__traverse@*", "@types/babel__traverse@^7.0.4", "@types/babel__traverse@^7.0.6": +"@types/babel__traverse@*", "@types/babel__traverse@^7.0.6": version "7.11.1" resolved "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.11.1.tgz" integrity sha512-Vs0hm0vPahPMYi9tDjtP66llufgO3ST16WXaSTtDGEl9cewAl3AibmxWw6TINOqHPT9z0uABKAYjT9jNSg4npw== dependencies: "@babel/types" "^7.3.0" -"@types/cacheable-request@^6.0.1": - version "6.0.2" - resolved "https://registry.yarnpkg.com/@types/cacheable-request/-/cacheable-request-6.0.2.tgz#c324da0197de0a98a2312156536ae262429ff6b9" - integrity sha512-B3xVo+dlKM6nnKTcmm5ZtY/OL8bOAOd2Olee9M1zft65ox50OzjEHW91sDiU9j6cvW8Ejg1/Qkf4xd2kugApUA== - dependencies: - "@types/http-cache-semantics" "*" - "@types/keyv" "*" - "@types/node" "*" - "@types/responselike" "*" - -"@types/caseless@*": - version "0.12.2" - resolved "https://registry.yarnpkg.com/@types/caseless/-/caseless-0.12.2.tgz#f65d3d6389e01eeb458bd54dc8f52b95a9463bc8" - integrity sha512-6ckxMjBBD8URvjB6J3NcnuAn5Pkl7t3TizAg+xdlzzQGSPSmBcXf8KoIH0ua/i+tio+ZRUHEXp0HEmvaR4kt0w== - -"@types/got@^9.6.9": - version "9.6.12" - resolved "https://registry.yarnpkg.com/@types/got/-/got-9.6.12.tgz#fd42a6e1f5f64cd6bb422279b08c30bb5a15a56f" - integrity sha512-X4pj/HGHbXVLqTpKjA2ahI4rV/nNBc9mGO2I/0CgAra+F2dKgMXnENv2SRpemScBzBAI4vMelIVYViQxlSE6xA== - dependencies: - "@types/node" "*" - "@types/tough-cookie" "*" - form-data "^2.5.0" - -"@types/graceful-fs@^4.1.2": +"@types/graceful-fs@^4.1.3": version "4.1.5" - resolved "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.5.tgz" + resolved "https://registry.yarnpkg.com/@types/graceful-fs/-/graceful-fs-4.1.5.tgz#21ffba0d98da4350db64891f92a9e5db3cdb4e15" integrity sha512-anKkLmZZ+xm4p8JWBf4hElkM4XR+EZeA2M9BAkkTldmcyDY4mbdIJnRghDJH3Ov5ooY7/UAoENtmdMSkaAd7Cw== dependencies: "@types/node" "*" -"@types/http-cache-semantics@*": - version "4.0.1" - resolved "https://registry.yarnpkg.com/@types/http-cache-semantics/-/http-cache-semantics-4.0.1.tgz#0ea7b61496902b95890dc4c3a116b60cb8dae812" - integrity sha512-SZs7ekbP8CN0txVG2xVRH6EgKmEm31BOxA07vkFaETzZz1xh+cbt8BcI0slpymvwhx5dlFnQG2rTlPVQn+iRPQ== - "@types/istanbul-lib-coverage@*", "@types/istanbul-lib-coverage@^2.0.0", "@types/istanbul-lib-coverage@^2.0.1": version "2.0.3" resolved "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.3.tgz" @@ -1123,62 +954,33 @@ dependencies: "@types/istanbul-lib-report" "*" -"@types/jest@^27.4.1": - version "27.4.1" - resolved "https://registry.yarnpkg.com/@types/jest/-/jest-27.4.1.tgz#185cbe2926eaaf9662d340cc02e548ce9e11ab6d" - integrity sha512-23iPJADSmicDVrWk+HT58LMJtzLAnB2AgIzplQuq/bSrGaxCrlvRFjGbXmamnnk/mAmCdLStiGqggu28ocUyiw== +"@types/jest@^28.1.6": + version "28.1.6" + resolved "https://registry.yarnpkg.com/@types/jest/-/jest-28.1.6.tgz#d6a9cdd38967d2d746861fb5be6b120e38284dd4" + integrity sha512-0RbGAFMfcBJKOmqRazM8L98uokwuwD5F8rHrv/ZMbrZBwVOWZUyPG6VFNscjYr/vjM3Vu4fRrCPbOs42AfemaQ== dependencies: - jest-matcher-utils "^27.0.0" - pretty-format "^27.0.0" - -"@types/js-yaml@^3.12.1": - version "3.12.6" - resolved "https://registry.npmjs.org/@types/js-yaml/-/js-yaml-3.12.6.tgz" - integrity sha512-cK4XqrLvP17X6c0C8n4iTbT59EixqyXL3Fk8/Rsk4dF3oX4dg70gYUXrXVUUHpnsGMPNlTQMqf+TVmNPX6FmSQ== + jest-matcher-utils "^28.0.0" + pretty-format "^28.0.0" -"@types/js-yaml@^4.0.1": - version "4.0.5" - resolved "https://registry.yarnpkg.com/@types/js-yaml/-/js-yaml-4.0.5.tgz#738dd390a6ecc5442f35e7f03fa1431353f7e138" - integrity sha512-FhpRzf927MNQdRZP0J5DLIdTXhjLYzeUTmLAu69mnVksLH9CJY3IuSeEgbKUki7GQZm0WqDkGzyxju2EZGD2wA== - -"@types/json-schema@^7.0.3": - version "7.0.7" - resolved "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.7.tgz" - integrity sha512-cxWFQVseBm6O9Gbw1IWb8r6OS4OhSt3hPZLkFApLjM8TEXROBuQGLAH2i2gZpcXdLBIrpXuTDhH7Vbm1iXmNGA== +"@types/json-schema@^7.0.9": + version "7.0.11" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.11.tgz#d421b6c527a3037f7c84433fd2c4229e016863d3" + integrity sha512-wOuvG1SN4Us4rez+tylwwwCV1psiNVOkJeM3AUWUNWg/jDQY2+HE/444y5gc+jBmRqASOm2Oeh5c1axHobwRKQ== "@types/json5@^0.0.29": version "0.0.29" resolved "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz" integrity sha1-7ihweulOEdK4J7y+UnC86n8+ce4= -"@types/keyv@*": - version "3.1.3" - resolved "https://registry.yarnpkg.com/@types/keyv/-/keyv-3.1.3.tgz#1c9aae32872ec1f20dcdaee89a9f3ba88f465e41" - integrity sha512-FXCJgyyN3ivVgRoml4h94G/p3kY+u/B86La+QptcqJaWtBWtmc6TtkNfS40n9bIvyLteHh7zXOtgbobORKPbDg== - dependencies: - "@types/node" "*" - -"@types/minipass@*": - version "3.1.2" - resolved "https://registry.yarnpkg.com/@types/minipass/-/minipass-3.1.2.tgz#e2d7f9df0698aff421dcf145b4fc05b8183b9030" - integrity sha512-foLGjgrJkUjLG/o2t2ymlZGEoBNBa/TfoUZ7oCTkOjP1T43UGBJspovJou/l3ZuHvye2ewR5cZNtp2zyWgILMA== - dependencies: - "@types/node" "*" - "@types/node@*": version "17.0.21" resolved "https://registry.yarnpkg.com/@types/node/-/node-17.0.21.tgz#864b987c0c68d07b4345845c3e63b75edd143644" integrity sha512-DBZCJbhII3r90XbQxI8Y9IjjiiOGlZ0Hr32omXIZvwwZ7p4DMMXGrKXVyPfuoBOri9XNtL0UK69jYIBIsRX3QQ== -"@types/node@^10.12.0": - version "10.17.60" - resolved "https://registry.yarnpkg.com/@types/node/-/node-10.17.60.tgz#35f3d6213daed95da7f0f73e75bcc6980e90597b" - integrity sha512-F0KIgDJfy2nA3zMLmWGKxcH2ZVEtCZXHHdOQs2gSaQ27+lNeEfGxzkIw90aXswATX7AZ33tahPbzy6KAfUreVw== - -"@types/node@^17.0.23": - version "17.0.23" - resolved "https://registry.yarnpkg.com/@types/node/-/node-17.0.23.tgz#3b41a6e643589ac6442bdbd7a4a3ded62f33f7da" - integrity sha512-UxDxWn7dl97rKVeVS61vErvw086aCYhDLyvRQZ5Rk65rZKepaFdm53GeqXaKBuOhED4e9uWq34IC3TdSdJJ2Gw== +"@types/node@^18.7.13": + version "18.7.13" + resolved "https://registry.yarnpkg.com/@types/node/-/node-18.7.13.tgz#23e6c5168333480d454243378b69e861ab5c011a" + integrity sha512-46yIhxSe5xEaJZXWdIBP7GU4HDTG8/eo0qd9atdiL+lFpA03y8KS+lkTN834TWJj5767GbWv4n/P6efyTFt1Dw== "@types/normalize-package-data@^2.4.0": version "2.4.0" @@ -1190,23 +992,6 @@ resolved "https://registry.yarnpkg.com/@types/prettier/-/prettier-2.4.4.tgz#5d9b63132df54d8909fce1c3f8ca260fdd693e17" integrity sha512-ReVR2rLTV1kvtlWFyuot+d1pkpG2Fw/XKE3PDAdj57rbM97ttSp9JZ2UsP+2EHTylra9cUf6JA7tGwW1INzUrA== -"@types/request@^2.47.1": - version "2.48.8" - resolved "https://registry.yarnpkg.com/@types/request/-/request-2.48.8.tgz#0b90fde3b655ab50976cb8c5ac00faca22f5a82c" - integrity sha512-whjk1EDJPcAR2kYHRbFl/lKeeKYTi05A15K9bnLInCVroNDCtXce57xKdI0/rQaA3K+6q0eFyUBPmqfSndUZdQ== - dependencies: - "@types/caseless" "*" - "@types/node" "*" - "@types/tough-cookie" "*" - form-data "^2.5.0" - -"@types/responselike@*", "@types/responselike@^1.0.0": - version "1.0.0" - resolved "https://registry.yarnpkg.com/@types/responselike/-/responselike-1.0.0.tgz#251f4fe7d154d2bad125abe1b429b23afd262e29" - integrity sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA== - dependencies: - "@types/node" "*" - "@types/semver@^7.3.9": version "7.3.9" resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.3.9.tgz#152c6c20a7688c30b967ec1841d31ace569863fc" @@ -1217,229 +1002,124 @@ resolved "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.0.tgz" integrity sha512-RJJrrySY7A8havqpGObOB4W92QXKJo63/jFLLgpvOtsGUqbQZ9Sbgl35KMm1DjC6j7AvmmU2bIno+3IyEaemaw== -"@types/stream-buffers@^3.0.3": - version "3.0.4" - resolved "https://registry.yarnpkg.com/@types/stream-buffers/-/stream-buffers-3.0.4.tgz#bf128182da7bc62722ca0ddf5458a9c65f76e648" - integrity sha512-qU/K1tb2yUdhXkLIATzsIPwbtX6BpZk0l3dPW6xqWyhfzzM1ECaQ/8faEnu3CNraLiQ9LHyQQPBGp7N9Fbs25w== - dependencies: - "@types/node" "*" - -"@types/tar@^4.0.3": - version "4.0.5" - resolved "https://registry.yarnpkg.com/@types/tar/-/tar-4.0.5.tgz#5f953f183e36a15c6ce3f336568f6051b7b183f3" - integrity sha512-cgwPhNEabHaZcYIy5xeMtux2EmYBitfqEceBUi2t5+ETy4dW6kswt6WX4+HqLeiiKOo42EXbGiDmVJ2x+vi37Q== - dependencies: - "@types/minipass" "*" - "@types/node" "*" - -"@types/tough-cookie@*": - version "4.0.1" - resolved "https://registry.yarnpkg.com/@types/tough-cookie/-/tough-cookie-4.0.1.tgz#8f80dd965ad81f3e1bc26d6f5c727e132721ff40" - integrity sha512-Y0K95ThC3esLEYD6ZuqNek29lNX2EM1qxV8y2FTLUB0ff5wWrk7az+mLrnNFUnaXcgKye22+sFBRXOgpPILZNg== - -"@types/underscore@^1.8.9": - version "1.11.4" - resolved "https://registry.yarnpkg.com/@types/underscore/-/underscore-1.11.4.tgz#62e393f8bc4bd8a06154d110c7d042a93751def3" - integrity sha512-uO4CD2ELOjw8tasUrAhvnn2W4A0ZECOvMjCivJr4gA9pGgjv+qxKWY9GLTMVEK8ej85BxQOocUyE7hImmSQYcg== - -"@types/ws@^6.0.1": - version "6.0.4" - resolved "https://registry.yarnpkg.com/@types/ws/-/ws-6.0.4.tgz#7797707c8acce8f76d8c34b370d4645b70421ff1" - integrity sha512-PpPrX7SZW9re6+Ha8ojZG4Se8AZXgf0GK6zmfqEuCsY49LFDNXO3SByp44X3dFEqtB73lkCDAdUazhAjVPiNwg== - dependencies: - "@types/node" "*" - "@types/yargs-parser@*": version "20.2.0" resolved "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-20.2.0.tgz" integrity sha512-37RSHht+gzzgYeobbG+KWryeAW8J33Nhr69cjTqSYymXVZEN9NbRYWoYlRtDhHKPVT1FyNKwaTPC1NynKZpzRA== -"@types/yargs@^16.0.0": - version "16.0.4" - resolved "https://registry.yarnpkg.com/@types/yargs/-/yargs-16.0.4.tgz#26aad98dd2c2a38e421086ea9ad42b9e51642977" - integrity sha512-T8Yc9wt/5LbJyCaLiHPReJa0kApcIgJ7Bn735GjItUfh08Z1pJvu8QZqb9s+mMvKV6WUQRV7K2R46YbjMXTTJw== +"@types/yargs@^17.0.8": + version "17.0.11" + resolved "https://registry.yarnpkg.com/@types/yargs/-/yargs-17.0.11.tgz#5e10ca33e219807c0eee0f08b5efcba9b6a42c06" + integrity sha512-aB4y9UDUXTSMxmM4MH+YnuR0g5Cph3FLQBoWoMB21DSvFVAxRVEHEMx3TLh+zUZYMCQtKiqazz0Q4Rre31f/OA== dependencies: "@types/yargs-parser" "*" -"@typescript-eslint/eslint-plugin@^4.20.0": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.22.0.tgz" - integrity sha512-U8SP9VOs275iDXaL08Ln1Fa/wLXfj5aTr/1c0t0j6CdbOnxh+TruXu1p4I0NAvdPBQgoPjHsgKn28mOi0FzfoA== +"@typescript-eslint/eslint-plugin@^5.1.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-5.32.0.tgz#e27e38cffa4a61226327c874a7be965e9a861624" + integrity sha512-CHLuz5Uz7bHP2WgVlvoZGhf0BvFakBJKAD/43Ty0emn4wXWv5k01ND0C0fHcl/Im8Td2y/7h44E9pca9qAu2ew== dependencies: - "@typescript-eslint/experimental-utils" "4.22.0" - "@typescript-eslint/scope-manager" "4.22.0" - debug "^4.1.1" + "@typescript-eslint/scope-manager" "5.32.0" + "@typescript-eslint/type-utils" "5.32.0" + "@typescript-eslint/utils" "5.32.0" + debug "^4.3.4" functional-red-black-tree "^1.0.1" - lodash "^4.17.15" - regexpp "^3.0.0" - semver "^7.3.2" - tsutils "^3.17.1" - -"@typescript-eslint/experimental-utils@4.22.0", "@typescript-eslint/experimental-utils@^4.0.1": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-4.22.0.tgz" - integrity sha512-xJXHHl6TuAxB5AWiVrGhvbGL8/hbiCQ8FiWwObO3r0fnvBdrbWEDy1hlvGQOAWc6qsCWuWMKdVWlLAEMpxnddg== - dependencies: - "@types/json-schema" "^7.0.3" - "@typescript-eslint/scope-manager" "4.22.0" - "@typescript-eslint/types" "4.22.0" - "@typescript-eslint/typescript-estree" "4.22.0" - eslint-scope "^5.0.0" - eslint-utils "^2.0.0" - -"@typescript-eslint/parser@4.8.1": - version "4.8.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-4.8.1.tgz#4fe2fbdbb67485bafc4320b3ae91e34efe1219d1" - integrity sha512-QND8XSVetATHK9y2Ltc/XBl5Ro7Y62YuZKnPEwnNPB8E379fDsvzJ1dMJ46fg/VOmk0hXhatc+GXs5MaXuL5Uw== - dependencies: - "@typescript-eslint/scope-manager" "4.8.1" - "@typescript-eslint/types" "4.8.1" - "@typescript-eslint/typescript-estree" "4.8.1" - debug "^4.1.1" - -"@typescript-eslint/parser@^4.20.0": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-4.22.0.tgz" - integrity sha512-z/bGdBJJZJN76nvAY9DkJANYgK3nlRstRRi74WHm3jjgf2I8AglrSY+6l7ogxOmn55YJ6oKZCLLy+6PW70z15Q== - dependencies: - "@typescript-eslint/scope-manager" "4.22.0" - "@typescript-eslint/types" "4.22.0" - "@typescript-eslint/typescript-estree" "4.22.0" - debug "^4.1.1" - -"@typescript-eslint/scope-manager@4.22.0": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-4.22.0.tgz" - integrity sha512-OcCO7LTdk6ukawUM40wo61WdeoA7NM/zaoq1/2cs13M7GyiF+T4rxuA4xM+6LeHWjWbss7hkGXjFDRcKD4O04Q== - dependencies: - "@typescript-eslint/types" "4.22.0" - "@typescript-eslint/visitor-keys" "4.22.0" - -"@typescript-eslint/scope-manager@4.8.1": - version "4.8.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-4.8.1.tgz#e343c475f8f1d15801b546cb17d7f309b768fdce" - integrity sha512-r0iUOc41KFFbZdPAdCS4K1mXivnSZqXS5D9oW+iykQsRlTbQRfuFRSW20xKDdYiaCoH+SkSLeIF484g3kWzwOQ== - dependencies: - "@typescript-eslint/types" "4.8.1" - "@typescript-eslint/visitor-keys" "4.8.1" - -"@typescript-eslint/types@4.22.0": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/types/-/types-4.22.0.tgz" - integrity sha512-sW/BiXmmyMqDPO2kpOhSy2Py5w6KvRRsKZnV0c4+0nr4GIcedJwXAq+RHNK4lLVEZAJYFltnnk1tJSlbeS9lYA== - -"@typescript-eslint/types@4.8.1": - version "4.8.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-4.8.1.tgz#23829c73c5fc6f4fcd5346a7780b274f72fee222" - integrity sha512-ave2a18x2Y25q5K05K/U3JQIe2Av4+TNi/2YuzyaXLAsDx6UZkz1boZ7nR/N6Wwae2PpudTZmHFXqu7faXfHmA== - -"@typescript-eslint/typescript-estree@4.22.0": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-4.22.0.tgz" - integrity sha512-TkIFeu5JEeSs5ze/4NID+PIcVjgoU3cUQUIZnH3Sb1cEn1lBo7StSV5bwPuJQuoxKXlzAObjYTilOEKRuhR5yg== - dependencies: - "@typescript-eslint/types" "4.22.0" - "@typescript-eslint/visitor-keys" "4.22.0" - debug "^4.1.1" - globby "^11.0.1" - is-glob "^4.0.1" - semver "^7.3.2" - tsutils "^3.17.1" - -"@typescript-eslint/typescript-estree@4.8.1": - version "4.8.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-4.8.1.tgz#7307e3f2c9e95df7daa8dc0a34b8c43b7ec0dd32" - integrity sha512-bJ6Fn/6tW2g7WIkCWh3QRlaSU7CdUUK52shx36/J7T5oTQzANvi6raoTsbwGM11+7eBbeem8hCCKbyvAc0X3sQ== - dependencies: - "@typescript-eslint/types" "4.8.1" - "@typescript-eslint/visitor-keys" "4.8.1" - debug "^4.1.1" - globby "^11.0.1" - is-glob "^4.0.1" - lodash "^4.17.15" - semver "^7.3.2" - tsutils "^3.17.1" - -"@typescript-eslint/visitor-keys@4.22.0": - version "4.22.0" - resolved "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-4.22.0.tgz" - integrity sha512-nnMu4F+s4o0sll6cBSsTeVsT4cwxB7zECK3dFxzEjPBii9xLpq4yqqsy/FU5zMfan6G60DKZSCXAa3sHJZrcYw== - dependencies: - "@typescript-eslint/types" "4.22.0" - eslint-visitor-keys "^2.0.0" + ignore "^5.2.0" + regexpp "^3.2.0" + semver "^7.3.7" + tsutils "^3.21.0" + +"@typescript-eslint/parser@5.32.0", "@typescript-eslint/parser@^5.1.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-5.32.0.tgz#1de243443bc6186fb153b9e395b842e46877ca5d" + integrity sha512-IxRtsehdGV9GFQ35IGm5oKKR2OGcazUoiNBxhRV160iF9FoyuXxjY+rIqs1gfnd+4eL98OjeGnMpE7RF/NBb3A== + dependencies: + "@typescript-eslint/scope-manager" "5.32.0" + "@typescript-eslint/types" "5.32.0" + "@typescript-eslint/typescript-estree" "5.32.0" + debug "^4.3.4" + +"@typescript-eslint/scope-manager@5.32.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-5.32.0.tgz#763386e963a8def470580cc36cf9228864190b95" + integrity sha512-KyAE+tUON0D7tNz92p1uetRqVJiiAkeluvwvZOqBmW9z2XApmk5WSMV9FrzOroAcVxJZB3GfUwVKr98Dr/OjOg== + dependencies: + "@typescript-eslint/types" "5.32.0" + "@typescript-eslint/visitor-keys" "5.32.0" + +"@typescript-eslint/type-utils@5.32.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-5.32.0.tgz#45a14506fe3fb908600b4cef2f70778f7b5cdc79" + integrity sha512-0gSsIhFDduBz3QcHJIp3qRCvVYbqzHg8D6bHFsDMrm0rURYDj+skBK2zmYebdCp+4nrd9VWd13egvhYFJj/wZg== + dependencies: + "@typescript-eslint/utils" "5.32.0" + debug "^4.3.4" + tsutils "^3.21.0" + +"@typescript-eslint/types@5.32.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-5.32.0.tgz#484273021eeeae87ddb288f39586ef5efeb6dcd8" + integrity sha512-EBUKs68DOcT/EjGfzywp+f8wG9Zw6gj6BjWu7KV/IYllqKJFPlZlLSYw/PTvVyiRw50t6wVbgv4p9uE2h6sZrQ== + +"@typescript-eslint/typescript-estree@5.32.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-5.32.0.tgz#282943f34babf07a4afa7b0ff347a8e7b6030d12" + integrity sha512-ZVAUkvPk3ITGtCLU5J4atCw9RTxK+SRc6hXqLtllC2sGSeMFWN+YwbiJR9CFrSFJ3w4SJfcWtDwNb/DmUIHdhg== + dependencies: + "@typescript-eslint/types" "5.32.0" + "@typescript-eslint/visitor-keys" "5.32.0" + debug "^4.3.4" + globby "^11.1.0" + is-glob "^4.0.3" + semver "^7.3.7" + tsutils "^3.21.0" + +"@typescript-eslint/utils@5.32.0", "@typescript-eslint/utils@^5.10.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-5.32.0.tgz#eccb6b672b94516f1afc6508d05173c45924840c" + integrity sha512-W7lYIAI5Zlc5K082dGR27Fczjb3Q57ECcXefKU/f0ajM5ToM0P+N9NmJWip8GmGu/g6QISNT+K6KYB+iSHjXCQ== + dependencies: + "@types/json-schema" "^7.0.9" + "@typescript-eslint/scope-manager" "5.32.0" + "@typescript-eslint/types" "5.32.0" + "@typescript-eslint/typescript-estree" "5.32.0" + eslint-scope "^5.1.1" + eslint-utils "^3.0.0" -"@typescript-eslint/visitor-keys@4.8.1": - version "4.8.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-4.8.1.tgz#794f68ee292d1b2e3aa9690ebedfcb3a8c90e3c3" - integrity sha512-3nrwXFdEYALQh/zW8rFwP4QltqsanCDz4CwWMPiIZmwlk9GlvBeueEIbq05SEq4ganqM0g9nh02xXgv5XI3PeQ== +"@typescript-eslint/visitor-keys@5.32.0": + version "5.32.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-5.32.0.tgz#b9715d0b11fdb5dd10fd0c42ff13987470525394" + integrity sha512-S54xOHZgfThiZ38/ZGTgB2rqx51CMJ5MCfVT2IplK4Q7hgzGfe0nLzLCcenDnc/cSjP568hdeKfeDcBgqNHD/g== dependencies: - "@typescript-eslint/types" "4.8.1" - eslint-visitor-keys "^2.0.0" + "@typescript-eslint/types" "5.32.0" + eslint-visitor-keys "^3.3.0" "@vercel/ncc@^0.33.3": version "0.33.3" resolved "https://registry.yarnpkg.com/@vercel/ncc/-/ncc-0.33.3.tgz#aacc6b3ea9f7b175e0c9a18c9b97e4005a2f4fcc" integrity sha512-JGZ11QV+/ZcfudW2Cz2JVp54/pJNXbsuWRgSh2ZmmZdQBKXqBtIGrwI1Wyx8nlbzAiEFe7FHi4K1zX4//jxTnQ== -abab@^2.0.3, abab@^2.0.5: - version "2.0.5" - resolved "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz" - integrity sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q== - -acorn-globals@^6.0.0: - version "6.0.0" - resolved "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz" - integrity sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg== - dependencies: - acorn "^7.1.1" - acorn-walk "^7.1.1" - -acorn-jsx@^5.3.1: - version "5.3.1" - resolved "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz" - integrity sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng== - -acorn-walk@^7.1.1: - version "7.2.0" - resolved "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz" - integrity sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA== +acorn-jsx@^5.3.2: + version "5.3.2" + resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" + integrity sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ== acorn-walk@^8.1.1: version "8.2.0" resolved "https://registry.yarnpkg.com/acorn-walk/-/acorn-walk-8.2.0.tgz#741210f2e2426454508853a2f44d0ab83b7f69c1" integrity sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA== -acorn@^7.1.1, acorn@^7.4.0: - version "7.4.1" - resolved "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz" - integrity sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A== - -acorn@^8.1.0: - version "8.1.1" - resolved "https://registry.npmjs.org/acorn/-/acorn-8.1.1.tgz" - integrity sha512-xYiIVjNuqtKXMxlRMDc6mZUhXehod4a3gbZ1qRlM7icK4EbxUFNLhWoPblCvFtB2Y9CIqHP3CF/rdxLItaQv8g== - -acorn@^8.2.4, acorn@^8.4.1: +acorn@^8.4.1: version "8.7.0" resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.7.0.tgz#90951fde0f8f09df93549481e5fc141445b791cf" integrity sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ== -agent-base@6: - version "6.0.2" - resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-6.0.2.tgz#49fff58577cfee3f37176feab4c22e00f86d7f77" - integrity sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ== - dependencies: - debug "4" - -aggregate-error@^3.0.0, aggregate-error@^3.1.0: - version "3.1.0" - resolved "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz" - integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA== - dependencies: - clean-stack "^2.0.0" - indent-string "^4.0.0" +acorn@^8.8.0: + version "8.8.0" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.8.0.tgz#88c0187620435c7f6015803f5539dae05a9dbea8" + integrity sha512-QOxyigPVrpZ2GXT+PFyZTl6TtOFc5egxHIP9IlQ+RbupQuX4RkT/Bee4/kQuC02Xkzg84JcT7oLYtDIQxp+v7w== -ajv@^6.10.0, ajv@^6.12.3, ajv@^6.12.4: +ajv@^6.10.0, ajv@^6.12.4: version "6.12.6" resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== @@ -1449,21 +1129,6 @@ ajv@^6.10.0, ajv@^6.12.3, ajv@^6.12.4: json-schema-traverse "^0.4.1" uri-js "^4.2.2" -ajv@^8.0.1: - version "8.1.0" - resolved "https://registry.npmjs.org/ajv/-/ajv-8.1.0.tgz" - integrity sha512-B/Sk2Ix7A36fs/ZkuGLIR86EdjbgR6fsAcbx9lOP/QBSXujDNbVmIS/U4Itz5k8fPFDeVZl/zQ/gJW4Jrq6XjQ== - dependencies: - fast-deep-equal "^3.1.1" - json-schema-traverse "^1.0.0" - require-from-string "^2.0.2" - uri-js "^4.2.2" - -ansi-colors@^4.1.1: - version "4.1.1" - resolved "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz" - integrity sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA== - ansi-escapes@^4.2.1: version "4.3.2" resolved "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz" @@ -1520,98 +1185,64 @@ argparse@^2.0.1: resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38" integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q== -array-includes@^3.1.1: - version "3.1.3" - resolved "https://registry.npmjs.org/array-includes/-/array-includes-3.1.3.tgz" - integrity sha512-gcem1KlBU7c9rB+Rq8/3PPKsK2kjqeEBa3bD5kkQo4nYlOHQCJqIJFqBXDEfwaRuYTT4E+FxA9xez7Gf/e3Q7A== +aria-query@^4.2.2: + version "4.2.2" + resolved "https://registry.yarnpkg.com/aria-query/-/aria-query-4.2.2.tgz#0d2ca6c9aceb56b8977e9fed6aed7e15bbd2f83b" + integrity sha512-o/HelwhuKpTj/frsOsbNLNgnNGVIFsVP/SW2BSF14gVl7kAfMOJ6/8wUAUvG1R1NHKrfG+2sHZTu0yauT1qBrA== + dependencies: + "@babel/runtime" "^7.10.2" + "@babel/runtime-corejs3" "^7.10.2" + +array-includes@^3.1.4, array-includes@^3.1.5: + version "3.1.5" + resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.1.5.tgz#2c320010db8d31031fd2a5f6b3bbd4b1aad31bdb" + integrity sha512-iSDYZMMyTPkiFasVqfuAQnWAYcvO/SeBSCGKePoEthjp4LEMTe4uLc7b025o4jAZpHhihh8xPo99TNWUWWkGDQ== dependencies: call-bind "^1.0.2" - define-properties "^1.1.3" - es-abstract "^1.18.0-next.2" + define-properties "^1.1.4" + es-abstract "^1.19.5" get-intrinsic "^1.1.1" - is-string "^1.0.5" + is-string "^1.0.7" array-union@^2.1.0: version "2.1.0" resolved "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz" integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw== -array.prototype.flat@^1.2.3: - version "1.2.4" - resolved "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.4.tgz" - integrity sha512-4470Xi3GAPAjZqFcljX2xzckv1qeKPizoNkiS0+O4IoPR2ZNpcjE0pkhdihlDouK+x6QOast26B4Q/O9DJnwSg== +array.prototype.flat@^1.2.5: + version "1.3.0" + resolved "https://registry.yarnpkg.com/array.prototype.flat/-/array.prototype.flat-1.3.0.tgz#0b0c1567bf57b38b56b4c97b8aa72ab45e4adc7b" + integrity sha512-12IUEkHsAhA4DY5s0FPgNXIdc8VRSqD9Zp78a5au9abH/SOBrsp082JOWFNTjkMozh8mqcdiKuaLGhPeYztxSw== dependencies: - call-bind "^1.0.0" + call-bind "^1.0.2" define-properties "^1.1.3" - es-abstract "^1.18.0-next.1" - -asn1@~0.2.3: - version "0.2.6" - resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.6.tgz#0d3a7bb6e64e02a90c0303b31f292868ea09a08d" - integrity sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ== - dependencies: - safer-buffer "~2.1.0" - -assert-plus@1.0.0, assert-plus@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" - integrity sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU= - -astral-regex@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz" - integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== - -async-limiter@~1.0.0: - version "1.0.1" - resolved "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz" - integrity sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ== + es-abstract "^1.19.2" + es-shim-unscopables "^1.0.0" -async-wait-until@^2.0.12: - version "2.0.12" - resolved "https://registry.yarnpkg.com/async-wait-until/-/async-wait-until-2.0.12.tgz#8a94683bf29e74642a8bcbb9385f6ea330d4383f" - integrity sha512-SXy/vDs6UPJMG6YeEYOQ4ilA/JnGxk187KPGqFx9O+qVxsjkSl+jH+3P50qSNyMpEmDgr8qOFGOKCJckWb1i7A== +ast-types-flow@^0.0.7: + version "0.0.7" + resolved "https://registry.yarnpkg.com/ast-types-flow/-/ast-types-flow-0.0.7.tgz#f70b735c6bca1a5c9c22d982c3e39e7feba3bdad" + integrity sha512-eBvWn1lvIApYMhzQMsu9ciLfkBY499mFZlNqG+/9WR7PVlroQw0vG30cOQQbaKz3sCEc44TAOu2ykzqXSNnwag== -asynckit@^0.4.0: - version "0.4.0" - resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" - integrity sha1-x57Zf380y48robyXkLzDZkdLS3k= - -aws-sdk@^2.1081.0: - version "2.1082.0" - resolved "https://registry.yarnpkg.com/aws-sdk/-/aws-sdk-2.1082.0.tgz#f6e107cc1473f17292386e7b9f79a5b8bf2a3350" - integrity sha512-aDrUZ63O/ocuC827ursDqFQAm3jhqsJu1DvMCCFg73y+FK9pXXNHp2mwdi3UeeHvtfxISCLCjuyO3VFd/tpVfA== - dependencies: - buffer "4.9.2" - events "1.1.1" - ieee754 "1.1.13" - jmespath "0.16.0" - querystring "0.2.0" - sax "1.2.1" - url "0.10.3" - uuid "3.3.2" - xml2js "0.4.19" - -aws-sign2@~0.7.0: - version "0.7.0" - resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.7.0.tgz#b46e890934a9591f2d2f6f86d7e6a9f1b3fe76a8" - integrity sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg= +axe-core@^4.4.3: + version "4.4.3" + resolved "https://registry.yarnpkg.com/axe-core/-/axe-core-4.4.3.tgz#11c74d23d5013c0fa5d183796729bc3482bd2f6f" + integrity sha512-32+ub6kkdhhWick/UjvEwRchgoetXqTK14INLqbGm5U2TzBkBNF3nQtLYm8ovxSkQWArjEQvftCKryjZaATu3w== -aws4@^1.8.0: - version "1.11.0" - resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.11.0.tgz#d61f46d83b2519250e2784daf5b09479a8b41c59" - integrity sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA== +axobject-query@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/axobject-query/-/axobject-query-2.2.0.tgz#943d47e10c0b704aa42275e20edf3722648989be" + integrity sha512-Td525n+iPOOyUQIeBfcASuG6uJsDOITl7Mds5gFyerkWiX7qhUTdYUBlSgNMyVqtSJqwpt1kXGLdUt6SykLMRA== -babel-jest@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/babel-jest/-/babel-jest-27.5.1.tgz#a1bf8d61928edfefd21da27eb86a695bfd691444" - integrity sha512-cdQ5dXjGRd0IBRATiQ4mZGlGlRE8kJpjPOixdNRdT+m3UcNqmYWN6rK6nvtXYfY3D76cb8s/O1Ss8ea24PIwcg== +babel-jest@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/babel-jest/-/babel-jest-28.1.3.tgz#c1187258197c099072156a0a121c11ee1e3917d5" + integrity sha512-epUaPOEWMk3cWX0M/sPvCHHCe9fMFAa/9hXEgKP8nFfNl/jlGkE9ucq9NqkZGXLDduCJYS0UvSlPUwC0S+rH6Q== dependencies: - "@jest/transform" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/transform" "^28.1.3" "@types/babel__core" "^7.1.14" babel-plugin-istanbul "^6.1.1" - babel-preset-jest "^27.5.1" + babel-preset-jest "^28.1.3" chalk "^4.0.0" graceful-fs "^4.2.9" slash "^3.0.0" @@ -1627,14 +1258,14 @@ babel-plugin-istanbul@^6.1.1: istanbul-lib-instrument "^5.0.4" test-exclude "^6.0.0" -babel-plugin-jest-hoist@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-27.5.1.tgz#9be98ecf28c331eb9f5df9c72d6f89deb8181c2e" - integrity sha512-50wCwD5EMNW4aRpOwtqzyZHIewTYNxLA4nhB+09d8BIssfNfzBRhkBIHiaPv1Si226TQSvp8gxAJm2iY2qs2hQ== +babel-plugin-jest-hoist@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-28.1.3.tgz#1952c4d0ea50f2d6d794353762278d1d8cca3fbe" + integrity sha512-Ys3tUKAmfnkRUpPdpa98eYrAR0nV+sSFUZZEGuQ2EbFd1y4SOLtD5QDNHAq+bb9a+bbXvYQC4b+ID/THIMcU6Q== dependencies: "@babel/template" "^7.3.3" "@babel/types" "^7.3.3" - "@types/babel__core" "^7.0.0" + "@types/babel__core" "^7.1.14" "@types/babel__traverse" "^7.0.6" babel-preset-current-node-syntax@^1.0.0: @@ -1655,12 +1286,12 @@ babel-preset-current-node-syntax@^1.0.0: "@babel/plugin-syntax-optional-chaining" "^7.8.3" "@babel/plugin-syntax-top-level-await" "^7.8.3" -babel-preset-jest@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/babel-preset-jest/-/babel-preset-jest-27.5.1.tgz#91f10f58034cb7989cb4f962b69fa6eef6a6bc81" - integrity sha512-Nptf2FzlPCWYuJg41HBqXVT8ym6bXOevuCTbhxlUpjwtysGaIWFvDEjp4y+G7fl13FgOdjs7P/DmErqH7da0Ag== +babel-preset-jest@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/babel-preset-jest/-/babel-preset-jest-28.1.3.tgz#5dfc20b99abed5db994406c2b9ab94c73aaa419d" + integrity sha512-L+fupJvlWAHbQfn74coNX3zf60LXMJsezNvvx8eIh7iOR1luJ1poxYgQk1F8PYtNq/6QODDHCqsSnTFSWC491A== dependencies: - babel-plugin-jest-hoist "^27.5.1" + babel-plugin-jest-hoist "^28.1.3" babel-preset-current-node-syntax "^1.0.0" balanced-match@^1.0.0: @@ -1668,33 +1299,6 @@ balanced-match@^1.0.0: resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== -base-64@^1.0.0: - version "1.0.0" - resolved "https://registry.npmjs.org/base-64/-/base-64-1.0.0.tgz" - integrity sha512-kwDPIFCGx0NZHog36dj+tHiwP4QMzsZ3AgMViUBKI0+V5n4U0ufTCUMhnQ04diaRI8EX/QcPfql7zlhZ7j4zgg== - -base64-js@^1.0.2: - version "1.5.1" - resolved "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz" - integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== - -base64url@^3.0.1: - version "3.0.1" - resolved "https://registry.npmjs.org/base64url/-/base64url-3.0.1.tgz" - integrity sha512-ir1UPr3dkwexU7FdV8qBBbNDRUhMmIekYMFZfi+C/sLNnRESKPl23nB9b2pltqfOQNnGzsDdId90AEtG5tCx4A== - -bcrypt-pbkdf@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz#a4301d389b6a43f9b67ff3ca11a3f6637e360e9e" - integrity sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4= - dependencies: - tweetnacl "^0.14.3" - -before-after-hook@^2.2.0: - version "2.2.2" - resolved "https://registry.yarnpkg.com/before-after-hook/-/before-after-hook-2.2.2.tgz#a6e8ca41028d90ee2c24222f201c90956091613e" - integrity sha512-3pZEU3NT5BFUo/AD5ERPWOgQOCZITni6iavr5AUw5AUwQjMlI0kzu5btnyD39AF0gUEsDPwJT+oY1ORBJijPjQ== - brace-expansion@^1.1.7: version "1.1.11" resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" @@ -1710,22 +1314,6 @@ braces@^3.0.1: dependencies: fill-range "^7.0.1" -browser-process-hrtime@^1.0.0: - version "1.0.0" - resolved "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz" - integrity sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow== - -browserslist@^4.14.5: - version "4.16.6" - resolved "https://registry.npmjs.org/browserslist/-/browserslist-4.16.6.tgz" - integrity sha512-Wspk/PqO+4W9qp5iUTJsa1B/QrYn1keNCcEP5OvP7WBwT4KaDly0uONYmC6Xa3Z5IqnUgS0KcgLYu1l74x0ZXQ== - dependencies: - caniuse-lite "^1.0.30001219" - colorette "^1.2.2" - electron-to-chromium "^1.3.723" - escalade "^3.1.1" - node-releases "^1.1.71" - browserslist@^4.17.5: version "4.19.3" resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.19.3.tgz#29b7caad327ecf2859485f696f9604214bedd383" @@ -1737,6 +1325,16 @@ browserslist@^4.17.5: node-releases "^2.0.2" picocolors "^1.0.0" +browserslist@^4.20.2, browserslist@^4.21.0: + version "4.21.3" + resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.21.3.tgz#5df277694eb3c48bc5c4b05af3e8b7e09c5a6d1a" + integrity sha512-898rgRXLAyRkM1GryrrBHGkqA5hlpkV5MhtZwg9QXeiyLUYs2k00Un05aX5l2/yJIOObYKOpS2JNo8nJDE7fWQ== + dependencies: + caniuse-lite "^1.0.30001370" + electron-to-chromium "^1.4.202" + node-releases "^2.0.6" + update-browserslist-db "^1.0.5" + bs-logger@0.x: version "0.2.6" resolved "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz" @@ -1756,55 +1354,10 @@ buffer-from@^1.0.0: resolved "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz" integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A== -buffer@4.9.2: - version "4.9.2" - resolved "https://registry.npmjs.org/buffer/-/buffer-4.9.2.tgz" - integrity sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg== - dependencies: - base64-js "^1.0.2" - ieee754 "^1.1.4" - isarray "^1.0.0" - -byline@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/byline/-/byline-5.0.0.tgz#741c5216468eadc457b03410118ad77de8c1ddb1" - integrity sha1-dBxSFkaOrcRXsDQQEYrXfejB3bE= - -cac@^6.6.1: - version "6.7.12" - resolved "https://registry.yarnpkg.com/cac/-/cac-6.7.12.tgz#6fb5ea2ff50bd01490dbda497f4ae75a99415193" - integrity sha512-rM7E2ygtMkJqD9c7WnFU6fruFcN3xe4FM5yUmgxhZzIKJk4uHl9U/fhwdajGFQbQuv43FAUo1Fe8gX/oIKDeSA== - -cacheable-lookup@^5.0.3: - version "5.0.4" - resolved "https://registry.yarnpkg.com/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz#5a6b865b2c44357be3d5ebc2a467b032719a7005" - integrity sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA== - -cacheable-request@^6.0.0: - version "6.1.0" - resolved "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz" - integrity sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg== - dependencies: - clone-response "^1.0.2" - get-stream "^5.1.0" - http-cache-semantics "^4.0.0" - keyv "^3.0.0" - lowercase-keys "^2.0.0" - normalize-url "^4.1.0" - responselike "^1.0.2" - -cacheable-request@^7.0.2: - version "7.0.2" - resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-7.0.2.tgz#ea0d0b889364a25854757301ca12b2da77f91d27" - integrity sha512-pouW8/FmiPQbuGpkXQ9BAPv/Mo5xDGANgSNXzTzJ8DrKGuXOssM4wIQRjfanNRh3Yu5cfYPvcorqbhg2KIJtew== - dependencies: - clone-response "^1.0.2" - get-stream "^5.1.0" - http-cache-semantics "^4.0.0" - keyv "^4.0.0" - lowercase-keys "^2.0.0" - normalize-url "^6.0.1" - responselike "^2.0.0" +builtin-modules@^3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-3.3.0.tgz#cae62812b89801e9656336e46223e030386be7b6" + integrity sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw== call-bind@^1.0.0, call-bind@^1.0.2: version "1.0.2" @@ -1824,30 +1377,20 @@ camelcase@^5.3.1: resolved "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz" integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== -camelcase@^6.0.0: - version "6.2.0" - resolved "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz" - integrity sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg== - camelcase@^6.2.0: version "6.3.0" resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.3.0.tgz#5685b95eb209ac9c0c177467778c9c84df58ba9a" integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== -caniuse-lite@^1.0.30001219: - version "1.0.30001230" - resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001230.tgz" - integrity sha512-5yBd5nWCBS+jWKTcHOzXwo5xzcj4ePE/yjtkZyUV1BTUmrBaA9MRGC+e7mxnqXSA90CmCA8L3eKLaSUkt099IQ== - caniuse-lite@^1.0.30001312: version "1.0.30001312" resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001312.tgz#e11eba4b87e24d22697dae05455d5aea28550d5f" integrity sha512-Wiz1Psk2MEK0pX3rUzWaunLTZzqS2JYZFzNKqAiJGiuxIjRPLgV6+VDPOg6lQOUxmDwhTlh198JsTTi8Hzw6aQ== -caseless@~0.12.0: - version "0.12.0" - resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc" - integrity sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw= +caniuse-lite@^1.0.30001370: + version "1.0.30001374" + resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001374.tgz#3dab138e3f5485ba2e74bd13eca7fe1037ce6f57" + integrity sha512-mWvzatRx3w+j5wx/mpFN5v5twlPrabG8NqX2c6e45LCpymdoGqNvRkRutFUqpRTXKFQFNQJasvK0YT7suW6/Hw== chalk@^2.0.0: version "2.4.2" @@ -1879,21 +1422,16 @@ char-regex@^1.0.2: resolved "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz" integrity sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw== -chownr@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/chownr/-/chownr-2.0.0.tgz#15bfbe53d2eab4cf70f18a8cd68ebe5b3cb1dece" - integrity sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ== - -ci-info@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz" - integrity sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ== - ci-info@^3.2.0: version "3.3.0" resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-3.3.0.tgz#b4ed1fb6818dea4803a55c623041f9165d2066b2" integrity sha512-riT/3vI5YpVH6/qomlDnJow6TBee2PBKSEpx3O32EGPYbWGIRsIlGRms3Sm74wYE1JMo8RnO04Hb12+v1J5ICw== +ci-info@^3.3.2: + version "3.3.2" + resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-3.3.2.tgz#6d2967ffa407466481c6c90b6e16b3098f080128" + integrity sha512-xmDt/QIAdeZ9+nfdPsaBCpMvHNLFiLdjj59qjqn+6iPe6YmHGQ35sBnQ8uslRBXFmXkiZQOJRjvQeoGppoTjjg== + cjs-module-lexer@^1.0.0: version "1.2.2" resolved "https://registry.yarnpkg.com/cjs-module-lexer/-/cjs-module-lexer-1.2.2.tgz#9f84ba3244a512f3a54e5277e8eef4c489864e40" @@ -1906,11 +1444,6 @@ clean-regexp@^1.0.0: dependencies: escape-string-regexp "^1.0.5" -clean-stack@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b" - integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A== - cliui@^7.0.2: version "7.0.4" resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz#a0265ee655476fc807aea9df3df8df7783808b4f" @@ -1920,13 +1453,6 @@ cliui@^7.0.2: strip-ansi "^6.0.0" wrap-ansi "^7.0.0" -clone-response@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.2.tgz#d1dc973920314df67fbeb94223b4ee350239e96b" - integrity sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws= - dependencies: - mimic-response "^1.0.0" - co@^4.6.0: version "4.6.0" resolved "https://registry.npmjs.org/co/-/co-4.6.0.tgz" @@ -1961,45 +1487,11 @@ color-name@~1.1.4: resolved "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== -colorette@^1.2.2: - version "1.2.2" - resolved "https://registry.npmjs.org/colorette/-/colorette-1.2.2.tgz" - integrity sha512-MKGMzyfeuutC/ZJ1cba9NqcNpfeqMUcYmyF1ZFY6/Cn7CNSAKx6a+s48sqLqyAiZuaP2TcqMhoo+dlwFnVxT9w== - -combined-stream@^1.0.6, combined-stream@^1.0.8, combined-stream@~1.0.6: - version "1.0.8" - resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" - integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== - dependencies: - delayed-stream "~1.0.0" - -commander-ts@^0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/commander-ts/-/commander-ts-0.2.0.tgz#14391337c1c725399cdfca5717da8e4fd0688eda" - integrity sha512-9XaUF3/3nmVtkDmAkijjhgEcwrMwKewaAJtN+GyTJBG3CJ5DfGB/JsXCVZcBW6SZB+QqiJxbK3e2/62tweMO8g== - dependencies: - commander "^7.2.0" - -commander@^7.2.0: - version "7.2.0" - resolved "https://registry.yarnpkg.com/commander/-/commander-7.2.0.tgz#a36cb57d0b501ce108e4d20559a150a391d97ab7" - integrity sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw== - -commander@^9.0.0: - version "9.0.0" - resolved "https://registry.yarnpkg.com/commander/-/commander-9.0.0.tgz#86d58f24ee98126568936bd1d3574e0308a99a40" - integrity sha512-JJfP2saEKbQqvW+FI93OYUB4ByV5cizMpFMiiJI8xDbBvQvSkIk0VvQdn1CZ8mqAO8Loq2h0gYTYtDFUZUeERw== - concat-map@0.0.1: version "0.0.1" resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s= -contains-path@^0.1.0: - version "0.1.0" - resolved "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz" - integrity sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo= - convert-source-map@^1.4.0, convert-source-map@^1.6.0, convert-source-map@^1.7.0: version "1.7.0" resolved "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz" @@ -2007,10 +1499,10 @@ convert-source-map@^1.4.0, convert-source-map@^1.6.0, convert-source-map@^1.7.0: dependencies: safe-buffer "~5.1.1" -core-util-is@1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" - integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac= +core-js-pure@^3.20.2: + version "3.24.1" + resolved "https://registry.yarnpkg.com/core-js-pure/-/core-js-pure-3.24.1.tgz#8839dde5da545521bf282feb7dc6d0b425f39fd3" + integrity sha512-r1nJk41QLLPyozHUUPmILCEMtMw24NG4oWK6RbsDdjzQgg9ZvrUsPBj1MnG0wXXp1DCDU6j+wUvEmBSrtRbLXg== create-require@^1.1.0: version "1.1.1" @@ -2033,45 +1525,10 @@ cross-spawn@^7.0.1, cross-spawn@^7.0.2, cross-spawn@^7.0.3: shebang-command "^2.0.0" which "^2.0.1" -cssom@^0.4.4: - version "0.4.4" - resolved "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz" - integrity sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw== - -cssom@~0.3.6: - version "0.3.8" - resolved "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz" - integrity sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg== - -cssstyle@^2.3.0: - version "2.3.0" - resolved "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz" - integrity sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A== - dependencies: - cssom "~0.3.6" - -dashdash@^1.12.0: - version "1.14.1" - resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0" - integrity sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA= - dependencies: - assert-plus "^1.0.0" - -data-urls@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz" - integrity sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ== - dependencies: - abab "^2.0.3" - whatwg-mimetype "^2.3.0" - whatwg-url "^8.0.0" - -debug@4, debug@^4.0.1, debug@^4.1.0, debug@^4.1.1: - version "4.3.3" - resolved "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz" - integrity sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q== - dependencies: - ms "2.1.2" +damerau-levenshtein@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz#b43d286ccbd36bc5b2f7ed41caf2d0aba1f8a6e7" + integrity sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA== debug@^2.6.9: version "2.6.9" @@ -2080,31 +1537,33 @@ debug@^2.6.9: dependencies: ms "2.0.0" -decimal.js@^10.2.1: - version "10.2.1" - resolved "https://registry.npmjs.org/decimal.js/-/decimal.js-10.2.1.tgz" - integrity sha512-KaL7+6Fw6i5A2XSnsbhm/6B+NuEA7TZ4vqxnd5tXz9sbKtrN9Srj8ab4vKVdK8YAqZO9P1kg45Y6YLoduPf+kw== +debug@^3.2.7: + version "3.2.7" + resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a" + integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ== + dependencies: + ms "^2.1.1" -decompress-response@^3.3.0: - version "3.3.0" - resolved "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz" - integrity sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M= +debug@^4.1.0, debug@^4.1.1: + version "4.3.3" + resolved "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz" + integrity sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q== dependencies: - mimic-response "^1.0.0" + ms "2.1.2" -decompress-response@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-6.0.0.tgz#ca387612ddb7e104bd16d85aab00d5ecf09c66fc" - integrity sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ== +debug@^4.3.2, debug@^4.3.4: + version "4.3.4" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" + integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== dependencies: - mimic-response "^3.1.0" + ms "2.1.2" dedent@^0.7.0: version "0.7.0" resolved "https://registry.npmjs.org/dedent/-/dedent-0.7.0.tgz" integrity sha1-JJXduvbrh0q7Dhvp3yLS5aVEMmw= -deep-is@^0.1.3, deep-is@~0.1.3: +deep-is@^0.1.3: version "0.1.3" resolved "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz" integrity sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ= @@ -2114,16 +1573,6 @@ deepmerge@^4.2.2: resolved "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz" integrity sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg== -defer-to-connect@^1.0.1: - version "1.1.3" - resolved "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz" - integrity sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ== - -defer-to-connect@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-2.0.1.tgz#8016bdb4143e4632b77a3449c6236277de520587" - integrity sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg== - define-properties@^1.1.3: version "1.1.3" resolved "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz" @@ -2131,30 +1580,23 @@ define-properties@^1.1.3: dependencies: object-keys "^1.0.12" -delayed-stream@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" - integrity sha1-3zrhmayt+31ECqrgsp4icrJOxhk= - -depd@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz" - integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw== - -deprecation@^2.0.0, deprecation@^2.3.1: - version "2.3.1" - resolved "https://registry.npmjs.org/deprecation/-/deprecation-2.3.1.tgz" - integrity sha512-xmHIy4F3scKVwMsQ4WnVaS8bHOx0DmVwRywosKhaILI0ywMDWPtBSku2HNxRvF7jtwDRsoEwYQSfbxj8b7RlJQ== +define-properties@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.4.tgz#0b14d7bd7fbeb2f3572c3a7eda80ea5d57fb05b1" + integrity sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA== + dependencies: + has-property-descriptors "^1.0.0" + object-keys "^1.1.1" detect-newline@^3.0.0: version "3.1.0" resolved "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz" integrity sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA== -diff-sequences@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/diff-sequences/-/diff-sequences-27.5.1.tgz#eaecc0d327fd68c8d9672a1e64ab8dccb2ef5327" - integrity sha512-k1gCAXAsNgLwEL+Y8Wvl+M6oEFj5bgazfZULpS5CneoPPXRaCCW7dm+q21Ky2VEE5X+VeRDBVg1Pcvvsr4TtNQ== +diff-sequences@^28.1.1: + version "28.1.1" + resolved "https://registry.yarnpkg.com/diff-sequences/-/diff-sequences-28.1.1.tgz#9989dc731266dc2903457a70e996f3a041913ac6" + integrity sha512-FU0iFaH/E23a+a718l8Qa/19bF9p06kgE0KipMOMadwa3SjnaElKzPaUC0vnibs6/B/9ni97s61mcejk8W1fQw== diff@^4.0.1: version "4.0.2" @@ -2168,13 +1610,12 @@ dir-glob@^3.0.1: dependencies: path-type "^4.0.0" -doctrine@1.5.0: - version "1.5.0" - resolved "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz" - integrity sha1-N53Ocw9hZvds76TmcHoVmwLFpvo= +doctrine@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-2.1.0.tgz#5cd01fc101621b42c4cd7f5d1a66243716d3f39d" + integrity sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw== dependencies: esutils "^2.0.2" - isarray "^1.0.0" doctrine@^3.0.0: version "3.0.0" @@ -2183,88 +1624,73 @@ doctrine@^3.0.0: dependencies: esutils "^2.0.2" -domexception@^2.0.1: - version "2.0.1" - resolved "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz" - integrity sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg== - dependencies: - webidl-conversions "^5.0.0" - -duplexer3@^0.1.4: - version "0.1.4" - resolved "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz" - integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI= - -ecc-jsbn@~0.1.1: - version "0.1.2" - resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9" - integrity sha1-OoOpBOVDUyh4dMVkt1SThoSamMk= - dependencies: - jsbn "~0.1.0" - safer-buffer "^2.1.0" - -electron-to-chromium@^1.3.723: - version "1.3.738" - resolved "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.738.tgz" - integrity sha512-vCMf4gDOpEylPSLPLSwAEsz+R3ShP02Y3cAKMZvTqule3XcPp7tgc/0ESI7IS6ZeyBlGClE50N53fIOkcIVnpw== +electron-to-chromium@^1.4.202: + version "1.4.211" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.211.tgz#afaa8b58313807501312d598d99b953568d60f91" + integrity sha512-BZSbMpyFQU0KBJ1JG26XGeFI3i4op+qOYGxftmZXFZoHkhLgsSv4DHDJfl8ogII3hIuzGt51PaZ195OVu0yJ9A== electron-to-chromium@^1.4.71: version "1.4.73" resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.73.tgz#422f6f514315bcace9615903e4a9b6b9fa283137" integrity sha512-RlCffXkE/LliqfA5m29+dVDPB2r72y2D2egMMfIy3Le8ODrxjuZNVo4NIC2yPL01N4xb4nZQLwzi6Z5tGIGLnA== -emittery@^0.8.1: - version "0.8.1" - resolved "https://registry.yarnpkg.com/emittery/-/emittery-0.8.1.tgz#bb23cc86d03b30aa75a7f734819dee2e1ba70860" - integrity sha512-uDfvUjVrfGJJhymx/kz6prltenw1u7WrCg1oa94zYY8xxVpLLUu045LAT0dhDZdXG58/EpPL/5kA180fQ/qudg== +emittery@^0.10.2: + version "0.10.2" + resolved "https://registry.yarnpkg.com/emittery/-/emittery-0.10.2.tgz#902eec8aedb8c41938c46e9385e9db7e03182933" + integrity sha512-aITqOwnLanpHLNXZJENbOgjUBeHocD+xsSJmNrjovKBW5HbSpW3d1pEls7GFQPUWXiwG9+0P4GtHfEqC/4M0Iw== emoji-regex@^8.0.0: version "8.0.0" resolved "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz" integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== -end-of-stream@^1.1.0: - version "1.4.4" - resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0" - integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q== - dependencies: - once "^1.4.0" - -enquirer@^2.3.5: - version "2.3.6" - resolved "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz" - integrity sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg== - dependencies: - ansi-colors "^4.1.1" +emoji-regex@^9.2.2: + version "9.2.2" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" + integrity sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg== -error-ex@^1.2.0, error-ex@^1.3.1: +error-ex@^1.3.1: version "1.3.2" resolved "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz" integrity sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g== dependencies: is-arrayish "^0.2.1" -es-abstract@^1.18.0-next.1, es-abstract@^1.18.0-next.2: - version "1.18.0" - resolved "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0.tgz" - integrity sha512-LJzK7MrQa8TS0ja2w3YNLzUgJCGPdPOV1yVvezjNnS89D+VR08+Szt2mz3YB2Dck/+w5tfIq/RoUAFqJJGM2yw== +es-abstract@^1.19.0, es-abstract@^1.19.1, es-abstract@^1.19.2, es-abstract@^1.19.5: + version "1.20.1" + resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.20.1.tgz#027292cd6ef44bd12b1913b828116f54787d1814" + integrity sha512-WEm2oBhfoI2sImeM4OF2zE2V3BYdSF+KnSi9Sidz51fQHd7+JuF8Xgcj9/0o+OWeIeIS/MiuNnlruQrJf16GQA== dependencies: call-bind "^1.0.2" es-to-primitive "^1.2.1" function-bind "^1.1.1" + function.prototype.name "^1.1.5" get-intrinsic "^1.1.1" + get-symbol-description "^1.0.0" has "^1.0.3" - has-symbols "^1.0.2" - is-callable "^1.2.3" - is-negative-zero "^2.0.1" - is-regex "^1.1.2" - is-string "^1.0.5" - object-inspect "^1.9.0" + has-property-descriptors "^1.0.0" + has-symbols "^1.0.3" + internal-slot "^1.0.3" + is-callable "^1.2.4" + is-negative-zero "^2.0.2" + is-regex "^1.1.4" + is-shared-array-buffer "^1.0.2" + is-string "^1.0.7" + is-weakref "^1.0.2" + object-inspect "^1.12.0" object-keys "^1.1.1" object.assign "^4.1.2" - string.prototype.trimend "^1.0.4" - string.prototype.trimstart "^1.0.4" - unbox-primitive "^1.0.0" + regexp.prototype.flags "^1.4.3" + string.prototype.trimend "^1.0.5" + string.prototype.trimstart "^1.0.5" + unbox-primitive "^1.0.2" + +es-shim-unscopables@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz#702e632193201e3edf8713635d083d378e510241" + integrity sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w== + dependencies: + has "^1.0.3" es-to-primitive@^1.2.1: version "1.2.1" @@ -2290,43 +1716,43 @@ escape-string-regexp@^2.0.0: resolved "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz" integrity sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w== -escodegen@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/escodegen/-/escodegen-2.0.0.tgz" - integrity sha512-mmHKys/C8BFUGI+MAWNcSYoORYLMdPzjrknd2Vc+bUsjN5bXcr8EhrNB+UTqfL1y3I9c4fw2ihgtMPQLBRiQxw== - dependencies: - esprima "^4.0.1" - estraverse "^5.2.0" - esutils "^2.0.2" - optionator "^0.8.1" - optionalDependencies: - source-map "~0.6.1" +escape-string-regexp@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" + integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== -eslint-config-prettier@8.1.0: - version "8.1.0" - resolved "https://registry.yarnpkg.com/eslint-config-prettier/-/eslint-config-prettier-8.1.0.tgz#4ef1eaf97afe5176e6a75ddfb57c335121abc5a6" - integrity sha512-oKMhGv3ihGbCIimCAjqkdzx2Q+jthoqnXSP+d86M9tptwugycmTFdVR4IpLgq2c4SHifbwO90z2fQ8/Aio73yw== +eslint-config-prettier@8.5.0: + version "8.5.0" + resolved "https://registry.yarnpkg.com/eslint-config-prettier/-/eslint-config-prettier-8.5.0.tgz#5a81680ec934beca02c7b1a61cf8ca34b66feab1" + integrity sha512-obmWKLUNCnhtQRKc+tmnYuQl0pFU1ibYJQ5BGhTVB08bHe9wC8qUeG7c08dj9XX+AuPj1YSGSQIHl1pnDHZR0Q== eslint-config-prettier@>=8.0.0: version "8.2.0" resolved "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.2.0.tgz" integrity sha512-dWV9EVeSo2qodOPi1iBYU/x6F6diHv8uujxbxr77xExs3zTAlNXvVZKiyLsQGNz7yPV2K49JY5WjPzNIuDc2Bw== -eslint-import-resolver-node@^0.3.4: - version "0.3.4" - resolved "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.4.tgz" - integrity sha512-ogtf+5AB/O+nM6DIeBUNr2fuT7ot9Qg/1harBfBtaP13ekEWFQEEMP94BCB7zaNW3gyY+8SHYF00rnqYwXKWOA== +eslint-import-resolver-node@^0.3.6: + version "0.3.6" + resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.6.tgz#4048b958395da89668252001dbd9eca6b83bacbd" + integrity sha512-0En0w03NRVMn9Uiyn8YRPDKvWjxCWkslUEhGNTdGx15RvPJYQ+lbOlqrlNI2vEAs4pDYK4f/HN2TbDmk5TP0iw== dependencies: - debug "^2.6.9" - resolve "^1.13.1" + debug "^3.2.7" + resolve "^1.20.0" -eslint-module-utils@^2.6.0: - version "2.6.0" - resolved "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.0.tgz" - integrity sha512-6j9xxegbqe8/kZY8cYpcp0xhbK0EgJlg3g9mib3/miLaExuuwc3n5UEfSnU6hWMbT0FAYVvDbL9RrRgpUeQIvA== +eslint-module-utils@^2.7.3: + version "2.7.3" + resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.7.3.tgz#ad7e3a10552fdd0642e1e55292781bd6e34876ee" + integrity sha512-088JEC7O3lDZM9xGe0RerkOMd0EjFl+Yvd1jPWIkMT5u3H9+HC34mWWPnqPrN13gieT9pBOO+Qt07Nb/6TresQ== dependencies: - debug "^2.6.9" - pkg-dir "^2.0.0" + debug "^3.2.7" + find-up "^2.1.0" + +eslint-plugin-escompat@^3.1.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-escompat/-/eslint-plugin-escompat-3.2.0.tgz#7d40efe8721d8fccb5d300f4fbda44f350a27301" + integrity sha512-obXAKKiZE/wB2fgIw0ZxCmp+8vpDsUw2inkaok1i7OVxY4cEds4Y9YCoky0f5V+q8rqZpTUJDv1R9ykWbXLX8Q== + dependencies: + browserslist "^4.21.0" eslint-plugin-eslint-comments@^3.2.0: version "3.2.0" @@ -2336,172 +1762,217 @@ eslint-plugin-eslint-comments@^3.2.0: escape-string-regexp "^1.0.5" ignore "^5.0.5" -eslint-plugin-github@^4.1.1: - version "4.1.3" - resolved "https://registry.npmjs.org/eslint-plugin-github/-/eslint-plugin-github-4.1.3.tgz" - integrity sha512-90MZbFt61jO+hCprPAPH1VSLylQPsynmMId79z/fCGQEhm3qzbAkUG3NqhNK8ltWZfFbXFYuXB4S5BnNttYGhQ== +eslint-plugin-filenames@^1.3.2: + version "1.3.2" + resolved "https://registry.yarnpkg.com/eslint-plugin-filenames/-/eslint-plugin-filenames-1.3.2.tgz#7094f00d7aefdd6999e3ac19f72cea058e590cf7" + integrity sha512-tqxJTiEM5a0JmRCUYQmxw23vtTxrb2+a3Q2mMOPhFxvt7ZQQJmdiuMby9B/vUAuVMghyP7oET+nIf6EO6CBd/w== + dependencies: + lodash.camelcase "4.3.0" + lodash.kebabcase "4.1.1" + lodash.snakecase "4.1.1" + lodash.upperfirst "4.3.1" + +eslint-plugin-github@4.3.7: + version "4.3.7" + resolved "https://registry.yarnpkg.com/eslint-plugin-github/-/eslint-plugin-github-4.3.7.tgz#596416a81240dcd1d3ba1cab6ddfed0a1827bf1a" + integrity sha512-tYZdXvAEz4JCMrC4NHIUoJTsLUvydCxff5OqB5hgU0vQbLmMkw6VOipN2KNe+T06pEhAWs1KBEwyq9cmMWRe7A== dependencies: - "@typescript-eslint/eslint-plugin" "^4.20.0" - "@typescript-eslint/parser" "^4.20.0" + "@typescript-eslint/eslint-plugin" "^5.1.0" + "@typescript-eslint/parser" "^5.1.0" eslint-config-prettier ">=8.0.0" + eslint-plugin-escompat "^3.1.0" eslint-plugin-eslint-comments "^3.2.0" - eslint-plugin-import "^2.22.1" - eslint-plugin-prettier "^3.3.1" + eslint-plugin-filenames "^1.3.2" + eslint-plugin-i18n-text "^1.0.1" + eslint-plugin-import "^2.25.2" + eslint-plugin-jsx-a11y "^6.6.0" + eslint-plugin-no-only-tests "^2.6.0" + eslint-plugin-prettier "^4.0.0" eslint-rule-documentation ">=1.0.0" + jsx-ast-utils "^3.3.2" prettier "^2.2.1" svg-element-attributes "^1.3.1" -eslint-plugin-import@^2.22.1: - version "2.22.1" - resolved "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.22.1.tgz" - integrity sha512-8K7JjINHOpH64ozkAhpT3sd+FswIZTfMZTjdx052pnWrgRCVfp8op9tbjpAk3DdUeI/Ba4C8OjdC0r90erHEOw== +eslint-plugin-i18n-text@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-i18n-text/-/eslint-plugin-i18n-text-1.0.1.tgz#69ce14f9af7d135cbe8114b1b144a57bb83291dc" + integrity sha512-3G3UetST6rdqhqW9SfcfzNYMpQXS7wNkJvp6dsXnjzGiku6Iu5hl3B0kmk6lIcFPwYjhQIY+tXVRtK9TlGT7RA== + +eslint-plugin-import@^2.25.2: + version "2.26.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.26.0.tgz#f812dc47be4f2b72b478a021605a59fc6fe8b88b" + integrity sha512-hYfi3FXaM8WPLf4S1cikh/r4IxnO6zrhZbEGz2b660EJRbuxgpDS5gkCuYgGWg2xxh2rBuIr4Pvhve/7c31koA== dependencies: - array-includes "^3.1.1" - array.prototype.flat "^1.2.3" - contains-path "^0.1.0" + array-includes "^3.1.4" + array.prototype.flat "^1.2.5" debug "^2.6.9" - doctrine "1.5.0" - eslint-import-resolver-node "^0.3.4" - eslint-module-utils "^2.6.0" + doctrine "^2.1.0" + eslint-import-resolver-node "^0.3.6" + eslint-module-utils "^2.7.3" has "^1.0.3" - minimatch "^3.0.4" - object.values "^1.1.1" - read-pkg-up "^2.0.0" - resolve "^1.17.0" - tsconfig-paths "^3.9.0" + is-core-module "^2.8.1" + is-glob "^4.0.3" + minimatch "^3.1.2" + object.values "^1.1.5" + resolve "^1.22.0" + tsconfig-paths "^3.14.1" + +eslint-plugin-jest@26.7.0: + version "26.7.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-jest/-/eslint-plugin-jest-26.7.0.tgz#41d405ac9143e1284a3401282db47ed459436778" + integrity sha512-/YNitdfG3o3cC6juZziAdkk6nfJt01jXVfj4AgaYVLs7bupHzRDL5K+eipdzhDXtQsiqaX1TzfwSuRlEgeln1A== + dependencies: + "@typescript-eslint/utils" "^5.10.0" + +eslint-plugin-jsx-a11y@^6.6.0: + version "6.6.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-jsx-a11y/-/eslint-plugin-jsx-a11y-6.6.1.tgz#93736fc91b83fdc38cc8d115deedfc3091aef1ff" + integrity sha512-sXgFVNHiWffBq23uiS/JaP6eVR622DqwB4yTzKvGZGcPq6/yZ3WmOZfuBks/vHWo9GaFOqC2ZK4i6+C35knx7Q== + dependencies: + "@babel/runtime" "^7.18.9" + aria-query "^4.2.2" + array-includes "^3.1.5" + ast-types-flow "^0.0.7" + axe-core "^4.4.3" + axobject-query "^2.2.0" + damerau-levenshtein "^1.0.8" + emoji-regex "^9.2.2" + has "^1.0.3" + jsx-ast-utils "^3.3.2" + language-tags "^1.0.5" + minimatch "^3.1.2" + semver "^6.3.0" -eslint-plugin-jest@24.1.3: - version "24.1.3" - resolved "https://registry.yarnpkg.com/eslint-plugin-jest/-/eslint-plugin-jest-24.1.3.tgz#fa3db864f06c5623ff43485ca6c0e8fc5fe8ba0c" - integrity sha512-dNGGjzuEzCE3d5EPZQ/QGtmlMotqnYWD/QpCZ1UuZlrMAdhG5rldh0N0haCvhGnUkSeuORS5VNROwF9Hrgn3Lg== - dependencies: - "@typescript-eslint/experimental-utils" "^4.0.1" +eslint-plugin-no-only-tests@^2.6.0: + version "2.6.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-no-only-tests/-/eslint-plugin-no-only-tests-2.6.0.tgz#19f6c9620bda02b9b9221b436c5f070e42628d76" + integrity sha512-T9SmE/g6UV1uZo1oHAqOvL86XWl7Pl2EpRpnLI8g/bkJu+h7XBCB+1LnubRZ2CUQXj805vh4/CYZdnqtVaEo2Q== -eslint-plugin-prettier@^3.3.1: - version "3.4.0" - resolved "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.4.0.tgz" - integrity sha512-UDK6rJT6INSfcOo545jiaOwB701uAIt2/dR7WnFQoGCVl1/EMqdANBmwUaqqQ45aXprsTGzSa39LI1PyuRBxxw== +eslint-plugin-prettier@4.2.1, eslint-plugin-prettier@^4.0.0: + version "4.2.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-prettier/-/eslint-plugin-prettier-4.2.1.tgz#651cbb88b1dab98bfd42f017a12fa6b2d993f94b" + integrity sha512-f/0rXLXUt0oFYs8ra4w49wYZBG5GKZpAYsJSm6rnYL5uVDjd+zowwMwVZHnAjf4edNrKpCDYfXDgmRE/Ak7QyQ== dependencies: prettier-linter-helpers "^1.0.0" -eslint-plugin-unicorn@28.0.2: - version "28.0.2" - resolved "https://registry.yarnpkg.com/eslint-plugin-unicorn/-/eslint-plugin-unicorn-28.0.2.tgz#ab9884ebae04590ecd9c1c294330d889a74b7c37" - integrity sha512-k4AoFP7n8/oq6lBXkdc9Flid6vw2B8j7aXFCxgzJCyKvmaKrCUFb1TFPhG9eSJQFZowqmymMPRtl8oo9NKLUbw== +eslint-plugin-unicorn@43.0.2: + version "43.0.2" + resolved "https://registry.yarnpkg.com/eslint-plugin-unicorn/-/eslint-plugin-unicorn-43.0.2.tgz#b189d58494c8a0985a4b89dba5dbfde3ad7575a5" + integrity sha512-DtqZ5mf/GMlfWoz1abIjq5jZfaFuHzGBZYIeuJfEoKKGWRHr2JiJR+ea+BF7Wx2N1PPRoT/2fwgiK1NnmNE3Hg== dependencies: - ci-info "^2.0.0" + "@babel/helper-validator-identifier" "^7.18.6" + ci-info "^3.3.2" clean-regexp "^1.0.0" - eslint-template-visitor "^2.2.2" - eslint-utils "^2.1.0" - eslint-visitor-keys "^2.0.0" - import-modules "^2.1.0" - lodash "^4.17.20" + eslint-utils "^3.0.0" + esquery "^1.4.0" + indent-string "^4.0.0" + is-builtin-module "^3.1.0" + lodash "^4.17.21" pluralize "^8.0.0" read-pkg-up "^7.0.1" - regexp-tree "^0.1.22" - reserved-words "^0.1.2" + regexp-tree "^0.1.24" safe-regex "^2.1.1" - semver "^7.3.4" + semver "^7.3.7" + strip-indent "^3.0.0" eslint-rule-documentation@>=1.0.0: version "1.0.23" resolved "https://registry.npmjs.org/eslint-rule-documentation/-/eslint-rule-documentation-1.0.23.tgz" integrity sha512-pWReu3fkohwyvztx/oQWWgld2iad25TfUdi6wvhhaDPIQjHU/pyvlKgXFw1kX31SQK2Nq9MH+vRDWB0ZLy8fYw== -eslint-scope@^5.0.0, eslint-scope@^5.1.0, eslint-scope@^5.1.1: +eslint-scope@^5.1.1: version "5.1.1" - resolved "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.1.tgz#e786e59a66cb92b3f6c1fb0d508aab174848f48c" integrity sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw== dependencies: esrecurse "^4.3.0" estraverse "^4.1.1" -eslint-template-visitor@^2.2.2: - version "2.3.2" - resolved "https://registry.npmjs.org/eslint-template-visitor/-/eslint-template-visitor-2.3.2.tgz" - integrity sha512-3ydhqFpuV7x1M9EK52BPNj6V0Kwu0KKkcIAfpUhwHbR8ocRln/oUHgfxQupY8O1h4Qv/POHDumb/BwwNfxbtnA== +eslint-scope@^7.1.1: + version "7.1.1" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-7.1.1.tgz#fff34894c2f65e5226d3041ac480b4513a163642" + integrity sha512-QKQM/UXpIiHcLqJ5AOyIW7XZmzjkzQXYE54n1++wb0u9V/abW3l9uQnxX8Z5Xd18xyKIMTUAyQ0k1e8pz6LUrw== dependencies: - "@babel/core" "^7.12.16" - "@babel/eslint-parser" "^7.12.16" - eslint-visitor-keys "^2.0.0" - esquery "^1.3.1" - multimap "^1.1.0" + esrecurse "^4.3.0" + estraverse "^5.2.0" -eslint-utils@^2.0.0, eslint-utils@^2.1.0: - version "2.1.0" - resolved "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz" - integrity sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg== +eslint-utils@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/eslint-utils/-/eslint-utils-3.0.0.tgz#8aebaface7345bb33559db0a1f13a1d2d48c3672" + integrity sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA== dependencies: - eslint-visitor-keys "^1.1.0" - -eslint-visitor-keys@^1.1.0, eslint-visitor-keys@^1.3.0: - version "1.3.0" - resolved "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz" - integrity sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ== + eslint-visitor-keys "^2.0.0" eslint-visitor-keys@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.0.0.tgz" integrity sha512-QudtT6av5WXels9WjIM7qz1XD1cWGvX4gGXvp/zBn9nXG02D0utdU3Em2m/QjTnrsk6bBjmCygl3rmj118msQQ== -eslint@7.17.0: - version "7.17.0" - resolved "https://registry.yarnpkg.com/eslint/-/eslint-7.17.0.tgz#4ccda5bf12572ad3bf760e6f195886f50569adb0" - integrity sha512-zJk08MiBgwuGoxes5sSQhOtibZ75pz0J35XTRlZOk9xMffhpA9BTbQZxoXZzOl5zMbleShbGwtw+1kGferfFwQ== +eslint-visitor-keys@^3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-3.3.0.tgz#f6480fa6b1f30efe2d1968aa8ac745b862469826" + integrity sha512-mQ+suqKJVyeuwGYHAdjMFqjCyfl8+Ldnxuyp3ldiMBFKkvytrXUZWaiPCEav8qDHKty44bD+qV1IP4T+w+xXRA== + +eslint@^8.21.0: + version "8.21.0" + resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.21.0.tgz#1940a68d7e0573cef6f50037addee295ff9be9ef" + integrity sha512-/XJ1+Qurf1T9G2M5IHrsjp+xrGT73RZf23xA1z5wB1ZzzEAWSZKvRwhWxTFp1rvkvCfwcvAUNAP31bhKTTGfDA== dependencies: - "@babel/code-frame" "^7.0.0" - "@eslint/eslintrc" "^0.2.2" + "@eslint/eslintrc" "^1.3.0" + "@humanwhocodes/config-array" "^0.10.4" + "@humanwhocodes/gitignore-to-minimatch" "^1.0.2" ajv "^6.10.0" chalk "^4.0.0" cross-spawn "^7.0.2" - debug "^4.0.1" + debug "^4.3.2" doctrine "^3.0.0" - enquirer "^2.3.5" - eslint-scope "^5.1.1" - eslint-utils "^2.1.0" - eslint-visitor-keys "^2.0.0" - espree "^7.3.1" - esquery "^1.2.0" + escape-string-regexp "^4.0.0" + eslint-scope "^7.1.1" + eslint-utils "^3.0.0" + eslint-visitor-keys "^3.3.0" + espree "^9.3.3" + esquery "^1.4.0" esutils "^2.0.2" - file-entry-cache "^6.0.0" + fast-deep-equal "^3.1.3" + file-entry-cache "^6.0.1" + find-up "^5.0.0" functional-red-black-tree "^1.0.1" - glob-parent "^5.0.0" - globals "^12.1.0" - ignore "^4.0.6" + glob-parent "^6.0.1" + globals "^13.15.0" + globby "^11.1.0" + grapheme-splitter "^1.0.4" + ignore "^5.2.0" import-fresh "^3.0.0" imurmurhash "^0.1.4" is-glob "^4.0.0" - js-yaml "^3.13.1" + js-yaml "^4.1.0" json-stable-stringify-without-jsonify "^1.0.1" levn "^0.4.1" - lodash "^4.17.19" - minimatch "^3.0.4" + lodash.merge "^4.6.2" + minimatch "^3.1.2" natural-compare "^1.4.0" optionator "^0.9.1" - progress "^2.0.0" - regexpp "^3.1.0" - semver "^7.2.1" - strip-ansi "^6.0.0" + regexpp "^3.2.0" + strip-ansi "^6.0.1" strip-json-comments "^3.1.0" - table "^6.0.4" text-table "^0.2.0" v8-compile-cache "^2.0.3" -espree@^7.3.0, espree@^7.3.1: - version "7.3.1" - resolved "https://registry.npmjs.org/espree/-/espree-7.3.1.tgz" - integrity sha512-v3JCNCE64umkFpmkFGqzVKsOT0tN1Zr+ueqLZfpV1Ob8e+CEgPWa+OxCoGH3tnhimMKIaBm4m/vaRpJ/krRz2g== +espree@^9.3.2, espree@^9.3.3: + version "9.3.3" + resolved "https://registry.yarnpkg.com/espree/-/espree-9.3.3.tgz#2dd37c4162bb05f433ad3c1a52ddf8a49dc08e9d" + integrity sha512-ORs1Rt/uQTqUKjDdGCyrtYxbazf5umATSf/K4qxjmZHORR6HJk+2s/2Pqe+Kk49HHINC/xNIrGfgh8sZcll0ng== dependencies: - acorn "^7.4.0" - acorn-jsx "^5.3.1" - eslint-visitor-keys "^1.3.0" + acorn "^8.8.0" + acorn-jsx "^5.3.2" + eslint-visitor-keys "^3.3.0" -esprima@^4.0.0, esprima@^4.0.1: +esprima@^4.0.0: version "4.0.1" resolved "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz" integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== -esquery@^1.2.0, esquery@^1.3.1: +esquery@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.4.0.tgz#2148ffc38b82e8c7057dfed48425b3e61f0f24a5" integrity sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w== @@ -2530,26 +2001,6 @@ esutils@^2.0.2: resolved "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz" integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== -events@1.1.1: - version "1.1.1" - resolved "https://registry.npmjs.org/events/-/events-1.1.1.tgz" - integrity sha1-nr23Y1rQmccNzEwqH1AEKI6L2SQ= - -execa@5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/execa/-/execa-5.0.0.tgz#4029b0007998a841fbd1032e5f4de86a3c1e3376" - integrity sha512-ov6w/2LCiuyO4RLYGdpFGjkcs0wMTgGE8PrkTHikeUy5iJekXyPIKUjifk5CsE0pt7sMCrMZ3YNqoCj6idQOnQ== - dependencies: - cross-spawn "^7.0.3" - get-stream "^6.0.0" - human-signals "^2.1.0" - is-stream "^2.0.0" - merge-stream "^2.0.0" - npm-run-path "^4.0.1" - onetime "^5.1.2" - signal-exit "^3.0.3" - strip-final-newline "^2.0.0" - execa@^5.0.0: version "5.1.1" resolved "https://registry.yarnpkg.com/execa/-/execa-5.1.1.tgz#f80ad9cbf4298f7bd1d4c9555c21e93741c411dd" @@ -2570,32 +2021,18 @@ exit@^0.1.2: resolved "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz" integrity sha1-BjJjj42HfMghB9MKD/8aF8uhzQw= -expect@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/expect/-/expect-27.5.1.tgz#83ce59f1e5bdf5f9d2b94b61d2050db48f3fef74" - integrity sha512-E1q5hSUG2AmYQwQJ041nvgpkODHQvB+RKlB4IYdru6uJsyFTRyZAP463M+1lINorwbqAmUggi6+WwkD8lCS/Dw== +expect@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/expect/-/expect-28.1.3.tgz#90a7c1a124f1824133dd4533cce2d2bdcb6603ec" + integrity sha512-eEh0xn8HlsuOBxFgIss+2mX85VAS4Qy3OSkjV7rlBWljtA4oWH37glVGyOZSZvErDT/yBywZdPGwCXuTvSG85g== dependencies: - "@jest/types" "^27.5.1" - jest-get-type "^27.5.1" - jest-matcher-utils "^27.5.1" - jest-message-util "^27.5.1" - -extend@~3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.2.tgz#f8b1136b4071fbd8eb140aff858b1019ec2915fa" - integrity sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g== - -extsprintf@1.3.0: - version "1.3.0" - resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.3.0.tgz#96918440e3041a7a414f8c52e3c574eb3c3e1e05" - integrity sha1-lpGEQOMEGnpBT4xS48V06zw+HgU= - -extsprintf@^1.2.0: - version "1.4.1" - resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.1.tgz#8d172c064867f235c0c84a596806d279bf4bcc07" - integrity sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA== + "@jest/expect-utils" "^28.1.3" + jest-get-type "^28.0.2" + jest-matcher-utils "^28.1.3" + jest-message-util "^28.1.3" + jest-util "^28.1.3" -fast-deep-equal@^3.1.1: +fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: version "3.1.3" resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== @@ -2605,24 +2042,23 @@ fast-diff@^1.1.2: resolved "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz" integrity sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w== -fast-glob@^3.1.1: - version "3.2.5" - resolved "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.5.tgz" - integrity sha512-2DtFcgT68wiTTiwZ2hNdJfcHNke9XOfnwmBRWXhmeKM8rF0TGwmC/Qto3S7RoZKp5cilZbxzO5iTNTQsJ+EeDg== +fast-glob@^3.2.9: + version "3.2.11" + resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.2.11.tgz#a1172ad95ceb8a16e20caa5c5e56480e5129c1d9" + integrity sha512-xrO3+1bxSo3ZVHAnqzyuewYT6aMFHRAd4Kcs92MAonjwQZLsK9d0SF1IyQ3k5PoirxTW0Oe/RqFgMQ6TcNE5Ew== dependencies: "@nodelib/fs.stat" "^2.0.2" "@nodelib/fs.walk" "^1.2.3" - glob-parent "^5.1.0" + glob-parent "^5.1.2" merge2 "^1.3.0" - micromatch "^4.0.2" - picomatch "^2.2.1" + micromatch "^4.0.4" fast-json-stable-stringify@2.x, fast-json-stable-stringify@^2.0.0: version "2.1.0" resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== -fast-levenshtein@^2.0.6, fast-levenshtein@~2.0.6: +fast-levenshtein@^2.0.6: version "2.0.6" resolved "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz" integrity sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc= @@ -2641,7 +2077,7 @@ fb-watchman@^2.0.0: dependencies: bser "2.1.1" -file-entry-cache@^6.0.0: +file-entry-cache@^6.0.1: version "6.0.1" resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-6.0.1.tgz#211b2dd9659cb0394b073e7323ac3c933d522027" integrity sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg== @@ -2655,7 +2091,7 @@ fill-range@^7.0.1: dependencies: to-regex-range "^5.0.1" -find-up@^2.0.0, find-up@^2.1.0: +find-up@^2.1.0: version "2.1.0" resolved "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz" integrity sha1-RdG35QbHF93UgndaK3eSCjwMV6c= @@ -2670,6 +2106,14 @@ find-up@^4.0.0, find-up@^4.1.0: locate-path "^5.0.0" path-exists "^4.0.0" +find-up@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" + integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng== + dependencies: + locate-path "^6.0.0" + path-exists "^4.0.0" + flat-cache@^3.0.4: version "3.0.4" resolved "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz" @@ -2683,45 +2127,6 @@ flatted@^3.1.0: resolved "https://registry.npmjs.org/flatted/-/flatted-3.1.1.tgz" integrity sha512-zAoAQiudy+r5SvnSw3KJy5os/oRJYHzrzja/tBDqrZtNhUw8bt6y8OBzMWcjWr+8liV8Eb6yOhw8WZ7VFZ5ZzA== -forever-agent@~0.6.1: - version "0.6.1" - resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91" - integrity sha1-+8cfDEGt6zf5bFd60e1C2P2sypE= - -form-data@^2.5.0: - version "2.5.1" - resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.5.1.tgz#f2cbec57b5e59e23716e128fe44d4e5dd23895f4" - integrity sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA== - dependencies: - asynckit "^0.4.0" - combined-stream "^1.0.6" - mime-types "^2.1.12" - -form-data@^3.0.0: - version "3.0.1" - resolved "https://registry.yarnpkg.com/form-data/-/form-data-3.0.1.tgz#ebd53791b78356a99af9a300d4282c4d5eb9755f" - integrity sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg== - dependencies: - asynckit "^0.4.0" - combined-stream "^1.0.8" - mime-types "^2.1.12" - -form-data@~2.3.2: - version "2.3.3" - resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.3.3.tgz#dcce52c05f644f298c6a7ab936bd724ceffbf3a6" - integrity sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ== - dependencies: - asynckit "^0.4.0" - combined-stream "^1.0.6" - mime-types "^2.1.12" - -fs-minipass@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-2.1.0.tgz#7f5036fdbf12c63c169190cbe4199c852271f9fb" - integrity sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg== - dependencies: - minipass "^3.0.0" - fs.realpath@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" @@ -2737,11 +2142,26 @@ function-bind@^1.1.1: resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d" integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A== +function.prototype.name@^1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/function.prototype.name/-/function.prototype.name-1.1.5.tgz#cce0505fe1ffb80503e6f9e46cc64e46a12a9621" + integrity sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.3" + es-abstract "^1.19.0" + functions-have-names "^1.2.2" + functional-red-black-tree@^1.0.1: version "1.0.1" resolved "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz" integrity sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc= +functions-have-names@^1.2.2: + version "1.2.3" + resolved "https://registry.yarnpkg.com/functions-have-names/-/functions-have-names-1.2.3.tgz#0404fe4ee2ba2f607f0e0ec3c80bae994133b834" + integrity sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ== + gensync@^1.0.0-beta.2: version "1.0.0-beta.2" resolved "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz" @@ -2761,45 +2181,48 @@ get-intrinsic@^1.0.2, get-intrinsic@^1.1.1: has "^1.0.3" has-symbols "^1.0.1" +get-intrinsic@^1.1.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.1.2.tgz#336975123e05ad0b7ba41f152ee4aadbea6cf598" + integrity sha512-Jfm3OyCxHh9DJyc28qGk+JmfkpO41A4XkneDSujN9MDXrm4oDKdHvndhZ2dN94+ERNfkYJWDclW6k2L/ZGHjXA== + dependencies: + function-bind "^1.1.1" + has "^1.0.3" + has-symbols "^1.0.3" + get-package-type@^0.1.0: version "0.1.0" resolved "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz" integrity sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q== -get-stream@^4.1.0: - version "4.1.0" - resolved "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz" - integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w== - dependencies: - pump "^3.0.0" - -get-stream@^5.1.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3" - integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA== - dependencies: - pump "^3.0.0" - get-stream@^6.0.0: version "6.0.1" resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-6.0.1.tgz#a262d8eef67aced57c2852ad6167526a43cbf7b7" integrity sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg== -getpass@^0.1.1: - version "0.1.7" - resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa" - integrity sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo= +get-symbol-description@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.0.0.tgz#7fdb81c900101fbd564dd5f1a30af5aadc1e58d6" + integrity sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw== dependencies: - assert-plus "^1.0.0" + call-bind "^1.0.2" + get-intrinsic "^1.1.1" -glob-parent@^5.0.0, glob-parent@^5.1.0: +glob-parent@^5.1.2: version "5.1.2" resolved "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz" integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== dependencies: is-glob "^4.0.1" -glob@^7.0.0, glob@^7.1.1, glob@^7.1.2, glob@^7.1.3, glob@^7.1.4: +glob-parent@^6.0.1: + version "6.0.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-6.0.2.tgz#6d237d99083950c79290f24c7642a3de9a28f9e3" + integrity sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A== + dependencies: + is-glob "^4.0.3" + +glob@^7.1.3, glob@^7.1.4: version "7.2.0" resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.0.tgz#d15535af7732e02e948f4c41628bd910293f6023" integrity sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q== @@ -2816,86 +2239,39 @@ globals@^11.1.0: resolved "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz" integrity sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA== -globals@^12.1.0: - version "12.4.0" - resolved "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz" - integrity sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg== +globals@^13.15.0: + version "13.17.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-13.17.0.tgz#902eb1e680a41da93945adbdcb5a9f361ba69bd4" + integrity sha512-1C+6nQRb1GwGMKm2dH/E7enFAMxGTmGI7/dEdhy/DNelv85w9B72t3uc5frtMNXIbzrarJJ/lTCjcaZwbLJmyw== dependencies: - type-fest "^0.8.1" + type-fest "^0.20.2" -globby@^11.0.1: - version "11.0.3" - resolved "https://registry.npmjs.org/globby/-/globby-11.0.3.tgz" - integrity sha512-ffdmosjA807y7+lA1NM0jELARVmYul/715xiILEjo3hBLPTcirgQNnXECn5g3mtR8TOLCVbkfua1Hpen25/Xcg== +globby@^11.1.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/globby/-/globby-11.1.0.tgz#bd4be98bb042f83d796f7e3811991fbe82a0d34b" + integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g== dependencies: array-union "^2.1.0" dir-glob "^3.0.1" - fast-glob "^3.1.1" - ignore "^5.1.4" - merge2 "^1.3.0" + fast-glob "^3.2.9" + ignore "^5.2.0" + merge2 "^1.4.1" slash "^3.0.0" -got@^11.8.0: - version "11.8.3" - resolved "https://registry.yarnpkg.com/got/-/got-11.8.3.tgz#f496c8fdda5d729a90b4905d2b07dbd148170770" - integrity sha512-7gtQ5KiPh1RtGS9/Jbv1ofDpBFuq42gyfEib+ejaRBJuj/3tQFeR5+gw57e4ipaU8c/rCjvX6fkQz2lyDlGAOg== - dependencies: - "@sindresorhus/is" "^4.0.0" - "@szmarczak/http-timer" "^4.0.5" - "@types/cacheable-request" "^6.0.1" - "@types/responselike" "^1.0.0" - cacheable-lookup "^5.0.3" - cacheable-request "^7.0.2" - decompress-response "^6.0.0" - http2-wrapper "^1.0.0-beta.5.2" - lowercase-keys "^2.0.0" - p-cancelable "^2.0.0" - responselike "^2.0.0" - -got@^9.6.0: - version "9.6.0" - resolved "https://registry.npmjs.org/got/-/got-9.6.0.tgz" - integrity sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q== - dependencies: - "@sindresorhus/is" "^0.14.0" - "@szmarczak/http-timer" "^1.1.2" - cacheable-request "^6.0.0" - decompress-response "^3.3.0" - duplexer3 "^0.1.4" - get-stream "^4.1.0" - lowercase-keys "^1.0.1" - mimic-response "^1.0.1" - p-cancelable "^1.0.0" - to-readable-stream "^1.0.0" - url-parse-lax "^3.0.0" - -graceful-fs@^4.1.2: - version "4.2.6" - resolved "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.6.tgz" - integrity sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ== - graceful-fs@^4.2.9: version "4.2.9" resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.9.tgz#041b05df45755e587a24942279b9d113146e1c96" integrity sha512-NtNxqUcXgpW2iMrfqSfR73Glt39K+BLwWsPs94yR63v45T0Wbej7eRmL5cWfwEgqXnmjQp3zaJTshdRW/qC2ZQ== -har-schema@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-2.0.0.tgz#a94c2224ebcac04782a0d9035521f24735b7ec92" - integrity sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI= - -har-validator@~5.1.3: - version "5.1.5" - resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-5.1.5.tgz#1f0803b9f8cb20c0fa13822df1ecddb36bde1efd" - integrity sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w== - dependencies: - ajv "^6.12.3" - har-schema "^2.0.0" +grapheme-splitter@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/grapheme-splitter/-/grapheme-splitter-1.0.4.tgz#9cf3a665c6247479896834af35cf1dbb4400767e" + integrity sha512-bzh50DW9kTPM00T8y4o8vQg89Di9oLJVLW/KaOGIXJWP/iqCN6WKYkbNOF04vFLJhwcpYUh9ydh/+5vpOqV4YQ== -has-bigints@^1.0.1: - version "1.0.1" - resolved "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.1.tgz" - integrity sha512-LSBS2LjbNBTf6287JEbEzvJgftkF5qFkmCo9hDRpAzKhUOlJ+hx8dd4USs00SgsUNwc4617J9ki5YtEClM2ffA== +has-bigints@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-bigints/-/has-bigints-1.0.2.tgz#0871bd3e3d51626f6ca0966668ba35d5602d6eaa" + integrity sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ== has-flag@^3.0.0: version "3.0.0" @@ -2907,11 +2283,30 @@ has-flag@^4.0.0: resolved "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz" integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== +has-property-descriptors@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz#610708600606d36961ed04c196193b6a607fa861" + integrity sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ== + dependencies: + get-intrinsic "^1.1.1" + has-symbols@^1.0.1, has-symbols@^1.0.2: version "1.0.2" resolved "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz" integrity sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw== +has-symbols@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8" + integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A== + +has-tostringtag@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.0.tgz#7e133818a7d394734f941e73c3d3f9291e658b25" + integrity sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ== + dependencies: + has-symbols "^1.0.2" + has@^1.0.3: version "1.0.3" resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796" @@ -2924,89 +2319,26 @@ hosted-git-info@^2.1.4: resolved "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz" integrity sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw== -html-encoding-sniffer@^2.0.1: - version "2.0.1" - resolved "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz" - integrity sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ== - dependencies: - whatwg-encoding "^1.0.5" - html-escaper@^2.0.0: version "2.0.2" resolved "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz" integrity sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg== -http-cache-semantics@^4.0.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz#49e91c5cbf36c9b94bcfcd71c23d5249ec74e390" - integrity sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ== - -http-proxy-agent@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz#8a8c8ef7f5932ccf953c296ca8291b95aa74aa3a" - integrity sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg== - dependencies: - "@tootallnate/once" "1" - agent-base "6" - debug "4" - -http-signature@~1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.2.0.tgz#9aecd925114772f3d95b65a60abb8f7c18fbace1" - integrity sha1-muzZJRFHcvPZW2WmCruPfBj7rOE= - dependencies: - assert-plus "^1.0.0" - jsprim "^1.2.2" - sshpk "^1.7.0" - -http2-wrapper@^1.0.0-beta.5.2: - version "1.0.3" - resolved "https://registry.yarnpkg.com/http2-wrapper/-/http2-wrapper-1.0.3.tgz#b8f55e0c1f25d4ebd08b3b0c2c079f9590800b3d" - integrity sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg== - dependencies: - quick-lru "^5.1.1" - resolve-alpn "^1.0.0" - -https-proxy-agent@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz#e2a90542abb68a762e0a0850f6c9edadfd8506b2" - integrity sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA== - dependencies: - agent-base "6" - debug "4" - human-signals@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-2.1.0.tgz#dc91fcba42e4d06e4abaed33b3e7a3c02f514ea0" integrity sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw== -iconv-lite@0.4.24: - version "0.4.24" - resolved "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz" - integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA== - dependencies: - safer-buffer ">= 2.1.2 < 3" - -ieee754@1.1.13: - version "1.1.13" - resolved "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz" - integrity sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg== - -ieee754@^1.1.4: - version "1.2.1" - resolved "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz" - integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== - -ignore@^4.0.6: - version "4.0.6" - resolved "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz" - integrity sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg== - -ignore@^5.0.5, ignore@^5.1.4: +ignore@^5.0.5: version "5.1.8" resolved "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz" integrity sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw== +ignore@^5.2.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.0.tgz#6d3bac8fa7fe0d45d9f9be7bac2fc279577e345a" + integrity sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ== + import-fresh@^3.0.0, import-fresh@^3.2.1: version "3.3.0" resolved "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz" @@ -3023,11 +2355,6 @@ import-local@^3.0.2: pkg-dir "^4.2.0" resolve-cwd "^3.0.0" -import-modules@^2.1.0: - version "2.1.0" - resolved "https://registry.npmjs.org/import-modules/-/import-modules-2.1.0.tgz" - integrity sha512-8HEWcnkbGpovH9yInoisxaSoIg9Brbul+Ju3Kqe2UsYDUBJD/iQjSgEj0zPcTDPKfPp2fs5xlv1i+JSye/m1/A== - imurmurhash@^0.1.4: version "0.1.4" resolved "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz" @@ -3051,10 +2378,14 @@ inherits@2: resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== -interpret@^1.0.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/interpret/-/interpret-1.4.0.tgz#665ab8bc4da27a774a40584e812e3e0fa45b1a1e" - integrity sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA== +internal-slot@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.0.3.tgz#7347e307deeea2faac2ac6205d4bc7d34967f59c" + integrity sha512-O0DB1JC/sPyZl7cIo78n5dR7eUSwwpYPiXRhTzNxZVAMUuB8vlnRFyLxdrVToks6XPLVnFfbzaVd5WLjhgg+vA== + dependencies: + get-intrinsic "^1.1.0" + has "^1.0.3" + side-channel "^1.0.4" is-arrayish@^0.2.1: version "0.2.1" @@ -3073,11 +2404,23 @@ is-boolean-object@^1.1.0: dependencies: call-bind "^1.0.0" -is-callable@^1.1.4, is-callable@^1.2.3: +is-builtin-module@^3.1.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/is-builtin-module/-/is-builtin-module-3.2.0.tgz#bb0310dfe881f144ca83f30100ceb10cf58835e0" + integrity sha512-phDA4oSGt7vl1n5tJvTWooWWAsXLY+2xCnxNqvKhGEzujg+A43wPlPOyDg3C8XQHN+6k/JTQWJ/j0dQh/qr+Hw== + dependencies: + builtin-modules "^3.3.0" + +is-callable@^1.1.4: version "1.2.3" resolved "https://registry.npmjs.org/is-callable/-/is-callable-1.2.3.tgz" integrity sha512-J1DcMe8UYTBSrKezuIUTUwjXsho29693unXM2YhJUTR2txK/eG47bvNa/wipPFmZFgr/N6f1GA66dv0mEyTIyQ== +is-callable@^1.2.4: + version "1.2.4" + resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.4.tgz#47301d58dd0259407865547853df6d61fe471945" + integrity sha512-nsuwtxZfMX67Oryl9LCQ+upnC0Z0BgpwntpS89m1H/TLF0zNfzfLMV/9Wa/6MZsj0acpEjAO0KF1xT6ZdLl95w== + is-core-module@^2.2.0: version "2.2.0" resolved "https://registry.npmjs.org/is-core-module/-/is-core-module-2.2.0.tgz" @@ -3092,6 +2435,13 @@ is-core-module@^2.8.1: dependencies: has "^1.0.3" +is-core-module@^2.9.0: + version "2.10.0" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.10.0.tgz#9012ede0a91c69587e647514e1d5277019e728ed" + integrity sha512-Erxj2n/LDAZ7H8WNJXd9tw38GYM3dv8rk8Zcs+jJuxYTW7sozH+SS8NtrSjVL1/vpLvWi1hxy96IzjJ3EHTJJg== + dependencies: + has "^1.0.3" + is-date-object@^1.0.1: version "1.0.2" resolved "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz" @@ -3119,10 +2469,17 @@ is-glob@^4.0.0, is-glob@^4.0.1: dependencies: is-extglob "^2.1.1" -is-negative-zero@^2.0.1: - version "2.0.1" - resolved "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.1.tgz" - integrity sha512-2z6JzQvZRa9A2Y7xC6dQQm4FSTSTNWjKIYYTt4246eMTJmIo0Q+ZyOsU66X8lxK1AbB92dFeglPLrhwpeRKO6w== +is-glob@^4.0.3: + version "4.0.3" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" + integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== + dependencies: + is-extglob "^2.1.1" + +is-negative-zero@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/is-negative-zero/-/is-negative-zero-2.0.2.tgz#7bf6f03a28003b8b3965de3ac26f664d765f3150" + integrity sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA== is-number-object@^1.0.4: version "1.0.4" @@ -3134,28 +2491,20 @@ is-number@^7.0.0: resolved "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz" integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== -is-plain-object@^3.0.0: - version "3.0.1" - resolved "https://registry.npmjs.org/is-plain-object/-/is-plain-object-3.0.1.tgz" - integrity sha512-Xnpx182SBMrr/aBik8y+GuR4U1L9FqMSojwDQwPMmxyC6bvEqly9UBCxhauBF5vNh2gwWJNX6oDV7O+OM4z34g== - -is-plain-object@^5.0.0: - version "5.0.0" - resolved "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz" - integrity sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q== - -is-potential-custom-element-name@^1.0.0, is-potential-custom-element-name@^1.0.1: - version "1.0.1" - resolved "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz" - integrity sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ== +is-regex@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.4.tgz#eef5663cd59fa4c0ae339505323df6854bb15958" + integrity sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg== + dependencies: + call-bind "^1.0.2" + has-tostringtag "^1.0.0" -is-regex@^1.1.2: - version "1.1.2" - resolved "https://registry.npmjs.org/is-regex/-/is-regex-1.1.2.tgz" - integrity sha512-axvdhb5pdhEVThqJzYXwMlVuZwC+FF2DpcOhTS+y/8jVq4trxyPgfcwIxIKiyeuLlSQYKkmUaPQJ8ZE4yNKXDg== +is-shared-array-buffer@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz#8f259c573b60b6a32d4058a1a07430c0a7344c79" + integrity sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA== dependencies: call-bind "^1.0.2" - has-symbols "^1.0.1" is-stream@^2.0.0: version "2.0.1" @@ -3167,6 +2516,13 @@ is-string@^1.0.5: resolved "https://registry.npmjs.org/is-string/-/is-string-1.0.5.tgz" integrity sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ== +is-string@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.0.7.tgz#0dd12bf2006f255bb58f695110eff7491eebc0fd" + integrity sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg== + dependencies: + has-tostringtag "^1.0.0" + is-symbol@^1.0.2, is-symbol@^1.0.3: version "1.0.3" resolved "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz" @@ -3174,31 +2530,18 @@ is-symbol@^1.0.2, is-symbol@^1.0.3: dependencies: has-symbols "^1.0.1" -is-typedarray@^1.0.0, is-typedarray@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" - integrity sha1-5HnICFjfDBsR3dppQPlgEfzaSpo= - -isarray@^1.0.0: - version "1.0.0" - resolved "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz" - integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= +is-weakref@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-weakref/-/is-weakref-1.0.2.tgz#9529f383a9338205e89765e0392efc2f100f06f2" + integrity sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ== + dependencies: + call-bind "^1.0.2" isexe@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA= -isomorphic-ws@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/isomorphic-ws/-/isomorphic-ws-4.0.1.tgz#55fd4cd6c5e6491e76dc125938dd863f5cd4f2dc" - integrity sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w== - -isstream@~0.1.2: - version "0.1.2" - resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a" - integrity sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo= - istanbul-lib-coverage@^3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz" @@ -3246,235 +2589,195 @@ istanbul-reports@^3.1.3: html-escaper "^2.0.0" istanbul-lib-report "^3.0.0" -jest-changed-files@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-changed-files/-/jest-changed-files-27.5.1.tgz#a348aed00ec9bf671cc58a66fcbe7c3dfd6a68f5" - integrity sha512-buBLMiByfWGCoMsLLzGUUSpAmIAGnbR2KJoMN10ziLhOLvP4e0SlypHnAel8iqQXTrcbmfEY9sSqae5sgUsTvw== +jest-changed-files@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-changed-files/-/jest-changed-files-28.1.3.tgz#d9aeee6792be3686c47cb988a8eaf82ff4238831" + integrity sha512-esaOfUWJXk2nfZt9SPyC8gA1kNfdKLkQWyzsMlqq8msYSlNKfmZxfRgZn4Cd4MGVUF+7v6dBs0d5TOAKa7iIiA== dependencies: - "@jest/types" "^27.5.1" execa "^5.0.0" - throat "^6.0.1" + p-limit "^3.1.0" -jest-circus@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-circus/-/jest-circus-27.5.1.tgz#37a5a4459b7bf4406e53d637b49d22c65d125ecc" - integrity sha512-D95R7x5UtlMA5iBYsOHFFbMD/GVA4R/Kdq15f7xYWUfWHBto9NYRsOvnSauTgdF+ogCpJ4tyKOXhUifxS65gdw== +jest-circus@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-circus/-/jest-circus-28.1.3.tgz#d14bd11cf8ee1a03d69902dc47b6bd4634ee00e4" + integrity sha512-cZ+eS5zc79MBwt+IhQhiEp0OeBddpc1n8MBo1nMB8A7oPMKEO+Sre+wHaLJexQUj9Ya/8NOBY0RESUgYjB6fow== dependencies: - "@jest/environment" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/environment" "^28.1.3" + "@jest/expect" "^28.1.3" + "@jest/test-result" "^28.1.3" + "@jest/types" "^28.1.3" "@types/node" "*" chalk "^4.0.0" co "^4.6.0" dedent "^0.7.0" - expect "^27.5.1" is-generator-fn "^2.0.0" - jest-each "^27.5.1" - jest-matcher-utils "^27.5.1" - jest-message-util "^27.5.1" - jest-runtime "^27.5.1" - jest-snapshot "^27.5.1" - jest-util "^27.5.1" - pretty-format "^27.5.1" + jest-each "^28.1.3" + jest-matcher-utils "^28.1.3" + jest-message-util "^28.1.3" + jest-runtime "^28.1.3" + jest-snapshot "^28.1.3" + jest-util "^28.1.3" + p-limit "^3.1.0" + pretty-format "^28.1.3" slash "^3.0.0" stack-utils "^2.0.3" - throat "^6.0.1" -jest-cli@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-cli/-/jest-cli-27.5.1.tgz#278794a6e6458ea8029547e6c6cbf673bd30b145" - integrity sha512-Hc6HOOwYq4/74/c62dEE3r5elx8wjYqxY0r0G/nFrLDPMFRu6RA/u8qINOIkvhxG7mMQ5EJsOGfRpI8L6eFUVw== +jest-cli@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-cli/-/jest-cli-28.1.3.tgz#558b33c577d06de55087b8448d373b9f654e46b2" + integrity sha512-roY3kvrv57Azn1yPgdTebPAXvdR2xfezaKKYzVxZ6It/5NCxzJym6tUI5P1zkdWhfUYkxEI9uZWcQdaFLo8mJQ== dependencies: - "@jest/core" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/core" "^28.1.3" + "@jest/test-result" "^28.1.3" + "@jest/types" "^28.1.3" chalk "^4.0.0" exit "^0.1.2" graceful-fs "^4.2.9" import-local "^3.0.2" - jest-config "^27.5.1" - jest-util "^27.5.1" - jest-validate "^27.5.1" + jest-config "^28.1.3" + jest-util "^28.1.3" + jest-validate "^28.1.3" prompts "^2.0.1" - yargs "^16.2.0" + yargs "^17.3.1" -jest-config@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-config/-/jest-config-27.5.1.tgz#5c387de33dca3f99ad6357ddeccd91bf3a0e4a41" - integrity sha512-5sAsjm6tGdsVbW9ahcChPAFCk4IlkQUknH5AvKjuLTSlcO/wCZKyFdn7Rg0EkC+OGgWODEy2hDpWB1PgzH0JNA== +jest-config@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-config/-/jest-config-28.1.3.tgz#e315e1f73df3cac31447eed8b8740a477392ec60" + integrity sha512-MG3INjByJ0J4AsNBm7T3hsuxKQqFIiRo/AUqb1q9LRKI5UU6Aar9JHbr9Ivn1TVwfUD9KirRoM/T6u8XlcQPHQ== dependencies: - "@babel/core" "^7.8.0" - "@jest/test-sequencer" "^27.5.1" - "@jest/types" "^27.5.1" - babel-jest "^27.5.1" + "@babel/core" "^7.11.6" + "@jest/test-sequencer" "^28.1.3" + "@jest/types" "^28.1.3" + babel-jest "^28.1.3" chalk "^4.0.0" ci-info "^3.2.0" deepmerge "^4.2.2" - glob "^7.1.1" + glob "^7.1.3" graceful-fs "^4.2.9" - jest-circus "^27.5.1" - jest-environment-jsdom "^27.5.1" - jest-environment-node "^27.5.1" - jest-get-type "^27.5.1" - jest-jasmine2 "^27.5.1" - jest-regex-util "^27.5.1" - jest-resolve "^27.5.1" - jest-runner "^27.5.1" - jest-util "^27.5.1" - jest-validate "^27.5.1" + jest-circus "^28.1.3" + jest-environment-node "^28.1.3" + jest-get-type "^28.0.2" + jest-regex-util "^28.0.2" + jest-resolve "^28.1.3" + jest-runner "^28.1.3" + jest-util "^28.1.3" + jest-validate "^28.1.3" micromatch "^4.0.4" parse-json "^5.2.0" - pretty-format "^27.5.1" + pretty-format "^28.1.3" slash "^3.0.0" strip-json-comments "^3.1.1" -jest-diff@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-27.5.1.tgz#a07f5011ac9e6643cf8a95a462b7b1ecf6680def" - integrity sha512-m0NvkX55LDt9T4mctTEgnZk3fmEg3NRYutvMPWM/0iPnkFj2wIeF45O1718cMSOFO1vINkqmxqD8vE37uTEbqw== +jest-diff@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-28.1.3.tgz#948a192d86f4e7a64c5264ad4da4877133d8792f" + integrity sha512-8RqP1B/OXzjjTWkqMX67iqgwBVJRgCyKD3L9nq+6ZqJMdvjE8RgHktqZ6jNrkdMT+dJuYNI3rhQpxaz7drJHfw== dependencies: chalk "^4.0.0" - diff-sequences "^27.5.1" - jest-get-type "^27.5.1" - pretty-format "^27.5.1" + diff-sequences "^28.1.1" + jest-get-type "^28.0.2" + pretty-format "^28.1.3" -jest-docblock@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-docblock/-/jest-docblock-27.5.1.tgz#14092f364a42c6108d42c33c8cf30e058e25f6c0" - integrity sha512-rl7hlABeTsRYxKiUfpHrQrG4e2obOiTQWfMEH3PxPjOtdsfLQO4ReWSZaQ7DETm4xu07rl4q/h4zcKXyU0/OzQ== +jest-docblock@^28.1.1: + version "28.1.1" + resolved "https://registry.yarnpkg.com/jest-docblock/-/jest-docblock-28.1.1.tgz#6f515c3bf841516d82ecd57a62eed9204c2f42a8" + integrity sha512-3wayBVNiOYx0cwAbl9rwm5kKFP8yHH3d/fkEaL02NPTkDojPtheGB7HZSFY4wzX+DxyrvhXz0KSCVksmCknCuA== dependencies: detect-newline "^3.0.0" -jest-each@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-each/-/jest-each-27.5.1.tgz#5bc87016f45ed9507fed6e4702a5b468a5b2c44e" - integrity sha512-1Ff6p+FbhT/bXQnEouYy00bkNSY7OUpfIcmdl8vZ31A1UUaurOLPA8a8BbJOF2RDUElwJhmeaV7LnagI+5UwNQ== +jest-each@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-each/-/jest-each-28.1.3.tgz#bdd1516edbe2b1f3569cfdad9acd543040028f81" + integrity sha512-arT1z4sg2yABU5uogObVPvSlSMQlDA48owx07BDPAiasW0yYpYHYOo4HHLz9q0BVzDVU4hILFjzJw0So9aCL/g== dependencies: - "@jest/types" "^27.5.1" + "@jest/types" "^28.1.3" chalk "^4.0.0" - jest-get-type "^27.5.1" - jest-util "^27.5.1" - pretty-format "^27.5.1" - -jest-environment-jsdom@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-environment-jsdom/-/jest-environment-jsdom-27.5.1.tgz#ea9ccd1fc610209655a77898f86b2b559516a546" - integrity sha512-TFBvkTC1Hnnnrka/fUb56atfDtJ9VMZ94JkjTbggl1PEpwrYtUBKMezB3inLmWqQsXYLcMwNoDQwoBTAvFfsfw== - dependencies: - "@jest/environment" "^27.5.1" - "@jest/fake-timers" "^27.5.1" - "@jest/types" "^27.5.1" + jest-get-type "^28.0.2" + jest-util "^28.1.3" + pretty-format "^28.1.3" + +jest-environment-node@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-environment-node/-/jest-environment-node-28.1.3.tgz#7e74fe40eb645b9d56c0c4b70ca4357faa349be5" + integrity sha512-ugP6XOhEpjAEhGYvp5Xj989ns5cB1K6ZdjBYuS30umT4CQEETaxSiPcZ/E1kFktX4GkrcM4qu07IIlDYX1gp+A== + dependencies: + "@jest/environment" "^28.1.3" + "@jest/fake-timers" "^28.1.3" + "@jest/types" "^28.1.3" "@types/node" "*" - jest-mock "^27.5.1" - jest-util "^27.5.1" - jsdom "^16.6.0" - -jest-environment-node@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-environment-node/-/jest-environment-node-27.5.1.tgz#dedc2cfe52fab6b8f5714b4808aefa85357a365e" - integrity sha512-Jt4ZUnxdOsTGwSRAfKEnE6BcwsSPNOijjwifq5sDFSA2kesnXTvNqKHYgM0hDq3549Uf/KzdXNYn4wMZJPlFLw== - dependencies: - "@jest/environment" "^27.5.1" - "@jest/fake-timers" "^27.5.1" - "@jest/types" "^27.5.1" - "@types/node" "*" - jest-mock "^27.5.1" - jest-util "^27.5.1" + jest-mock "^28.1.3" + jest-util "^28.1.3" -jest-fail-on-console@^2.3.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/jest-fail-on-console/-/jest-fail-on-console-2.3.0.tgz#bf22af0e535512ec7e220223279e3fb7ae621b8b" - integrity sha512-RwTq79+v5OccU59VmhoTvsCyTQ11YeOtLL6xLtkks/H7afLay6gASk6a4qMlyLlpM36t0rPW+DQsJ4ZNcCS2ug== +jest-fail-on-console@^2.4.2: + version "2.4.2" + resolved "https://registry.yarnpkg.com/jest-fail-on-console/-/jest-fail-on-console-2.4.2.tgz#cfe790cc592e46119a6842f127039d01f046db87" + integrity sha512-CdulWZvfI+cz4+dXQr6p0BhhexFjLnIIBR/7YcpzPXFxrNozAruWkEjR1RU89cd7WXYwckX5ygvHuHQa3NjbOQ== dependencies: chalk "^4.1.0" -jest-get-type@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-27.5.1.tgz#3cd613c507b0f7ace013df407a1c1cd578bcb4f1" - integrity sha512-2KY95ksYSaK7DMBWQn6dQz3kqAf3BB64y2udeG+hv4KfSOb9qwcYQstTJc1KCbsix+wLZWZYN8t7nwX3GOBLRw== +jest-get-type@^28.0.2: + version "28.0.2" + resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-28.0.2.tgz#34622e628e4fdcd793d46db8a242227901fcf203" + integrity sha512-ioj2w9/DxSYHfOm5lJKCdcAmPJzQXmbM/Url3rhlghrPvT3tt+7a/+oXc9azkKmLvoiXjtV83bEWqi+vs5nlPA== -jest-haste-map@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-haste-map/-/jest-haste-map-27.5.1.tgz#9fd8bd7e7b4fa502d9c6164c5640512b4e811e7f" - integrity sha512-7GgkZ4Fw4NFbMSDSpZwXeBiIbx+t/46nJ2QitkOjvwPYyZmqttu2TDSimMHP1EkPOi4xUZAN1doE5Vd25H4Jng== +jest-haste-map@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-haste-map/-/jest-haste-map-28.1.3.tgz#abd5451129a38d9841049644f34b034308944e2b" + integrity sha512-3S+RQWDXccXDKSWnkHa/dPwt+2qwA8CJzR61w3FoYCvoo3Pn8tvGcysmMF0Bj0EX5RYvAI2EIvC57OmotfdtKA== dependencies: - "@jest/types" "^27.5.1" - "@types/graceful-fs" "^4.1.2" + "@jest/types" "^28.1.3" + "@types/graceful-fs" "^4.1.3" "@types/node" "*" anymatch "^3.0.3" fb-watchman "^2.0.0" graceful-fs "^4.2.9" - jest-regex-util "^27.5.1" - jest-serializer "^27.5.1" - jest-util "^27.5.1" - jest-worker "^27.5.1" + jest-regex-util "^28.0.2" + jest-util "^28.1.3" + jest-worker "^28.1.3" micromatch "^4.0.4" - walker "^1.0.7" + walker "^1.0.8" optionalDependencies: fsevents "^2.3.2" -jest-jasmine2@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-jasmine2/-/jest-jasmine2-27.5.1.tgz#a037b0034ef49a9f3d71c4375a796f3b230d1ac4" - integrity sha512-jtq7VVyG8SqAorDpApwiJJImd0V2wv1xzdheGHRGyuT7gZm6gG47QEskOlzsN1PG/6WNaCo5pmwMHDf3AkG2pQ== +jest-leak-detector@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-leak-detector/-/jest-leak-detector-28.1.3.tgz#a6685d9b074be99e3adee816ce84fd30795e654d" + integrity sha512-WFVJhnQsiKtDEo5lG2mM0v40QWnBM+zMdHHyJs8AWZ7J0QZJS59MsyKeJHWhpBZBH32S48FOVvGyOFT1h0DlqA== dependencies: - "@jest/environment" "^27.5.1" - "@jest/source-map" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/types" "^27.5.1" - "@types/node" "*" - chalk "^4.0.0" - co "^4.6.0" - expect "^27.5.1" - is-generator-fn "^2.0.0" - jest-each "^27.5.1" - jest-matcher-utils "^27.5.1" - jest-message-util "^27.5.1" - jest-runtime "^27.5.1" - jest-snapshot "^27.5.1" - jest-util "^27.5.1" - pretty-format "^27.5.1" - throat "^6.0.1" - -jest-leak-detector@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-leak-detector/-/jest-leak-detector-27.5.1.tgz#6ec9d54c3579dd6e3e66d70e3498adf80fde3fb8" - integrity sha512-POXfWAMvfU6WMUXftV4HolnJfnPOGEu10fscNCA76KBpRRhcMN2c8d3iT2pxQS3HLbA+5X4sOUPzYO2NUyIlHQ== - dependencies: - jest-get-type "^27.5.1" - pretty-format "^27.5.1" - -jest-matcher-utils@^27.0.0, jest-matcher-utils@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-matcher-utils/-/jest-matcher-utils-27.5.1.tgz#9c0cdbda8245bc22d2331729d1091308b40cf8ab" - integrity sha512-z2uTx/T6LBaCoNWNFWwChLBKYxTMcGBRjAt+2SbP929/Fflb9aa5LGma654Rz8z9HLxsrUaYzxE9T/EFIL/PAw== + jest-get-type "^28.0.2" + pretty-format "^28.1.3" + +jest-matcher-utils@^28.0.0, jest-matcher-utils@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-matcher-utils/-/jest-matcher-utils-28.1.3.tgz#5a77f1c129dd5ba3b4d7fc20728806c78893146e" + integrity sha512-kQeJ7qHemKfbzKoGjHHrRKH6atgxMk8Enkk2iPQ3XwO6oE/KYD8lMYOziCkeSB9G4adPM4nR1DE8Tf5JeWH6Bw== dependencies: chalk "^4.0.0" - jest-diff "^27.5.1" - jest-get-type "^27.5.1" - pretty-format "^27.5.1" + jest-diff "^28.1.3" + jest-get-type "^28.0.2" + pretty-format "^28.1.3" -jest-message-util@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-message-util/-/jest-message-util-27.5.1.tgz#bdda72806da10d9ed6425e12afff38cd1458b6cf" - integrity sha512-rMyFe1+jnyAAf+NHwTclDz0eAaLkVDdKVHHBFWsBWHnnh5YeJMNWWsv7AbFYXfK3oTqvL7VTWkhNLu1jX24D+g== +jest-message-util@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-message-util/-/jest-message-util-28.1.3.tgz#232def7f2e333f1eecc90649b5b94b0055e7c43d" + integrity sha512-PFdn9Iewbt575zKPf1286Ht9EPoJmYT7P0kY+RibeYZ2XtOr53pDLEFoTWXbd1h4JiGiWpTBC84fc8xMXQMb7g== dependencies: "@babel/code-frame" "^7.12.13" - "@jest/types" "^27.5.1" + "@jest/types" "^28.1.3" "@types/stack-utils" "^2.0.0" chalk "^4.0.0" graceful-fs "^4.2.9" micromatch "^4.0.4" - pretty-format "^27.5.1" + pretty-format "^28.1.3" slash "^3.0.0" stack-utils "^2.0.3" -jest-mock@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-mock/-/jest-mock-27.5.1.tgz#19948336d49ef4d9c52021d34ac7b5f36ff967d6" - integrity sha512-K4jKbY1d4ENhbrG2zuPWaQBvDly+iZ2yAW+T1fATN78hc0sInwn7wZB8XtlNnvHug5RMwV897Xm4LqmPM4e2Og== +jest-mock@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-mock/-/jest-mock-28.1.3.tgz#d4e9b1fc838bea595c77ab73672ebf513ab249da" + integrity sha512-o3J2jr6dMMWYVH4Lh/NKmDXdosrsJgi4AviS8oXLujcjpCMBb1FMsblDnOXKZKfSiHLxYub1eS0IHuRXsio9eA== dependencies: - "@jest/types" "^27.5.1" + "@jest/types" "^28.1.3" "@types/node" "*" jest-pnp-resolver@^1.2.2: @@ -3482,200 +2785,174 @@ jest-pnp-resolver@^1.2.2: resolved "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz" integrity sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w== -jest-regex-util@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-regex-util/-/jest-regex-util-27.5.1.tgz#4da143f7e9fd1e542d4aa69617b38e4a78365b95" - integrity sha512-4bfKq2zie+x16okqDXjXn9ql2B0dScQu+vcwe4TvFVhkVyuWLqpZrZtXxLLWoXYgn0E87I6r6GRYHF7wFZBUvg== +jest-regex-util@^28.0.2: + version "28.0.2" + resolved "https://registry.yarnpkg.com/jest-regex-util/-/jest-regex-util-28.0.2.tgz#afdc377a3b25fb6e80825adcf76c854e5bf47ead" + integrity sha512-4s0IgyNIy0y9FK+cjoVYoxamT7Zeo7MhzqRGx7YDYmaQn1wucY9rotiGkBzzcMXTtjrCAP/f7f+E0F7+fxPNdw== -jest-resolve-dependencies@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-resolve-dependencies/-/jest-resolve-dependencies-27.5.1.tgz#d811ecc8305e731cc86dd79741ee98fed06f1da8" - integrity sha512-QQOOdY4PE39iawDn5rzbIePNigfe5B9Z91GDD1ae/xNDlu9kaat8QQ5EKnNmVWPV54hUdxCVwwj6YMgR2O7IOg== +jest-resolve-dependencies@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-resolve-dependencies/-/jest-resolve-dependencies-28.1.3.tgz#8c65d7583460df7275c6ea2791901fa975c1fe66" + integrity sha512-qa0QO2Q0XzQoNPouMbCc7Bvtsem8eQgVPNkwn9LnS+R2n8DaVDPL/U1gngC0LTl1RYXJU0uJa2BMC2DbTfFrHA== dependencies: - "@jest/types" "^27.5.1" - jest-regex-util "^27.5.1" - jest-snapshot "^27.5.1" + jest-regex-util "^28.0.2" + jest-snapshot "^28.1.3" -jest-resolve@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-resolve/-/jest-resolve-27.5.1.tgz#a2f1c5a0796ec18fe9eb1536ac3814c23617b384" - integrity sha512-FFDy8/9E6CV83IMbDpcjOhumAQPDyETnU2KZ1O98DwTnz8AOBsW/Xv3GySr1mOZdItLR+zDZ7I/UdTFbgSOVCw== +jest-resolve@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-resolve/-/jest-resolve-28.1.3.tgz#cfb36100341ddbb061ec781426b3c31eb51aa0a8" + integrity sha512-Z1W3tTjE6QaNI90qo/BJpfnvpxtaFTFw5CDgwpyE/Kz8U/06N1Hjf4ia9quUhCh39qIGWF1ZuxFiBiJQwSEYKQ== dependencies: - "@jest/types" "^27.5.1" chalk "^4.0.0" graceful-fs "^4.2.9" - jest-haste-map "^27.5.1" + jest-haste-map "^28.1.3" jest-pnp-resolver "^1.2.2" - jest-util "^27.5.1" - jest-validate "^27.5.1" + jest-util "^28.1.3" + jest-validate "^28.1.3" resolve "^1.20.0" resolve.exports "^1.1.0" slash "^3.0.0" -jest-runner@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-runner/-/jest-runner-27.5.1.tgz#071b27c1fa30d90540805c5645a0ec167c7b62e5" - integrity sha512-g4NPsM4mFCOwFKXO4p/H/kWGdJp9V8kURY2lX8Me2drgXqG7rrZAx5kv+5H7wtt/cdFIjhqYx1HrlqWHaOvDaQ== +jest-runner@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-runner/-/jest-runner-28.1.3.tgz#5eee25febd730b4713a2cdfd76bdd5557840f9a1" + integrity sha512-GkMw4D/0USd62OVO0oEgjn23TM+YJa2U2Wu5zz9xsQB1MxWKDOlrnykPxnMsN0tnJllfLPinHTka61u0QhaxBA== dependencies: - "@jest/console" "^27.5.1" - "@jest/environment" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/transform" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/console" "^28.1.3" + "@jest/environment" "^28.1.3" + "@jest/test-result" "^28.1.3" + "@jest/transform" "^28.1.3" + "@jest/types" "^28.1.3" "@types/node" "*" chalk "^4.0.0" - emittery "^0.8.1" + emittery "^0.10.2" graceful-fs "^4.2.9" - jest-docblock "^27.5.1" - jest-environment-jsdom "^27.5.1" - jest-environment-node "^27.5.1" - jest-haste-map "^27.5.1" - jest-leak-detector "^27.5.1" - jest-message-util "^27.5.1" - jest-resolve "^27.5.1" - jest-runtime "^27.5.1" - jest-util "^27.5.1" - jest-worker "^27.5.1" - source-map-support "^0.5.6" - throat "^6.0.1" - -jest-runtime@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-runtime/-/jest-runtime-27.5.1.tgz#4896003d7a334f7e8e4a53ba93fb9bcd3db0a1af" - integrity sha512-o7gxw3Gf+H2IGt8fv0RiyE1+r83FJBRruoA+FXrlHw6xEyBsU8ugA6IPfTdVyA0w8HClpbK+DGJxH59UrNMx8A== - dependencies: - "@jest/environment" "^27.5.1" - "@jest/fake-timers" "^27.5.1" - "@jest/globals" "^27.5.1" - "@jest/source-map" "^27.5.1" - "@jest/test-result" "^27.5.1" - "@jest/transform" "^27.5.1" - "@jest/types" "^27.5.1" + jest-docblock "^28.1.1" + jest-environment-node "^28.1.3" + jest-haste-map "^28.1.3" + jest-leak-detector "^28.1.3" + jest-message-util "^28.1.3" + jest-resolve "^28.1.3" + jest-runtime "^28.1.3" + jest-util "^28.1.3" + jest-watcher "^28.1.3" + jest-worker "^28.1.3" + p-limit "^3.1.0" + source-map-support "0.5.13" + +jest-runtime@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-runtime/-/jest-runtime-28.1.3.tgz#a57643458235aa53e8ec7821949e728960d0605f" + integrity sha512-NU+881ScBQQLc1JHG5eJGU7Ui3kLKrmwCPPtYsJtBykixrM2OhVQlpMmFWJjMyDfdkGgBMNjXCGB/ebzsgNGQw== + dependencies: + "@jest/environment" "^28.1.3" + "@jest/fake-timers" "^28.1.3" + "@jest/globals" "^28.1.3" + "@jest/source-map" "^28.1.2" + "@jest/test-result" "^28.1.3" + "@jest/transform" "^28.1.3" + "@jest/types" "^28.1.3" chalk "^4.0.0" cjs-module-lexer "^1.0.0" collect-v8-coverage "^1.0.0" execa "^5.0.0" glob "^7.1.3" graceful-fs "^4.2.9" - jest-haste-map "^27.5.1" - jest-message-util "^27.5.1" - jest-mock "^27.5.1" - jest-regex-util "^27.5.1" - jest-resolve "^27.5.1" - jest-snapshot "^27.5.1" - jest-util "^27.5.1" + jest-haste-map "^28.1.3" + jest-message-util "^28.1.3" + jest-mock "^28.1.3" + jest-regex-util "^28.0.2" + jest-resolve "^28.1.3" + jest-snapshot "^28.1.3" + jest-util "^28.1.3" slash "^3.0.0" strip-bom "^4.0.0" -jest-serializer@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-serializer/-/jest-serializer-27.5.1.tgz#81438410a30ea66fd57ff730835123dea1fb1f64" - integrity sha512-jZCyo6iIxO1aqUxpuBlwTDMkzOAJS4a3eYz3YzgxxVQFwLeSA7Jfq5cbqCY+JLvTDrWirgusI/0KwxKMgrdf7w== - dependencies: - "@types/node" "*" - graceful-fs "^4.2.9" - -jest-snapshot@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-snapshot/-/jest-snapshot-27.5.1.tgz#b668d50d23d38054a51b42c4039cab59ae6eb6a1" - integrity sha512-yYykXI5a0I31xX67mgeLw1DZ0bJB+gpq5IpSuCAoyDi0+BhgU/RIrL+RTzDmkNTchvDFWKP8lp+w/42Z3us5sA== +jest-snapshot@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-snapshot/-/jest-snapshot-28.1.3.tgz#17467b3ab8ddb81e2f605db05583d69388fc0668" + integrity sha512-4lzMgtiNlc3DU/8lZfmqxN3AYD6GGLbl+72rdBpXvcV+whX7mDrREzkPdp2RnmfIiWBg1YbuFSkXduF2JcafJg== dependencies: - "@babel/core" "^7.7.2" + "@babel/core" "^7.11.6" "@babel/generator" "^7.7.2" "@babel/plugin-syntax-typescript" "^7.7.2" "@babel/traverse" "^7.7.2" - "@babel/types" "^7.0.0" - "@jest/transform" "^27.5.1" - "@jest/types" "^27.5.1" - "@types/babel__traverse" "^7.0.4" + "@babel/types" "^7.3.3" + "@jest/expect-utils" "^28.1.3" + "@jest/transform" "^28.1.3" + "@jest/types" "^28.1.3" + "@types/babel__traverse" "^7.0.6" "@types/prettier" "^2.1.5" babel-preset-current-node-syntax "^1.0.0" chalk "^4.0.0" - expect "^27.5.1" + expect "^28.1.3" graceful-fs "^4.2.9" - jest-diff "^27.5.1" - jest-get-type "^27.5.1" - jest-haste-map "^27.5.1" - jest-matcher-utils "^27.5.1" - jest-message-util "^27.5.1" - jest-util "^27.5.1" + jest-diff "^28.1.3" + jest-get-type "^28.0.2" + jest-haste-map "^28.1.3" + jest-matcher-utils "^28.1.3" + jest-message-util "^28.1.3" + jest-util "^28.1.3" natural-compare "^1.4.0" - pretty-format "^27.5.1" - semver "^7.3.2" + pretty-format "^28.1.3" + semver "^7.3.5" -jest-util@^27.0.0, jest-util@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-util/-/jest-util-27.5.1.tgz#3ba9771e8e31a0b85da48fe0b0891fb86c01c2f9" - integrity sha512-Kv2o/8jNvX1MQ0KGtw480E/w4fBCDOnH6+6DmeKi6LZUIlKA5kwY0YNdlzaWTiVgxqAqik11QyxDOKk543aKXw== +jest-util@^28.0.0, jest-util@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-util/-/jest-util-28.1.3.tgz#f4f932aa0074f0679943220ff9cbba7e497028b0" + integrity sha512-XdqfpHwpcSRko/C35uLYFM2emRAltIIKZiJ9eAmhjsj0CqZMa0p1ib0R5fWIqGhn1a103DebTbpqIaP1qCQ6tQ== dependencies: - "@jest/types" "^27.5.1" + "@jest/types" "^28.1.3" "@types/node" "*" chalk "^4.0.0" ci-info "^3.2.0" graceful-fs "^4.2.9" picomatch "^2.2.3" -jest-validate@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-validate/-/jest-validate-27.5.1.tgz#9197d54dc0bdb52260b8db40b46ae668e04df067" - integrity sha512-thkNli0LYTmOI1tDB3FI1S1RTp/Bqyd9pTarJwL87OIBFuqEb5Apv5EaApEudYg4g86e3CT6kM0RowkhtEnCBQ== +jest-validate@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-validate/-/jest-validate-28.1.3.tgz#e322267fd5e7c64cea4629612c357bbda96229df" + integrity sha512-SZbOGBWEsaTxBGCOpsRWlXlvNkvTkY0XxRfh7zYmvd8uL5Qzyg0CHAXiXKROflh801quA6+/DsT4ODDthOC/OA== dependencies: - "@jest/types" "^27.5.1" + "@jest/types" "^28.1.3" camelcase "^6.2.0" chalk "^4.0.0" - jest-get-type "^27.5.1" + jest-get-type "^28.0.2" leven "^3.1.0" - pretty-format "^27.5.1" + pretty-format "^28.1.3" -jest-watcher@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-watcher/-/jest-watcher-27.5.1.tgz#71bd85fb9bde3a2c2ec4dc353437971c43c642a2" - integrity sha512-z676SuD6Z8o8qbmEGhoEUFOM1+jfEiL3DXHK/xgEiG2EyNYfFG60jluWcupY6dATjfEsKQuibReS1djInQnoVw== +jest-watcher@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-watcher/-/jest-watcher-28.1.3.tgz#c6023a59ba2255e3b4c57179fc94164b3e73abd4" + integrity sha512-t4qcqj9hze+jviFPUN3YAtAEeFnr/azITXQEMARf5cMwKY2SMBRnCQTXLixTl20OR6mLh9KLMrgVJgJISym+1g== dependencies: - "@jest/test-result" "^27.5.1" - "@jest/types" "^27.5.1" + "@jest/test-result" "^28.1.3" + "@jest/types" "^28.1.3" "@types/node" "*" ansi-escapes "^4.2.1" chalk "^4.0.0" - jest-util "^27.5.1" + emittery "^0.10.2" + jest-util "^28.1.3" string-length "^4.0.1" -jest-worker@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-27.5.1.tgz#8d146f0900e8973b106b6f73cc1e9a8cb86f8db0" - integrity sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg== +jest-worker@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-28.1.3.tgz#7e3c4ce3fa23d1bb6accb169e7f396f98ed4bb98" + integrity sha512-CqRA220YV/6jCo8VWvAt1KKx6eek1VIHMPeLEbpcfSfkEeWyBNppynM/o6q+Wmw+sOhos2ml34wZbSX3G13//g== dependencies: "@types/node" "*" merge-stream "^2.0.0" supports-color "^8.0.0" -jest@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest/-/jest-27.5.1.tgz#dadf33ba70a779be7a6fc33015843b51494f63fc" - integrity sha512-Yn0mADZB89zTtjkPJEXwrac3LHudkQMR+Paqa8uxJHCBr9agxztUifWCyiYrjhMPBoUVBjyny0I7XH6ozDr7QQ== +jest@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/jest/-/jest-28.1.3.tgz#e9c6a7eecdebe3548ca2b18894a50f45b36dfc6b" + integrity sha512-N4GT5on8UkZgH0O5LUavMRV1EDEhNTL0KEfRmDIeZHSV7p2XgLoY9t9VDUgL6o+yfdgYHVxuz81G8oB9VG5uyA== dependencies: - "@jest/core" "^27.5.1" + "@jest/core" "^28.1.3" + "@jest/types" "^28.1.3" import-local "^3.0.2" - jest-cli "^27.5.1" - -jmespath@0.16.0: - version "0.16.0" - resolved "https://registry.yarnpkg.com/jmespath/-/jmespath-0.16.0.tgz#b15b0a85dfd4d930d43e69ed605943c802785076" - integrity sha512-9FzQjJ7MATs1tSpnco1K6ayiYE3figslrXA72G2HQ/n76RzvYlofyi5QM+iX4YRs/pu3yzxlVQSST23+dMDknw== - -jose@^1.27.1: - version "1.28.1" - resolved "https://registry.npmjs.org/jose/-/jose-1.28.1.tgz" - integrity sha512-6JK28rFu5ENp/yxMwM+iN7YeaInnY9B9Bggjkz5fuwLiJhbVrl2O4SJr65bdNBPl9y27fdC3Mymh+FVCvozLIg== - dependencies: - "@panva/asn1.js" "^1.0.0" - -jose@^2.0.5: - version "2.0.5" - resolved "https://registry.yarnpkg.com/jose/-/jose-2.0.5.tgz#29746a18d9fff7dcf9d5d2a6f62cb0c7cd27abd3" - integrity sha512-BAiDNeDKTMgk4tvD0BbxJ8xHEHBZgpeRZ1zGPPsitSyMgjoMWiLGYAE7H7NpP5h0lPppQajQs871E8NHUrzVPA== - dependencies: - "@panva/asn1.js" "^1.0.0" + jest-cli "^28.1.3" js-tokens@^4.0.0: version "4.0.0" @@ -3697,91 +2974,11 @@ js-yaml@^4.1.0: dependencies: argparse "^2.0.1" -jsbn@~0.1.0: - version "0.1.1" - resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513" - integrity sha1-peZUwuWi3rXyAdls77yoDA7y9RM= - -jsdom@^16.4.0: - version "16.5.3" - resolved "https://registry.npmjs.org/jsdom/-/jsdom-16.5.3.tgz" - integrity sha512-Qj1H+PEvUsOtdPJ056ewXM4UJPCi4hhLA8wpiz9F2YvsRBhuFsXxtrIFAgGBDynQA9isAMGE91PfUYbdMPXuTA== - dependencies: - abab "^2.0.5" - acorn "^8.1.0" - acorn-globals "^6.0.0" - cssom "^0.4.4" - cssstyle "^2.3.0" - data-urls "^2.0.0" - decimal.js "^10.2.1" - domexception "^2.0.1" - escodegen "^2.0.0" - html-encoding-sniffer "^2.0.1" - is-potential-custom-element-name "^1.0.0" - nwsapi "^2.2.0" - parse5 "6.0.1" - request "^2.88.2" - request-promise-native "^1.0.9" - saxes "^5.0.1" - symbol-tree "^3.2.4" - tough-cookie "^4.0.0" - w3c-hr-time "^1.0.2" - w3c-xmlserializer "^2.0.0" - webidl-conversions "^6.1.0" - whatwg-encoding "^1.0.5" - whatwg-mimetype "^2.3.0" - whatwg-url "^8.5.0" - ws "^7.4.4" - xml-name-validator "^3.0.0" - -jsdom@^16.6.0: - version "16.7.0" - resolved "https://registry.yarnpkg.com/jsdom/-/jsdom-16.7.0.tgz#918ae71965424b197c819f8183a754e18977b710" - integrity sha512-u9Smc2G1USStM+s/x1ru5Sxrl6mPYCbByG1U/hUmqaVsm4tbNyS7CicOSRyuGQYZhTu0h84qkZZQ/I+dzizSVw== - dependencies: - abab "^2.0.5" - acorn "^8.2.4" - acorn-globals "^6.0.0" - cssom "^0.4.4" - cssstyle "^2.3.0" - data-urls "^2.0.0" - decimal.js "^10.2.1" - domexception "^2.0.1" - escodegen "^2.0.0" - form-data "^3.0.0" - html-encoding-sniffer "^2.0.1" - http-proxy-agent "^4.0.1" - https-proxy-agent "^5.0.0" - is-potential-custom-element-name "^1.0.1" - nwsapi "^2.2.0" - parse5 "6.0.1" - saxes "^5.0.1" - symbol-tree "^3.2.4" - tough-cookie "^4.0.0" - w3c-hr-time "^1.0.2" - w3c-xmlserializer "^2.0.0" - webidl-conversions "^6.1.0" - whatwg-encoding "^1.0.5" - whatwg-mimetype "^2.3.0" - whatwg-url "^8.5.0" - ws "^7.4.6" - xml-name-validator "^3.0.0" - jsesc@^2.5.1: version "2.5.2" resolved "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz" integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA== -json-buffer@3.0.0: - version "3.0.0" - resolved "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz" - integrity sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg= - -json-buffer@3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.1.tgz#9338802a30d3b6605fbe0613e094008ca8c05a13" - integrity sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ== - json-parse-even-better-errors@^2.3.0: version "2.3.1" resolved "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz" @@ -3792,38 +2989,11 @@ json-schema-traverse@^0.4.1: resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== -json-schema-traverse@^1.0.0: - version "1.0.0" - resolved "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz" - integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug== - -json-schema@0.4.0: - version "0.4.0" - resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.4.0.tgz#f7de4cf6efab838ebaeb3236474cbba5a1930ab5" - integrity sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA== - json-stable-stringify-without-jsonify@^1.0.1: version "1.0.1" resolved "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz" integrity sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE= -json-stream@^1.0.0: - version "1.0.0" - resolved "https://registry.npmjs.org/json-stream/-/json-stream-1.0.0.tgz" - integrity sha1-GjhU4o0rvuqzHMfd9oPS3cVlJwg= - -json-stringify-safe@~5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" - integrity sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus= - -json5@2.x, json5@^2.1.2: - version "2.2.0" - resolved "https://registry.npmjs.org/json5/-/json5-2.2.0.tgz" - integrity sha512-f+8cldu7X/y7RAJurMEJmdoKXGB/X550w2Nr3tTbezL6RwEE/iMcm+tZnXeoZtKuOq6ft8+CqzEkrIgx1fPoQA== - dependencies: - minimist "^1.2.5" - json5@^1.0.1: version "1.0.1" resolved "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz" @@ -3831,58 +3001,42 @@ json5@^1.0.1: dependencies: minimist "^1.2.0" -jsonpath-plus@^0.19.0: - version "0.19.0" - resolved "https://registry.yarnpkg.com/jsonpath-plus/-/jsonpath-plus-0.19.0.tgz#b901e57607055933dc9a8bef0cc25160ee9dd64c" - integrity sha512-GSVwsrzW9LsA5lzsqe4CkuZ9wp+kxBb2GwNniaWzI2YFn5Ig42rSW8ZxVpWXaAfakXNrx5pgY5AbQq7kzX29kg== - -jsprim@^1.2.2: - version "1.4.2" - resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.2.tgz#712c65533a15c878ba59e9ed5f0e26d5b77c5feb" - integrity sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw== +json5@^2.1.2: + version "2.2.0" + resolved "https://registry.npmjs.org/json5/-/json5-2.2.0.tgz" + integrity sha512-f+8cldu7X/y7RAJurMEJmdoKXGB/X550w2Nr3tTbezL6RwEE/iMcm+tZnXeoZtKuOq6ft8+CqzEkrIgx1fPoQA== dependencies: - assert-plus "1.0.0" - extsprintf "1.3.0" - json-schema "0.4.0" - verror "1.10.0" + minimist "^1.2.5" -keyv@^3.0.0: - version "3.1.0" - resolved "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz" - integrity sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA== - dependencies: - json-buffer "3.0.0" +json5@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.1.tgz#655d50ed1e6f95ad1a3caababd2b0efda10b395c" + integrity sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA== -keyv@^4.0.0: - version "4.1.1" - resolved "https://registry.yarnpkg.com/keyv/-/keyv-4.1.1.tgz#02c538bfdbd2a9308cc932d4096f05ae42bfa06a" - integrity sha512-tGv1yP6snQVDSM4X6yxrv2zzq/EvpW+oYiUz6aueW1u9CtS8RzUQYxxmFwgZlO2jSgCxQbchhxaqXXp2hnKGpQ== +jsx-ast-utils@^3.3.2: + version "3.3.2" + resolved "https://registry.yarnpkg.com/jsx-ast-utils/-/jsx-ast-utils-3.3.2.tgz#afe5efe4332cd3515c065072bd4d6b0aa22152bd" + integrity sha512-4ZCADZHRkno244xlNnn4AOG6sRQ7iBZ5BbgZ4vW4y5IZw7cVUD1PPeblm1xx/nfmMxPdt/LHsXZW8z/j58+l9Q== dependencies: - json-buffer "3.0.1" + array-includes "^3.1.5" + object.assign "^4.1.2" kleur@^3.0.3: version "3.0.3" resolved "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz" integrity sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w== -kubernetes-client@^9.0.0: - version "9.0.0" - resolved "https://registry.npmjs.org/kubernetes-client/-/kubernetes-client-9.0.0.tgz" - integrity sha512-Qy8o42dZVHB9P+cIiKdWpQbz/65l/qW1fDYvlzzeSLftmL1Ne3HEiM+0TmKAwNuRW0pTJN2tRWhcccToclxJ8g== +language-subtag-registry@~0.3.2: + version "0.3.22" + resolved "https://registry.yarnpkg.com/language-subtag-registry/-/language-subtag-registry-0.3.22.tgz#2e1500861b2e457eba7e7ae86877cbd08fa1fd1d" + integrity sha512-tN0MCzyWnoz/4nHS6uxdlFWoUZT7ABptwKPQ52Ea7URk6vll88bWBVhodtnlfEuCcKWNGoc+uGbw1cwa9IKh/w== + +language-tags@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/language-tags/-/language-tags-1.0.5.tgz#d321dbc4da30ba8bf3024e040fa5c14661f9193a" + integrity sha512-qJhlO9cGXi6hBGKoxEG/sKZDAHD5Hnu9Hs4WbOY3pCWXDhw0N8x1NenNzm2EnNLkLkk7J2SdxAkDSbb6ftT+UQ== dependencies: - "@kubernetes/client-node" "0.10.2" - camelcase "^6.0.0" - deepmerge "^4.2.2" - depd "^2.0.0" - js-yaml "^3.13.1" - json-stream "^1.0.0" - openid-client "^3.14.0" - pump "^3.0.0" - qs "^6.9.0" - request "^2.88.2" - swagger-fluent "^5.0.3" - url-join "^4.0.1" - ws "^7.2.3" + language-subtag-registry "~0.3.2" leven@^3.1.0: version "3.1.0" @@ -3897,29 +3051,11 @@ levn@^0.4.1: prelude-ls "^1.2.1" type-check "~0.4.0" -levn@~0.3.0: - version "0.3.0" - resolved "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz" - integrity sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4= - dependencies: - prelude-ls "~1.1.2" - type-check "~0.3.2" - lines-and-columns@^1.1.6: version "1.1.6" resolved "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.1.6.tgz" integrity sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA= -load-json-file@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz" - integrity sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg= - dependencies: - graceful-fs "^4.1.2" - parse-json "^2.2.0" - pify "^2.0.0" - strip-bom "^3.0.0" - locate-path@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz" @@ -3935,41 +3071,48 @@ locate-path@^5.0.0: dependencies: p-locate "^4.1.0" -lodash.clonedeep@^4.5.0: - version "4.5.0" - resolved "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz" - integrity sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8= +locate-path@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286" + integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw== + dependencies: + p-locate "^5.0.0" + +lodash.camelcase@4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz#b28aa6288a2b9fc651035c7711f65ab6190331a6" + integrity sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA== -lodash.flatten@^4.4.0: - version "4.4.0" - resolved "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz" - integrity sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8= +lodash.kebabcase@4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/lodash.kebabcase/-/lodash.kebabcase-4.1.1.tgz#8489b1cb0d29ff88195cceca448ff6d6cc295c36" + integrity sha512-N8XRTIMMqqDgSy4VLKPnJ/+hpGZN+PHQiJnSenYqPaVV/NCqEogTnAdZLQiGKhxX+JCs8waWq2t1XHWKOmlY8g== lodash.memoize@4.x: version "4.1.2" resolved "https://registry.yarnpkg.com/lodash.memoize/-/lodash.memoize-4.1.2.tgz#bcc6c49a42a2840ed997f323eada5ecd182e0bfe" integrity sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4= -lodash.truncate@^4.4.2: - version "4.4.2" - resolved "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz" - integrity sha1-WjUNoLERO4N+z//VgSy+WNbq4ZM= +lodash.merge@^4.6.2: + version "4.6.2" + resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" + integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ== + +lodash.snakecase@4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/lodash.snakecase/-/lodash.snakecase-4.1.1.tgz#39d714a35357147837aefd64b5dcbb16becd8f8d" + integrity sha512-QZ1d4xoBHYUeuouhEq3lk3Uq7ldgyFXGBhg04+oRLnIz8o9T65Eh+8YdroUwn846zchkA9yDsDl5CVVaV2nqYw== + +lodash.upperfirst@4.3.1: + version "4.3.1" + resolved "https://registry.yarnpkg.com/lodash.upperfirst/-/lodash.upperfirst-4.3.1.tgz#1365edf431480481ef0d1c68957a5ed99d49f7ce" + integrity sha512-sReKOYJIJf74dhJONhU4e0/shzi1trVbSWDOhKYE5XV2O+H7Sb2Dihwuc7xWxVl+DgFPyTqIN3zMfT9cq5iWDg== -lodash@^4.17.15, lodash@^4.17.19, lodash@^4.17.20, lodash@^4.7.0: +lodash@^4.17.21: version "4.17.21" resolved "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz" integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== -lowercase-keys@^1.0.0, lowercase-keys@^1.0.1: - version "1.0.1" - resolved "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz" - integrity sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA== - -lowercase-keys@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479" - integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA== - lru-cache@^6.0.0: version "6.0.0" resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" @@ -3984,29 +3127,29 @@ make-dir@^3.0.0: dependencies: semver "^6.0.0" -make-error@1.x, make-error@^1.1.1, make-error@^1.3.6: +make-error@1.x, make-error@^1.1.1: version "1.3.6" resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw== -makeerror@1.0.x: - version "1.0.11" - resolved "https://registry.npmjs.org/makeerror/-/makeerror-1.0.11.tgz" - integrity sha1-4BpckQnyr3lmDk6LlYd5AYT1qWw= +makeerror@1.0.12: + version "1.0.12" + resolved "https://registry.yarnpkg.com/makeerror/-/makeerror-1.0.12.tgz#3e5dd2079a82e812e983cc6610c4a2cb0eaa801a" + integrity sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg== dependencies: - tmpl "1.0.x" + tmpl "1.0.5" merge-stream@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== -merge2@^1.3.0: +merge2@^1.3.0, merge2@^1.4.1: version "1.4.1" resolved "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz" integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== -micromatch@^4.0.2, micromatch@^4.0.4: +micromatch@^4.0.4: version "4.0.4" resolved "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz" integrity sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg== @@ -4014,65 +3157,28 @@ micromatch@^4.0.2, micromatch@^4.0.4: braces "^3.0.1" picomatch "^2.2.3" -mime-db@1.51.0: - version "1.51.0" - resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.51.0.tgz#d9ff62451859b18342d960850dc3cfb77e63fb0c" - integrity sha512-5y8A56jg7XVQx2mbv1lu49NR4dokRnhZYTtL+KGfaa27uq4pSTXkwQkFJl4pkRMyNFz/EtYDSkiiEHx3F7UN6g== - -mime-types@^2.1.12, mime-types@~2.1.19: - version "2.1.34" - resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.34.tgz#5a712f9ec1503511a945803640fafe09d3793c24" - integrity sha512-6cP692WwGIs9XXdOO4++N+7qjqv0rqxxVvJ3VHPh/Sc9mVZcQP+ZGhkKiTvWMQRr2tbHkJP/Yn7Y0npb3ZBs4A== - dependencies: - mime-db "1.51.0" - mimic-fn@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== -mimic-response@^1.0.0, mimic-response@^1.0.1: +min-indent@^1.0.0: version "1.0.1" - resolved "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz" - integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ== + resolved "https://registry.yarnpkg.com/min-indent/-/min-indent-1.0.1.tgz#a63f681673b30571fbe8bc25686ae746eefa9869" + integrity sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg== -mimic-response@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-3.1.0.tgz#2d1d59af9c1b129815accc2c46a022a5ce1fa3c9" - integrity sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ== - -minimatch@^3.0.4: +minimatch@^3.0.4, minimatch@^3.1.2: version "3.1.2" resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== dependencies: brace-expansion "^1.1.7" -minimist@^1.2.0, minimist@^1.2.5: +minimist@^1.2.0, minimist@^1.2.5, minimist@^1.2.6: version "1.2.6" resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.6.tgz#8637a5b759ea0d6e98702cfb3a9283323c93af44" integrity sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q== -minipass@^3.0.0: - version "3.1.6" - resolved "https://registry.yarnpkg.com/minipass/-/minipass-3.1.6.tgz#3b8150aa688a711a1521af5e8779c1d3bb4f45ee" - integrity sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ== - dependencies: - yallist "^4.0.0" - -minizlib@^2.1.1: - version "2.1.2" - resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-2.1.2.tgz#e90d3466ba209b932451508a11ce3d3632145931" - integrity sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg== - dependencies: - minipass "^3.0.0" - yallist "^4.0.0" - -mkdirp@^1.0.3: - version "1.0.4" - resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e" - integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw== - ms@2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz" @@ -4083,44 +3189,32 @@ ms@2.1.2: resolved "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz" integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== -multimap@^1.1.0: - version "1.1.0" - resolved "https://registry.npmjs.org/multimap/-/multimap-1.1.0.tgz" - integrity sha512-0ZIR9PasPxGXmRsEF8jsDzndzHDj7tIav+JUmvIFB/WHswliFnquxECT/De7GR4yg99ky/NlRKJT82G1y271bw== - -nanoid@^3.3.1: - version "3.3.1" - resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.1.tgz#6347a18cac88af88f58af0b3594b723d5e99bb35" - integrity sha512-n6Vs/3KGyxPQd6uO0eH4Bv0ojGSUvuLlIHtC3Y0kEO23YRge8H9x1GCzLn28YX0H66pMkxuaeESFq4tKISKwdw== +ms@^2.1.1: + version "2.1.3" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" + integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== natural-compare@^1.4.0: version "1.4.0" resolved "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz" integrity sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc= -node-fetch@^2.6.1: - version "2.6.7" - resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.6.7.tgz#24de9fba827e3b4ae44dc8b20256a379160052ad" - integrity sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ== - dependencies: - whatwg-url "^5.0.0" - node-int64@^0.4.0: version "0.4.0" resolved "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz" integrity sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs= -node-releases@^1.1.71: - version "1.1.72" - resolved "https://registry.npmjs.org/node-releases/-/node-releases-1.1.72.tgz" - integrity sha512-LLUo+PpH3dU6XizX3iVoubUNheF/owjXCZZ5yACDxNnPtgFuludV1ZL3ayK1kVep42Rmm0+R9/Y60NQbZ2bifw== - node-releases@^2.0.2: version "2.0.2" resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.2.tgz#7139fe71e2f4f11b47d4d2986aaf8c48699e0c01" integrity sha512-XxYDdcQ6eKqp/YjI+tb2C5WM2LgjnZrfYg4vgQt49EK268b6gYCHsBLrK2qvJo4FmCtqmKezb0WZFK4fkrZNsg== -normalize-package-data@^2.3.2, normalize-package-data@^2.5.0: +node-releases@^2.0.6: + version "2.0.6" + resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.6.tgz#8a7088c63a55e493845683ebf3c828d8c51c5503" + integrity sha512-PiVXnNuFm5+iYkLBNeq5211hvO38y63T0i2KKh2KnUs3RpzJ+JtODFjkD8yjLwnDkTYF1eKXheUwdssR+NRZdg== + +normalize-package-data@^2.5.0: version "2.5.0" resolved "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz" integrity sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA== @@ -4135,16 +3229,6 @@ normalize-path@^3.0.0: resolved "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz" integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== -normalize-url@^4.1.0: - version "4.5.1" - resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-4.5.1.tgz#0dd90cf1288ee1d1313b87081c9a5932ee48518a" - integrity sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA== - -normalize-url@^6.0.1: - version "6.1.0" - resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-6.1.0.tgz#40d0885b535deffe3f3147bec877d05fe4c5668a" - integrity sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A== - npm-run-path@^4.0.1: version "4.0.1" resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-4.0.1.tgz#b7ecd1e5ed53da8e37a55e1c2269e0b97ed748ea" @@ -4152,20 +3236,10 @@ npm-run-path@^4.0.1: dependencies: path-key "^3.0.0" -nwsapi@^2.2.0: - version "2.2.0" - resolved "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.0.tgz" - integrity sha512-h2AatdwYH+JHiZpv7pt/gSX1XoRGb7L/qSIeuqA6GwYoF9w1vP1cw42TO0aI2pNyshRK5893hNSl+1//vHK7hQ== - -oauth-sign@~0.9.0: - version "0.9.0" - resolved "https://registry.yarnpkg.com/oauth-sign/-/oauth-sign-0.9.0.tgz#47a7b016baa68b5fa0ecf3dee08a85c679ac6455" - integrity sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ== - -object-hash@^2.0.1: - version "2.2.0" - resolved "https://registry.yarnpkg.com/object-hash/-/object-hash-2.2.0.tgz#5ad518581eefc443bd763472b8ff2e9c2c0d54a5" - integrity sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw== +object-inspect@^1.12.0: + version "1.12.2" + resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.12.2.tgz#c0641f26394532f28ab8d796ab954e43c009a8ea" + integrity sha512-z+cPxW0QGUp0mcqcsgQyLVRDoXFQbXOwBaqyF7VIgI4TWNQsDHrBpUQslRmIfAoYWdYzs6UlKJtB2XJpTaNSpQ== object-inspect@^1.9.0: version "1.11.1" @@ -4187,22 +3261,16 @@ object.assign@^4.1.2: has-symbols "^1.0.1" object-keys "^1.1.1" -object.values@^1.1.1: - version "1.1.3" - resolved "https://registry.npmjs.org/object.values/-/object.values-1.1.3.tgz" - integrity sha512-nkF6PfDB9alkOUxpf1HNm/QlkeW3SReqL5WXeBLpEJJnlPSvRaDQpW3gQTksTN3fgJX4hL42RzKyOin6ff3tyw== +object.values@^1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.1.5.tgz#959f63e3ce9ef108720333082131e4a459b716ac" + integrity sha512-QUZRW0ilQ3PnPpbNtgdNV1PDbEqLIiSFB3l+EnGtBQ/8SUTLj1PZwtQHABZtLgwpJZTSZhuGLOGk57Drx2IvYg== dependencies: call-bind "^1.0.2" define-properties "^1.1.3" - es-abstract "^1.18.0-next.2" - has "^1.0.3" + es-abstract "^1.19.1" -oidc-token-hash@^5.0.0, oidc-token-hash@^5.0.1: - version "5.0.1" - resolved "https://registry.npmjs.org/oidc-token-hash/-/oidc-token-hash-5.0.1.tgz" - integrity sha512-EvoOtz6FIEBzE+9q253HsLCVRiK/0doEJ2HCvvqMQb3dHZrP3WlJKYtJ55CRTw4jmYomzH4wkPuCj/I3ZvpKxQ== - -once@^1.3.0, once@^1.3.1, once@^1.4.0: +once@^1.3.0: version "1.4.0" resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E= @@ -4216,46 +3284,6 @@ onetime@^5.1.2: dependencies: mimic-fn "^2.1.0" -openid-client@^3.14.0: - version "3.15.10" - resolved "https://registry.npmjs.org/openid-client/-/openid-client-3.15.10.tgz" - integrity sha512-C9r6/iVzNQ7aGp0krS5mFIY5nY8AH6ajYCH0Njns6AXy2fM3Khw/dY97QlaFJWW2QLhec6xfEk23LZw9EeX66Q== - dependencies: - "@types/got" "^9.6.9" - base64url "^3.0.1" - got "^9.6.0" - jose "^1.27.1" - lru-cache "^6.0.0" - make-error "^1.3.6" - object-hash "^2.0.1" - oidc-token-hash "^5.0.0" - p-any "^3.0.0" - -openid-client@^4.1.1: - version "4.9.1" - resolved "https://registry.yarnpkg.com/openid-client/-/openid-client-4.9.1.tgz#4f00a9d1566c0fa08f0dd5986cf0e6b1e5d14186" - integrity sha512-DYUF07AHjI3QDKqKbn2F7RqozT4hyi4JvmpodLrq0HHoNP7t/AjeG/uqiBK1/N2PZSAQEThVjDLHSmJN4iqu/w== - dependencies: - aggregate-error "^3.1.0" - got "^11.8.0" - jose "^2.0.5" - lru-cache "^6.0.0" - make-error "^1.3.6" - object-hash "^2.0.1" - oidc-token-hash "^5.0.1" - -optionator@^0.8.1: - version "0.8.3" - resolved "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz" - integrity sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA== - dependencies: - deep-is "~0.1.3" - fast-levenshtein "~2.0.6" - levn "~0.3.0" - prelude-ls "~1.1.2" - type-check "~0.3.2" - word-wrap "~1.2.3" - optionator@^0.9.1: version "0.9.1" resolved "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz" @@ -4268,24 +3296,6 @@ optionator@^0.9.1: type-check "^0.4.0" word-wrap "^1.2.3" -p-any@^3.0.0: - version "3.0.0" - resolved "https://registry.npmjs.org/p-any/-/p-any-3.0.0.tgz" - integrity sha512-5rqbqfsRWNb0sukt0awwgJMlaep+8jV45S15SKKB34z4UuzjcofIfnriCBhWjZP2jbVtjt9yRl7buB6RlKsu9w== - dependencies: - p-cancelable "^2.0.0" - p-some "^5.0.0" - -p-cancelable@^1.0.0: - version "1.1.0" - resolved "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz" - integrity sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw== - -p-cancelable@^2.0.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-2.1.1.tgz#aab7fbd416582fa32a3db49859c122487c5ed2cf" - integrity sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg== - p-limit@^1.1.0: version "1.3.0" resolved "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz" @@ -4300,6 +3310,13 @@ p-limit@^2.2.0: dependencies: p-try "^2.0.0" +p-limit@^3.0.2, p-limit@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" + integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ== + dependencies: + yocto-queue "^0.1.0" + p-locate@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz" @@ -4314,13 +3331,12 @@ p-locate@^4.1.0: dependencies: p-limit "^2.2.0" -p-some@^5.0.0: +p-locate@^5.0.0: version "5.0.0" - resolved "https://registry.npmjs.org/p-some/-/p-some-5.0.0.tgz" - integrity sha512-Js5XZxo6vHjB9NOYAzWDYAIyyiPvva0DWESAIWIK7uhSpGsyg5FwUPxipU/SOQx5x9EqhOh545d1jo6cVkitig== + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834" + integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw== dependencies: - aggregate-error "^3.0.0" - p-cancelable "^2.0.0" + p-limit "^3.0.2" p-try@^1.0.0: version "1.0.0" @@ -4339,13 +3355,6 @@ parent-module@^1.0.0: dependencies: callsites "^3.0.0" -parse-json@^2.2.0: - version "2.2.0" - resolved "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz" - integrity sha1-9ID0BDTvgHQfhGkJn43qGPVaTck= - dependencies: - error-ex "^1.2.0" - parse-json@^5.0.0, parse-json@^5.2.0: version "5.2.0" resolved "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz" @@ -4356,11 +3365,6 @@ parse-json@^5.0.0, parse-json@^5.2.0: json-parse-even-better-errors "^2.3.0" lines-and-columns "^1.1.6" -parse5@6.0.1: - version "6.0.1" - resolved "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz" - integrity sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw== - path-exists@^3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz" @@ -4386,50 +3390,26 @@ path-parse@^1.0.6, path-parse@^1.0.7: resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735" integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw== -path-type@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz" - integrity sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM= - dependencies: - pify "^2.0.0" - path-type@^4.0.0: version "4.0.0" resolved "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz" integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw== -performance-now@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b" - integrity sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns= - picocolors@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== -picomatch@^2.0.4, picomatch@^2.2.1, picomatch@^2.2.3: +picomatch@^2.0.4, picomatch@^2.2.3: version "2.2.3" resolved "https://registry.npmjs.org/picomatch/-/picomatch-2.2.3.tgz" integrity sha512-KpELjfwcCDUb9PeigTs2mBJzXUPzAuP2oPcA989He8Rte0+YUAjw1JVedDhuTKPkHjSYzMN3npC9luThGYEKdg== -pify@^2.0.0: - version "2.3.0" - resolved "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz" - integrity sha1-7RQaasBDqEnqWISY59yosVMw6Qw= - pirates@^4.0.4: version "4.0.5" resolved "https://registry.yarnpkg.com/pirates/-/pirates-4.0.5.tgz#feec352ea5c3268fb23a37c702ab1699f35a5f3b" integrity sha512-8V9+HQPupnaXMA23c5hvl69zXvTwTzyAYasnkb0Tts4XvO4CliqONMOnvlq26rkhLC3nWDFBJf73LU1e1VZLaQ== -pkg-dir@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/pkg-dir/-/pkg-dir-2.0.0.tgz" - integrity sha1-9tXREJ4Z1j7fQo4L1X4Sd3YVM0s= - dependencies: - find-up "^2.1.0" - pkg-dir@^4.2.0: version "4.2.0" resolved "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz" @@ -4447,16 +3427,6 @@ prelude-ls@^1.2.1: resolved "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz" integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g== -prelude-ls@~1.1.2: - version "1.1.2" - resolved "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz" - integrity sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ= - -prepend-http@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz" - integrity sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc= - prettier-linter-helpers@^1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz" @@ -4464,39 +3434,25 @@ prettier-linter-helpers@^1.0.0: dependencies: fast-diff "^1.1.2" +prettier@2.7.1: + version "2.7.1" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.7.1.tgz#e235806850d057f97bb08368a4f7d899f7760c64" + integrity sha512-ujppO+MkdPqoVINuDFDRLClm7D78qbDt0/NR+wp5FqEZOoTNAjPHWj17QRhu7geIHJfcNhRk1XVQmF8Bp3ye+g== + prettier@^2.2.1: version "2.2.1" resolved "https://registry.npmjs.org/prettier/-/prettier-2.2.1.tgz" integrity sha512-PqyhM2yCjg/oKkFPtTGUojv7gnZAoG80ttl45O6x2Ug/rMJw4wcc9k6aaf2hibP7BGVCCM33gZoGjyvt9mm16Q== -prettier@^2.5.1: - version "2.5.1" - resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.5.1.tgz#fff75fa9d519c54cf0fce328c1017d94546bc56a" - integrity sha512-vBZcPRUR5MZJwoyi3ZoyQlc1rXeEck8KgeC9AwwOn+exuxLxq5toTRDTSaVrXHxelDMHy9zlicw8u66yxoSUFg== - -pretty-format@^27.0.0: - version "27.4.2" - resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-27.4.2.tgz#e4ce92ad66c3888423d332b40477c87d1dac1fb8" - integrity sha512-p0wNtJ9oLuvgOQDEIZ9zQjZffK7KtyR6Si0jnXULIDwrlNF8Cuir3AZP0hHv0jmKuNN/edOnbMjnzd4uTcmWiw== +pretty-format@^28.0.0, pretty-format@^28.1.3: + version "28.1.3" + resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-28.1.3.tgz#c9fba8cedf99ce50963a11b27d982a9ae90970d5" + integrity sha512-8gFb/To0OmxHR9+ZTb14Df2vNxdGCX8g1xWGUTqUw5TiZvcQf5sHKObd5UcPyLLyowNwDAMTF3XWOG1B6mxl1Q== dependencies: - "@jest/types" "^27.4.2" + "@jest/schemas" "^28.1.3" ansi-regex "^5.0.1" ansi-styles "^5.0.0" - react-is "^17.0.1" - -pretty-format@^27.5.1: - version "27.5.1" - resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-27.5.1.tgz#2181879fdea51a7a5851fb39d920faa63f01d88e" - integrity sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ== - dependencies: - ansi-regex "^5.0.1" - ansi-styles "^5.0.0" - react-is "^17.0.1" - -progress@^2.0.0: - version "2.0.3" - resolved "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz" - integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA== + react-is "^18.0.0" prompts@^2.0.1: version "2.4.1" @@ -4506,68 +3462,20 @@ prompts@^2.0.1: kleur "^3.0.3" sisteransi "^1.0.5" -psl@^1.1.28, psl@^1.1.33: - version "1.8.0" - resolved "https://registry.yarnpkg.com/psl/-/psl-1.8.0.tgz#9326f8bcfb013adcc005fdff056acce020e51c24" - integrity sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ== - -pump@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64" - integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww== - dependencies: - end-of-stream "^1.1.0" - once "^1.3.1" - -punycode@1.3.2: - version "1.3.2" - resolved "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz" - integrity sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0= - -punycode@^2.1.0, punycode@^2.1.1: +punycode@^2.1.0: version "2.1.1" resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.1.1.tgz#b58b010ac40c22c5657616c8d2c2c02c7bf479ec" integrity sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A== -qs@^6.9.0: - version "6.10.1" - resolved "https://registry.npmjs.org/qs/-/qs-6.10.1.tgz" - integrity sha512-M528Hph6wsSVOBiYUnGf+K/7w0hNshs/duGsNXPUCLH5XAqjEtiPGwNONLV0tBH8NoGb0mvD5JubnUTrujKDTg== - dependencies: - side-channel "^1.0.4" - -qs@~6.5.2: - version "6.5.3" - resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.3.tgz#3aeeffc91967ef6e35c0e488ef46fb296ab76aad" - integrity sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA== - -querystring@0.2.0: - version "0.2.0" - resolved "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz" - integrity sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA= - queue-microtask@^1.2.2: version "1.2.3" resolved "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz" integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== -quick-lru@^5.1.1: - version "5.1.1" - resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-5.1.1.tgz#366493e6b3e42a3a6885e2e99d18f80fb7a8c932" - integrity sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA== - -react-is@^17.0.1: - version "17.0.2" - resolved "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz" - integrity sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w== - -read-pkg-up@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz" - integrity sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4= - dependencies: - find-up "^2.0.0" - read-pkg "^2.0.0" +react-is@^18.0.0: + version "18.2.0" + resolved "https://registry.yarnpkg.com/react-is/-/react-is-18.2.0.tgz#199431eeaaa2e09f86427efbb4f1473edb47609b" + integrity sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w== read-pkg-up@^7.0.1: version "7.0.1" @@ -4578,15 +3486,6 @@ read-pkg-up@^7.0.1: read-pkg "^5.2.0" type-fest "^0.8.1" -read-pkg@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz" - integrity sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg= - dependencies: - load-json-file "^2.0.0" - normalize-package-data "^2.3.2" - path-type "^2.0.0" - read-pkg@^5.2.0: version "5.2.0" resolved "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz" @@ -4597,90 +3496,40 @@ read-pkg@^5.2.0: parse-json "^5.0.0" type-fest "^0.6.0" -rechoir@^0.6.2: - version "0.6.2" - resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.6.2.tgz#85204b54dba82d5742e28c96756ef43af50e3384" - integrity sha1-hSBLVNuoLVdC4oyWdW70OvUOM4Q= - dependencies: - resolve "^1.1.6" +regenerator-runtime@^0.13.4: + version "0.13.9" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz#8925742a98ffd90814988d7566ad30ca3b263b52" + integrity sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA== -reflect-metadata@^0.1.13: - version "0.1.13" - resolved "https://registry.yarnpkg.com/reflect-metadata/-/reflect-metadata-0.1.13.tgz#67ae3ca57c972a2aa1642b10fe363fe32d49dc08" - integrity sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg== +regexp-tree@^0.1.24: + version "0.1.24" + resolved "https://registry.yarnpkg.com/regexp-tree/-/regexp-tree-0.1.24.tgz#3d6fa238450a4d66e5bc9c4c14bb720e2196829d" + integrity sha512-s2aEVuLhvnVJW6s/iPgEGK6R+/xngd2jNQ+xy4bXNDKxZKJH6jpPHY6kVeVv1IeLCHgswRj+Kl3ELaDjG6V1iw== -regexp-tree@^0.1.22, regexp-tree@~0.1.1: +regexp-tree@~0.1.1: version "0.1.23" resolved "https://registry.npmjs.org/regexp-tree/-/regexp-tree-0.1.23.tgz" integrity sha512-+7HWfb4Bvu8Rs2eQTUIpX9I/PlQkYOuTNbRpKLJlQpSgwSkzFYh+pUj0gtvglnOZLKB6YgnIgRuJ2/IlpL48qw== -regexpp@^3.0.0, regexpp@^3.1.0: - version "3.1.0" - resolved "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz" - integrity sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q== - -request-promise-core@1.1.4: - version "1.1.4" - resolved "https://registry.npmjs.org/request-promise-core/-/request-promise-core-1.1.4.tgz" - integrity sha512-TTbAfBBRdWD7aNNOoVOBH4pN/KigV6LyapYNNlAPA8JwbovRti1E88m3sYAwsLi5ryhPKsE9APwnjFTgdUjTpw== +regexp.prototype.flags@^1.4.3: + version "1.4.3" + resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz#87cab30f80f66660181a3bb7bf5981a872b367ac" + integrity sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA== dependencies: - lodash "^4.17.19" + call-bind "^1.0.2" + define-properties "^1.1.3" + functions-have-names "^1.2.2" -request-promise-native@^1.0.9: - version "1.0.9" - resolved "https://registry.npmjs.org/request-promise-native/-/request-promise-native-1.0.9.tgz" - integrity sha512-wcW+sIUiWnKgNY0dqCpOZkUbF/I+YPi+f09JZIDa39Ec+q82CpSYniDp+ISgTTbKmnpJWASeJBPZmoxH84wt3g== - dependencies: - request-promise-core "1.1.4" - stealthy-require "^1.1.1" - tough-cookie "^2.3.3" - -request@^2.88.0, request@^2.88.2: - version "2.88.2" - resolved "https://registry.npmjs.org/request/-/request-2.88.2.tgz" - integrity sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw== - dependencies: - aws-sign2 "~0.7.0" - aws4 "^1.8.0" - caseless "~0.12.0" - combined-stream "~1.0.6" - extend "~3.0.2" - forever-agent "~0.6.1" - form-data "~2.3.2" - har-validator "~5.1.3" - http-signature "~1.2.0" - is-typedarray "~1.0.0" - isstream "~0.1.2" - json-stringify-safe "~5.0.1" - mime-types "~2.1.19" - oauth-sign "~0.9.0" - performance-now "^2.1.0" - qs "~6.5.2" - safe-buffer "^5.1.2" - tough-cookie "~2.5.0" - tunnel-agent "^0.6.0" - uuid "^3.3.2" +regexpp@^3.2.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/regexpp/-/regexpp-3.2.0.tgz#0425a2768d8f23bad70ca4b90461fa2f1213e1b2" + integrity sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg== require-directory@^2.1.1: version "2.1.1" resolved "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz" integrity sha1-jGStX9MNqxyXbiNE/+f3kqam30I= -require-from-string@^2.0.2: - version "2.0.2" - resolved "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz" - integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw== - -reserved-words@^0.1.2: - version "0.1.2" - resolved "https://registry.npmjs.org/reserved-words/-/reserved-words-0.1.2.tgz" - integrity sha1-AKCUD5jNUBrqqsMWQR2a3FKzGrE= - -resolve-alpn@^1.0.0: - version "1.2.1" - resolved "https://registry.yarnpkg.com/resolve-alpn/-/resolve-alpn-1.2.1.tgz#b7adbdac3546aaaec20b45e7d8265927072726f9" - integrity sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g== - resolve-cwd@^3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz" @@ -4703,16 +3552,7 @@ resolve.exports@^1.1.0: resolved "https://registry.yarnpkg.com/resolve.exports/-/resolve.exports-1.1.0.tgz#5ce842b94b05146c0e03076985d1d0e7e48c90c9" integrity sha512-J1l+Zxxp4XK3LUDZ9m60LRJF/mAe4z6a4xyabPHk7pvK5t35dACV32iIjJDFeWZFfZlO29w6SZ67knR0tHzJtQ== -resolve@^1.1.6, resolve@^1.20.0: - version "1.22.0" - resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.0.tgz#5e0b8c67c15df57a89bdbabe603a002f21731198" - integrity sha512-Hhtrw0nLeSrFQ7phPp4OOcVjLPIeMnRlr5mcnVuMe7M/7eBn98A3hmFRLoFo3DLZkivSYwhRUJTyPyWAk56WLw== - dependencies: - is-core-module "^2.8.1" - path-parse "^1.0.7" - supports-preserve-symlinks-flag "^1.0.0" - -resolve@^1.10.0, resolve@^1.13.1, resolve@^1.17.0: +resolve@^1.10.0: version "1.20.0" resolved "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz" integrity sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A== @@ -4720,30 +3560,29 @@ resolve@^1.10.0, resolve@^1.13.1, resolve@^1.17.0: is-core-module "^2.2.0" path-parse "^1.0.6" -responselike@^1.0.2: - version "1.0.2" - resolved "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz" - integrity sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec= +resolve@^1.20.0: + version "1.22.0" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.0.tgz#5e0b8c67c15df57a89bdbabe603a002f21731198" + integrity sha512-Hhtrw0nLeSrFQ7phPp4OOcVjLPIeMnRlr5mcnVuMe7M/7eBn98A3hmFRLoFo3DLZkivSYwhRUJTyPyWAk56WLw== dependencies: - lowercase-keys "^1.0.0" + is-core-module "^2.8.1" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" -responselike@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/responselike/-/responselike-2.0.0.tgz#26391bcc3174f750f9a79eacc40a12a5c42d7723" - integrity sha512-xH48u3FTB9VsZw7R+vvgaKeLKzT6jOogbQhEe/jewwnZgzPcnyWui2Av6JpoYZF/91uueC+lqhWqeURw5/qhCw== +resolve@^1.22.0: + version "1.22.1" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.1.tgz#27cb2ebb53f91abb49470a928bba7558066ac177" + integrity sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw== dependencies: - lowercase-keys "^2.0.0" + is-core-module "^2.9.0" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" reusify@^1.0.4: version "1.0.4" resolved "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz" integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== -rfc4648@^1.3.0: - version "1.5.1" - resolved "https://registry.yarnpkg.com/rfc4648/-/rfc4648-1.5.1.tgz#b0b16756e33d9de8c0c7833e94b28e627ec372a4" - integrity sha512-60e/YWs2/D3MV1ErdjhJHcmlgnyLUiG4X/14dgsfm9/zmCWLN16xI6YqJYSCd/OANM7bUNzJqPY5B8/02S9Ibw== - rimraf@^3.0.0, rimraf@^3.0.2: version "3.0.2" resolved "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz" @@ -4758,11 +3597,6 @@ run-parallel@^1.1.9: dependencies: queue-microtask "^1.2.2" -safe-buffer@^5.0.1, safe-buffer@^5.1.2: - version "5.2.1" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" - integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== - safe-buffer@~5.1.1: version "5.1.2" resolved "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz" @@ -4775,34 +3609,12 @@ safe-regex@^2.1.1: dependencies: regexp-tree "~0.1.1" -"safer-buffer@>= 2.1.2 < 3", safer-buffer@^2.0.2, safer-buffer@^2.1.0, safer-buffer@~2.1.0: - version "2.1.2" - resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" - integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== - -sax@1.2.1: - version "1.2.1" - resolved "https://registry.npmjs.org/sax/-/sax-1.2.1.tgz" - integrity sha1-e45lYZCyKOgaZq6nSEgNgozS03o= - -sax@>=0.6.0: - version "1.2.4" - resolved "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz" - integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw== - -saxes@^5.0.1: - version "5.0.1" - resolved "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz" - integrity sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw== - dependencies: - xmlchars "^2.2.0" - "semver@2 || 3 || 4 || 5": version "5.7.1" resolved "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz" integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== -semver@7.x, semver@^7.2.1, semver@^7.3.2, semver@^7.3.4, semver@^7.3.5: +semver@7.x, semver@^7.3.5: version "7.3.5" resolved "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz" integrity sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ== @@ -4814,6 +3626,13 @@ semver@^6.0.0, semver@^6.3.0: resolved "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz" integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== +semver@^7.3.7: + version "7.3.7" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.7.tgz#12c5b649afdbf9049707796e22a4028814ce523f" + integrity sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g== + dependencies: + lru-cache "^6.0.0" + shebang-command@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" @@ -4826,15 +3645,6 @@ shebang-regex@^3.0.0: resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== -shelljs@^0.8.2, shelljs@^0.8.5: - version "0.8.5" - resolved "https://registry.yarnpkg.com/shelljs/-/shelljs-0.8.5.tgz#de055408d8361bed66c669d2f000538ced8ee20c" - integrity sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow== - dependencies: - glob "^7.0.0" - interpret "^1.0.0" - rechoir "^0.6.2" - side-channel@^1.0.4: version "1.0.4" resolved "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz" @@ -4844,12 +3654,7 @@ side-channel@^1.0.4: get-intrinsic "^1.0.2" object-inspect "^1.9.0" -signal-exit@^3.0.2: - version "3.0.3" - resolved "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz" - integrity sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA== - -signal-exit@^3.0.3: +signal-exit@^3.0.3, signal-exit@^3.0.7: version "3.0.7" resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== @@ -4864,19 +3669,10 @@ slash@^3.0.0: resolved "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz" integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== -slice-ansi@^4.0.0: - version "4.0.0" - resolved "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz" - integrity sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ== - dependencies: - ansi-styles "^4.0.0" - astral-regex "^2.0.0" - is-fullwidth-code-point "^3.0.0" - -source-map-support@^0.5.6: - version "0.5.19" - resolved "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz" - integrity sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw== +source-map-support@0.5.13: + version "0.5.13" + resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.13.tgz#31b24a9c2e73c2de85066c0feb7d44767ed52932" + integrity sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w== dependencies: buffer-from "^1.0.0" source-map "^0.6.0" @@ -4886,16 +3682,11 @@ source-map@^0.5.0: resolved "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz" integrity sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w= -source-map@^0.6.0, source-map@^0.6.1, source-map@~0.6.1: +source-map@^0.6.0, source-map@^0.6.1: version "0.6.1" resolved "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz" integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== -source-map@^0.7.3: - version "0.7.3" - resolved "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz" - integrity sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ== - spdx-correct@^3.0.0: version "3.1.1" resolved "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz" @@ -4927,21 +3718,6 @@ sprintf-js@~1.0.2: resolved "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz" integrity sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw= -sshpk@^1.7.0: - version "1.17.0" - resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.17.0.tgz#578082d92d4fe612b13007496e543fa0fbcbe4c5" - integrity sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ== - dependencies: - asn1 "~0.2.3" - assert-plus "^1.0.0" - bcrypt-pbkdf "^1.0.0" - dashdash "^1.12.0" - ecc-jsbn "~0.1.1" - getpass "^0.1.1" - jsbn "~0.1.0" - safer-buffer "^2.0.2" - tweetnacl "~0.14.0" - stack-utils@^2.0.3: version "2.0.5" resolved "https://registry.yarnpkg.com/stack-utils/-/stack-utils-2.0.5.tgz#d25265fca995154659dbbfba3b49254778d2fdd5" @@ -4949,16 +3725,6 @@ stack-utils@^2.0.3: dependencies: escape-string-regexp "^2.0.0" -stealthy-require@^1.1.1: - version "1.1.1" - resolved "https://registry.npmjs.org/stealthy-require/-/stealthy-require-1.1.1.tgz" - integrity sha1-NbCYdbT/SfJqd35QmzCQoyJr8ks= - -stream-buffers@^3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/stream-buffers/-/stream-buffers-3.0.2.tgz#5249005a8d5c2d00b3a32e6e0a6ea209dc4f3521" - integrity sha512-DQi1h8VEBA/lURbSwFtEHnSTb9s2/pwLEaFuNhXwy1Dx3Sa0lOuYT2yNUr4/j2fs8oCAMANtrZ5OrPZtyVs3MQ== - string-length@^4.0.1: version "4.0.2" resolved "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz" @@ -4976,21 +3742,32 @@ string-width@^4.1.0, string-width@^4.2.0: is-fullwidth-code-point "^3.0.0" strip-ansi "^6.0.0" -string.prototype.trimend@^1.0.4: - version "1.0.4" - resolved "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.4.tgz" - integrity sha512-y9xCjw1P23Awk8EvTpcyL2NIr1j7wJ39f+k6lvRnSMz+mz9CGz9NYPelDk42kOz6+ql8xjfK8oYzy3jAP5QU5A== +string-width@^4.2.3: + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + +string.prototype.trimend@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.5.tgz#914a65baaab25fbdd4ee291ca7dde57e869cb8d0" + integrity sha512-I7RGvmjV4pJ7O3kdf+LXFpVfdNOxtCW/2C8f6jNiW4+PQchwxkCDzlk1/7p+Wl4bqFIZeF47qAHXLuHHWKAxog== dependencies: call-bind "^1.0.2" - define-properties "^1.1.3" + define-properties "^1.1.4" + es-abstract "^1.19.5" -string.prototype.trimstart@^1.0.4: - version "1.0.4" - resolved "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.4.tgz" - integrity sha512-jh6e984OBfvxS50tdY2nRZnoC5/mLFKOREQfw8t5yytkoUsJRNxvI/E39qu1sD0OtWI3OC0XgKSmcWwziwYuZw== +string.prototype.trimstart@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.5.tgz#5466d93ba58cfa2134839f81d7f42437e8c01fef" + integrity sha512-THx16TJCGlsN0o6dl2o6ncWUsdgnLRSA23rRE5pyGBw/mLr3Ej/R2LaqCtgP8VNMGZsvMWnf9ooZPyY2bHvUFg== dependencies: call-bind "^1.0.2" - define-properties "^1.1.3" + define-properties "^1.1.4" + es-abstract "^1.19.5" strip-ansi@^6.0.0: version "6.0.0" @@ -4999,6 +3776,13 @@ strip-ansi@^6.0.0: dependencies: ansi-regex "^5.0.0" +strip-ansi@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + strip-bom@^3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz" @@ -5014,6 +3798,13 @@ strip-final-newline@^2.0.0: resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-2.0.0.tgz#89b852fb2fcbe936f6f4b3187afb0a12c1ab58ad" integrity sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA== +strip-indent@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-3.0.0.tgz#c32e1cee940b6b3432c771bc2c54bcce73cd3001" + integrity sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ== + dependencies: + min-indent "^1.0.0" + strip-json-comments@^3.1.0, strip-json-comments@^3.1.1: version "3.1.1" resolved "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz" @@ -5058,47 +3849,6 @@ svg-element-attributes@^1.3.1: resolved "https://registry.npmjs.org/svg-element-attributes/-/svg-element-attributes-1.3.1.tgz" integrity sha512-Bh05dSOnJBf3miNMqpsormfNtfidA/GxQVakhtn0T4DECWKeXQRQUceYjJ+OxYiiLdGe4Jo9iFV8wICFapFeIA== -swagger-fluent@^5.0.3: - version "5.0.3" - resolved "https://registry.npmjs.org/swagger-fluent/-/swagger-fluent-5.0.3.tgz" - integrity sha512-i43ADMtPi7dxAN75Lw50SlncMB31FgaVwXqKioR8SWs+Yon2RbiLU1J1PGMXA4N8cSt9Vz5RHzaoKjz/+iW88g== - dependencies: - deepmerge "^4.2.2" - is-plain-object "^3.0.0" - request "^2.88.0" - -symbol-tree@^3.2.4: - version "3.2.4" - resolved "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz" - integrity sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw== - -table@^6.0.4: - version "6.3.1" - resolved "https://registry.npmjs.org/table/-/table-6.3.1.tgz" - integrity sha512-kNpMVSN4fj9KY4G6tNDVIT59uaG8ZELGQ+cmFSqivmWkCXJLd00VfRmtyHa8X7AeM75PQ/6/TtEtWjTDs1jXJw== - dependencies: - ajv "^8.0.1" - is-boolean-object "^1.1.0" - is-number-object "^1.0.4" - is-string "^1.0.5" - lodash.clonedeep "^4.5.0" - lodash.flatten "^4.4.0" - lodash.truncate "^4.4.2" - slice-ansi "^4.0.0" - string-width "^4.2.0" - -tar@^6.1.11: - version "6.1.11" - resolved "https://registry.yarnpkg.com/tar/-/tar-6.1.11.tgz#6760a38f003afa1b2ffd0ffe9e9abbd0eab3d621" - integrity sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA== - dependencies: - chownr "^2.0.0" - fs-minipass "^2.0.0" - minipass "^3.0.0" - minizlib "^2.1.1" - mkdirp "^1.0.3" - yallist "^4.0.0" - terminal-link@^2.0.0: version "2.1.1" resolved "https://registry.npmjs.org/terminal-link/-/terminal-link-2.1.1.tgz" @@ -5121,28 +3871,9 @@ text-table@^0.2.0: resolved "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz" integrity sha1-f17oI66AUgfACvLfSoTsP8+lcLQ= -throat@^6.0.1: - version "6.0.1" - resolved "https://registry.yarnpkg.com/throat/-/throat-6.0.1.tgz#d514fedad95740c12c2d7fc70ea863eb51ade375" - integrity sha512-8hmiGIJMDlwjg7dlJ4yKGLK8EsYqKgPWbG3b4wjJddKNwc7N7Dpn08Df4szr/sZdMVeOstrdYSsqzX6BYbcB+w== - -tmp-promise@^3.0.2: - version "3.0.3" - resolved "https://registry.yarnpkg.com/tmp-promise/-/tmp-promise-3.0.3.tgz#60a1a1cc98c988674fcbfd23b6e3367bdeac4ce7" - integrity sha512-RwM7MoPojPxsOBYnyd2hy0bxtIlVrihNs9pj5SUvY8Zz1sQcQG2tG1hSr8PDxfgEB8RNKDhqbIlroIarSNDNsQ== - dependencies: - tmp "^0.2.0" - -tmp@^0.2.0: - version "0.2.1" - resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.2.1.tgz#8457fc3037dcf4719c251367a1af6500ee1ccf14" - integrity sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ== - dependencies: - rimraf "^3.0.0" - -tmpl@1.0.x: +tmpl@1.0.5: version "1.0.5" - resolved "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz" + resolved "https://registry.yarnpkg.com/tmpl/-/tmpl-1.0.5.tgz#8683e0b902bb9c20c4f726e3c0b69f36518c07cc" integrity sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw== to-fast-properties@^2.0.0: @@ -5150,11 +3881,6 @@ to-fast-properties@^2.0.0: resolved "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz" integrity sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4= -to-readable-stream@^1.0.0: - version "1.0.0" - resolved "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz" - integrity sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q== - to-regex-range@^5.0.1: version "5.0.1" resolved "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz" @@ -5162,48 +3888,19 @@ to-regex-range@^5.0.1: dependencies: is-number "^7.0.0" -tough-cookie@^2.3.3, tough-cookie@~2.5.0: - version "2.5.0" - resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.5.0.tgz#cd9fb2a0aa1d5a12b473bd9fb96fa3dcff65ade2" - integrity sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g== - dependencies: - psl "^1.1.28" - punycode "^2.1.1" - -tough-cookie@^4.0.0: - version "4.0.0" - resolved "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.0.0.tgz" - integrity sha512-tHdtEpQCMrc1YLrMaqXXcj6AxhYi/xgit6mZu1+EDWUn+qhUf8wMQoFIy9NXuq23zAwtcB0t/MjACGR18pcRbg== - dependencies: - psl "^1.1.33" - punycode "^2.1.1" - universalify "^0.1.2" - -tr46@^2.0.2: - version "2.0.2" - resolved "https://registry.npmjs.org/tr46/-/tr46-2.0.2.tgz" - integrity sha512-3n1qG+/5kg+jrbTzwAykB5yRYtQCTqOGKq5U5PE3b0a1/mzo6snDhjGS0zJVJunO0NrT3Dg1MLy5TjWP/UJppg== - dependencies: - punycode "^2.1.1" - -tr46@~0.0.3: - version "0.0.3" - resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" - integrity sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o= - -ts-jest@^27.1.3: - version "27.1.3" - resolved "https://registry.yarnpkg.com/ts-jest/-/ts-jest-27.1.3.tgz#1f723e7e74027c4da92c0ffbd73287e8af2b2957" - integrity sha512-6Nlura7s6uM9BVUAoqLH7JHyMXjz8gluryjpPXxr3IxZdAXnU6FhjvVLHFtfd1vsE1p8zD1OJfskkc0jhTSnkA== +ts-jest@^28.0.7: + version "28.0.7" + resolved "https://registry.yarnpkg.com/ts-jest/-/ts-jest-28.0.7.tgz#e18757a9e44693da9980a79127e5df5a98b37ac6" + integrity sha512-wWXCSmTwBVmdvWrOpYhal79bDpioDy4rTT+0vyUnE3ZzM7LOAAGG9NXwzkEL/a516rQEgnMmS/WKP9jBPCVJyA== dependencies: bs-logger "0.x" fast-json-stable-stringify "2.x" - jest-util "^27.0.0" - json5 "2.x" + jest-util "^28.0.0" + json5 "^2.2.1" lodash.memoize "4.x" make-error "1.x" semver "7.x" - yargs-parser "20.x" + yargs-parser "^21.0.1" ts-node@10.4.0: version "10.4.0" @@ -5223,45 +3920,28 @@ ts-node@10.4.0: make-error "^1.1.1" yn "3.1.1" -tsconfig-paths@^3.9.0: - version "3.9.0" - resolved "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.9.0.tgz" - integrity sha512-dRcuzokWhajtZWkQsDVKbWyY+jgcLC5sqJhg2PSgf4ZkH2aHPvaOY8YWGhmjb68b5qqTfasSsDO9k7RUiEmZAw== +tsconfig-paths@^3.14.1: + version "3.14.1" + resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.14.1.tgz#ba0734599e8ea36c862798e920bcf163277b137a" + integrity sha512-fxDhWnFSLt3VuTwtvJt5fpwxBHg5AdKWMsgcPOOIilyjymcYVZoCQF8fvFRezCNfblEXmi+PcM1eYHeOAgXCOQ== dependencies: "@types/json5" "^0.0.29" json5 "^1.0.1" - minimist "^1.2.0" + minimist "^1.2.6" strip-bom "^3.0.0" -tslib@^1.8.1, tslib@^1.9.3: +tslib@^1.8.1: version "1.14.1" resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00" integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg== -tsutils@^3.17.1: +tsutils@^3.21.0: version "3.21.0" - resolved "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz" + resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-3.21.0.tgz#b48717d394cea6c1e096983eed58e9d61715b623" integrity sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA== dependencies: tslib "^1.8.1" -tunnel-agent@^0.6.0: - version "0.6.0" - resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd" - integrity sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0= - dependencies: - safe-buffer "^5.0.1" - -tunnel@0.0.6, tunnel@^0.0.6: - version "0.0.6" - resolved "https://registry.npmjs.org/tunnel/-/tunnel-0.0.6.tgz" - integrity sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg== - -tweetnacl@^0.14.3, tweetnacl@~0.14.0: - version "0.14.5" - resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-0.14.5.tgz#5ae68177f192d4456269d108afa93ff8743f4f64" - integrity sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q= - type-check@^0.4.0, type-check@~0.4.0: version "0.4.0" resolved "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz" @@ -5269,18 +3949,16 @@ type-check@^0.4.0, type-check@~0.4.0: dependencies: prelude-ls "^1.2.1" -type-check@~0.3.2: - version "0.3.2" - resolved "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz" - integrity sha1-WITKtRLPHTVeP7eE8wgEsrUg23I= - dependencies: - prelude-ls "~1.1.2" - type-detect@4.0.8: version "4.0.8" resolved "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz" integrity sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g== +type-fest@^0.20.2: + version "0.20.2" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4" + integrity sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ== + type-fest@^0.21.3: version "0.21.3" resolved "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz" @@ -5296,51 +3974,28 @@ type-fest@^0.8.1: resolved "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz" integrity sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA== -typedarray-to-buffer@^3.1.5: - version "3.1.5" - resolved "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz" - integrity sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q== - dependencies: - is-typedarray "^1.0.0" +typescript@4.3.5: + version "4.3.5" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.3.5.tgz#4d1c37cc16e893973c45a06886b7113234f119f4" + integrity sha512-DqQgihaQ9cUrskJo9kIyW/+g0Vxsk8cDtZ52a3NGh0YNTfpUSArXSohyUGnvbPazEPLu398C0UxmKSOrPumUzA== -typescript@4.1.3: - version "4.1.3" - resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.1.3.tgz#519d582bd94cba0cf8934c7d8e8467e473f53bb7" - integrity sha512-B3ZIOf1IKeH2ixgHhj6la6xdwR9QrLC5d1VKeCSY4tvkqhF2eqd9O7txNlS0PO3GrBAFIdr3L1ndNwteUbZLYg== - -unbox-primitive@^1.0.0: - version "1.0.1" - resolved "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.1.tgz" - integrity sha512-tZU/3NqK3dA5gpE1KtyiJUrEB0lxnGkMFHptJ7q6ewdZ8s12QrODwNbhIJStmJkd1QDXa1NRA8aF2A1zk/Ypyw== +unbox-primitive@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.0.2.tgz#29032021057d5e6cdbd08c5129c226dff8ed6f9e" + integrity sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw== dependencies: - function-bind "^1.1.1" - has-bigints "^1.0.1" - has-symbols "^1.0.2" + call-bind "^1.0.2" + has-bigints "^1.0.2" + has-symbols "^1.0.3" which-boxed-primitive "^1.0.2" -underscore@^1.9.1: - version "1.13.2" - resolved "https://registry.yarnpkg.com/underscore/-/underscore-1.13.2.tgz#276cea1e8b9722a8dbed0100a407dda572125881" - integrity sha512-ekY1NhRzq0B08g4bGuX4wd2jZx5GnKz6mKSqFL4nqBlfyMGiG10gDFhDTMEfYmDL6Jy0FUIZp7wiRB+0BP7J2g== - -unity-changeset@^1.6.0: - version "1.6.0" - resolved "https://registry.yarnpkg.com/unity-changeset/-/unity-changeset-1.6.0.tgz#b2a3cab1350b10669cb53341ce562f3b93f38ddf" - integrity sha512-ke90Zhr2C5q2SPvinDKTr1dwgAPI0R6kxcO7gHPgQXntbx/Ag8hR0xLwd9K6UR2ykGFmDnUVjdbWfYlkhX8tFQ== +update-browserslist-db@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/update-browserslist-db/-/update-browserslist-db-1.0.5.tgz#be06a5eedd62f107b7c19eb5bcefb194411abf38" + integrity sha512-dteFFpCyvuDdr9S/ff1ISkKt/9YZxKjI9WlRR99c180GaztJtRa/fn18FdxGVKVsnPY7/a/FDN68mcvUmP4U7Q== dependencies: - cac "^6.6.1" - jsdom "^16.4.0" - node-fetch "^2.6.1" - -universal-user-agent@^6.0.0: - version "6.0.0" - resolved "https://registry.npmjs.org/universal-user-agent/-/universal-user-agent-6.0.0.tgz" - integrity sha512-isyNax3wXoKaulPDZWHQqbmIx1k2tb9fb3GGDBRxCscfYV2Ch7WxPArBsFEG8s/safwXTT7H4QGhaIkTp9447w== - -universalify@^0.1.2: - version "0.1.2" - resolved "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz" - integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== + escalade "^3.1.1" + picocolors "^1.0.0" uri-js@^4.2.2: version "4.4.1" @@ -5349,54 +4004,19 @@ uri-js@^4.2.2: dependencies: punycode "^2.1.0" -url-join@^4.0.1: - version "4.0.1" - resolved "https://registry.npmjs.org/url-join/-/url-join-4.0.1.tgz" - integrity sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA== - -url-parse-lax@^3.0.0: - version "3.0.0" - resolved "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz" - integrity sha1-FrXK/Afb42dsGxmZF3gj1lA6yww= - dependencies: - prepend-http "^2.0.0" - -url@0.10.3: - version "0.10.3" - resolved "https://registry.npmjs.org/url/-/url-0.10.3.tgz" - integrity sha1-Ah5NnHcF8hu/N9A861h2dAJ3TGQ= - dependencies: - punycode "1.3.2" - querystring "0.2.0" - -uuid@3.3.2: - version "3.3.2" - resolved "https://registry.npmjs.org/uuid/-/uuid-3.3.2.tgz" - integrity sha512-yXJmeNaw3DnnKAOKJE51sL/ZaYfWJRl1pK9dr19YFCu0ObS231AB1/LbqTKRAQ5kw8A90rA6fr4riOUpTZvQZA== - -uuid@^3.3.2: - version "3.4.0" - resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.4.0.tgz#b23e4358afa8a202fe7a100af1f5f883f02007ee" - integrity sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A== - -uuid@^8.3.2: - version "8.3.2" - resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2" - integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg== - v8-compile-cache@^2.0.3: version "2.3.0" resolved "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz" integrity sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA== -v8-to-istanbul@^8.1.0: - version "8.1.1" - resolved "https://registry.yarnpkg.com/v8-to-istanbul/-/v8-to-istanbul-8.1.1.tgz#77b752fd3975e31bbcef938f85e9bd1c7a8d60ed" - integrity sha512-FGtKtv3xIpR6BYhvgH8MI/y78oT7d8Au3ww4QIxymrCtZEh5b8gCw2siywE+puhEmuWKDtmfrvF5UlB298ut3w== +v8-to-istanbul@^9.0.1: + version "9.0.1" + resolved "https://registry.yarnpkg.com/v8-to-istanbul/-/v8-to-istanbul-9.0.1.tgz#b6f994b0b5d4ef255e17a0d17dc444a9f5132fa4" + integrity sha512-74Y4LqY74kLE6IFyIjPtkSTWzUZmj8tdHT9Ii/26dvQ6K9Dl2NbEfj0XgU2sHCtKgt5VupqhlO/5aWuqS+IY1w== dependencies: + "@jridgewell/trace-mapping" "^0.3.12" "@types/istanbul-lib-coverage" "^2.0.1" convert-source-map "^1.6.0" - source-map "^0.7.3" validate-npm-package-license@^3.0.1: version "3.0.4" @@ -5406,79 +4026,12 @@ validate-npm-package-license@^3.0.1: spdx-correct "^3.0.0" spdx-expression-parse "^3.0.0" -verror@1.10.0: - version "1.10.0" - resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400" - integrity sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA= - dependencies: - assert-plus "^1.0.0" - core-util-is "1.0.2" - extsprintf "^1.2.0" - -w3c-hr-time@^1.0.2: - version "1.0.2" - resolved "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz" - integrity sha512-z8P5DvDNjKDoFIHK7q8r8lackT6l+jo/Ye3HOle7l9nICP9lf1Ci25fy9vHd0JOWewkIFzXIEig3TdKT7JQ5fQ== - dependencies: - browser-process-hrtime "^1.0.0" - -w3c-xmlserializer@^2.0.0: - version "2.0.0" - resolved "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-2.0.0.tgz" - integrity sha512-4tzD0mF8iSiMiNs30BiLO3EpfGLZUT2MSX/G+o7ZywDzliWQ3OPtTZ0PTC3B3ca1UAf4cJMHB+2Bf56EriJuRA== - dependencies: - xml-name-validator "^3.0.0" - -walker@^1.0.7: - version "1.0.7" - resolved "https://registry.npmjs.org/walker/-/walker-1.0.7.tgz" - integrity sha1-L3+bj9ENZ3JisYqITijRlhjgKPs= - dependencies: - makeerror "1.0.x" - -webidl-conversions@^3.0.0: - version "3.0.1" - resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871" - integrity sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE= - -webidl-conversions@^5.0.0: - version "5.0.0" - resolved "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz" - integrity sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA== - -webidl-conversions@^6.1.0: - version "6.1.0" - resolved "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz" - integrity sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w== - -whatwg-encoding@^1.0.5: - version "1.0.5" - resolved "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz" - integrity sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw== - dependencies: - iconv-lite "0.4.24" - -whatwg-mimetype@^2.3.0: - version "2.3.0" - resolved "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz" - integrity sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g== - -whatwg-url@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d" - integrity sha1-lmRU6HZUYuN2RNNib2dCzotwll0= - dependencies: - tr46 "~0.0.3" - webidl-conversions "^3.0.0" - -whatwg-url@^8.0.0, whatwg-url@^8.5.0: - version "8.5.0" - resolved "https://registry.npmjs.org/whatwg-url/-/whatwg-url-8.5.0.tgz" - integrity sha512-fy+R77xWv0AiqfLl4nuGUlQ3/6b5uNfQ4WAbGQVMYshCTCCPK9psC1nWh3XHuxGVCtlcDDQPQW1csmmIQo+fwg== +walker@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/walker/-/walker-1.0.8.tgz#bd498db477afe573dc04185f011d3ab8a8d7653f" + integrity sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ== dependencies: - lodash "^4.7.0" - tr46 "^2.0.2" - webidl-conversions "^6.1.0" + makeerror "1.0.12" which-boxed-primitive@^1.0.2: version "1.0.2" @@ -5498,7 +4051,7 @@ which@^2.0.1: dependencies: isexe "^2.0.0" -word-wrap@^1.2.3, word-wrap@~1.2.3: +word-wrap@^1.2.3: version "1.2.3" resolved "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz" integrity sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ== @@ -5517,55 +4070,13 @@ wrappy@1: resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= -write-file-atomic@^3.0.0: - version "3.0.3" - resolved "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz" - integrity sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q== +write-file-atomic@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-4.0.1.tgz#9faa33a964c1c85ff6f849b80b42a88c2c537c8f" + integrity sha512-nSKUxgAbyioruk6hU87QzVbY279oYT6uiwgDoujth2ju4mJ+TZau7SQBhtbTmUyuNYTuXnSyRn66FV0+eCgcrQ== dependencies: imurmurhash "^0.1.4" - is-typedarray "^1.0.0" - signal-exit "^3.0.2" - typedarray-to-buffer "^3.1.5" - -ws@^6.1.0: - version "6.2.2" - resolved "https://registry.npmjs.org/ws/-/ws-6.2.2.tgz" - integrity sha512-zmhltoSR8u1cnDsD43TX59mzoMZsLKqUweyYBAIvTngR3shc0W6aOZylZmq/7hqyVxPdi+5Ud2QInblgyE72fw== - dependencies: - async-limiter "~1.0.0" - -ws@^7.2.3, ws@^7.4.4: - version "7.5.6" - resolved "https://registry.npmjs.org/ws/-/ws-7.5.6.tgz" - integrity sha512-6GLgCqo2cy2A2rjCNFlxQS6ZljG/coZfZXclldI8FB/1G3CCI36Zd8xy2HrFVACi8tfk5XrgLQEk+P0Tnz9UcA== - -ws@^7.3.1, ws@^7.4.6: - version "7.5.7" - resolved "https://registry.yarnpkg.com/ws/-/ws-7.5.7.tgz#9e0ac77ee50af70d58326ecff7e85eb3fa375e67" - integrity sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A== - -xml-name-validator@^3.0.0: - version "3.0.0" - resolved "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz" - integrity sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw== - -xml2js@0.4.19: - version "0.4.19" - resolved "https://registry.npmjs.org/xml2js/-/xml2js-0.4.19.tgz" - integrity sha512-esZnJZJOiJR9wWKMyuvSE1y6Dq5LCuJanqhxslH2bxM6duahNZ+HMpCLhBQGZkbX6xRf8x1Y2eJlgt2q3qo49Q== - dependencies: - sax ">=0.6.0" - xmlbuilder "~9.0.1" - -xmlbuilder@~9.0.1: - version "9.0.7" - resolved "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-9.0.7.tgz" - integrity sha1-Ey7mPS7FVlxVfiD0wi35rKaGsQ0= - -xmlchars@^2.2.0: - version "2.2.0" - resolved "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz" - integrity sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw== + signal-exit "^3.0.7" y18n@^5.0.5: version "5.0.8" @@ -5577,35 +4088,30 @@ yallist@^4.0.0: resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== -yaml@^1.10.2: - version "1.10.2" - resolved "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz" - integrity sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg== - -yargs-parser@20.x: - version "20.2.7" - resolved "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz" - integrity sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw== - -yargs-parser@^20.2.2: - version "20.2.9" - resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee" - integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w== +yargs-parser@^21.0.0, yargs-parser@^21.0.1: + version "21.1.1" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-21.1.1.tgz#9096bceebf990d21bb31fa9516e0ede294a77d35" + integrity sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw== -yargs@^16.2.0: - version "16.2.0" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-16.2.0.tgz#1c82bf0f6b6a66eafce7ef30e376f49a12477f66" - integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw== +yargs@^17.3.1: + version "17.5.1" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.5.1.tgz#e109900cab6fcb7fd44b1d8249166feb0b36e58e" + integrity sha512-t6YAJcxDkNX7NFYiVtKvWUz8l+PaKTLiL63mJYWR2GnHq2gjEWISzsLp9wg3aY36dY1j+gfIEL3pIF+XlJJfbA== dependencies: cliui "^7.0.2" escalade "^3.1.1" get-caller-file "^2.0.5" require-directory "^2.1.1" - string-width "^4.2.0" + string-width "^4.2.3" y18n "^5.0.5" - yargs-parser "^20.2.2" + yargs-parser "^21.0.0" yn@3.1.1: version "3.1.1" resolved "https://registry.yarnpkg.com/yn/-/yn-3.1.1.tgz#1e87401a09d767c1d5eab26a6e4c185182d2eb50" integrity sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q== + +yocto-queue@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" + integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==

$func_code$count_code
$name$count