From 22adebd5ab2bd14fe620e3a1aeb3ddeed161794b Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 29 May 2024 19:11:39 +0200 Subject: [PATCH 01/49] update pre-commits and skip unittest method --- .pre-commit-config.yaml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 8faab34f..7cbe1199 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.4.5 + rev: v0.4.6 hooks: - id: ruff @@ -41,11 +41,11 @@ repos: - id: pyproject-fmt - repo: https://github.com/codespell-project/codespell - rev: v2.2.6 + rev: v2.3.0 hooks: - id: codespell args: - - --ignore-words-list=degreeE,degreee,varn,poit,uint,sur,herat,claus,tung,messsages + - --ignore-words-list=degreeE,degreee,varn,poit,uint,sur,herat,claus,tung,messsages,assertin exclude: > (?x)^( .*\.xml| From 7422292b1b981d09bcd5c7258c3158193a932d9b Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 29 May 2024 19:11:48 +0200 Subject: [PATCH 02/49] split cmd and remove $ --- README.md | 24 +++++++++++++----------- 1 file changed, 13 insertions(+), 11 deletions(-) diff --git a/README.md b/README.md index a01b1ff3..cdbd4edf 100644 --- a/README.md +++ b/README.md @@ -204,9 +204,10 @@ optional arguments: ### Check a local file against CF 1.6 ``` -$ compliance-checker --test=cf:1.6 compliance_checker/tests/data/examples/hycom_global.nc - +compliance-checker --test=cf:1.6 compliance_checker/tests/data/examples/hycom_global.nc +``` +``` -------------------------------------------------------------------------------- IOOS Compliance Checker Report cf:1.6 check @@ -247,7 +248,7 @@ contains valid packing: are not of type float or double. The remote dataset url is taken from the Data URL section of an OPeNDAP endpoint. ```shell -$ compliance-checker --test=acdd:1.3 "http://sos.maracoos.org/stable/dodsC/hrecos/stationHRMARPH-agg.ncml" +compliance-checker --test=acdd:1.3 "http://sos.maracoos.org/stable/dodsC/hrecos/stationHRMARPH-agg.ncml" ``` ### Checking against remote ERDDAP Datasets @@ -255,7 +256,7 @@ $ compliance-checker --test=acdd:1.3 "http://sos.maracoos.org/stable/dodsC/hreco ERDDAP datasets are becoming a popular way to access data. Supply an ERDDAP `TableDAP` or `GridDAP` URL to the checker: ```shell -$ compliance-checker --test ioos:1.2 "https://pae-paha.pacioos.hawaii.edu/erddap/griddap/pibhmc_bathy_60m_guam" +compliance-checker --test ioos:1.2 "https://pae-paha.pacioos.hawaii.edu/erddap/griddap/pibhmc_bathy_60m_guam" ``` Ensure to supply the URL *without* the format extension at the end (no `.nc`, `.ncCF`, etc.). @@ -276,37 +277,37 @@ Some examples of ERDDAP datasets: ### Write results to text file ```shell -$ compliance-checker --test=acdd:1.3 --format=text --output=/tmp/report.txt compliance_checker/tests/data/examples/hycom_global.nc +compliance-checker --test=acdd:1.3 --format=text --output=/tmp/report.txt compliance_checker/tests/data/examples/hycom_global.nc ``` ### Write results to JSON file ```shell -$ compliance-checker --test=acdd:1.3 --format=json --output=/tmp/report.json compliance_checker/tests/data/examples/hycom_global.nc +compliance-checker --test=acdd:1.3 --format=json --output=/tmp/report.json compliance_checker/tests/data/examples/hycom_global.nc ``` ### Write results to HTML file ```shell -$ compliance-checker --test=acdd:1.3 --format=html --output=/tmp/report.html compliance_checker/tests/data/examples/hycom_global.nc +compliance-checker --test=acdd:1.3 --format=html --output=/tmp/report.html compliance_checker/tests/data/examples/hycom_global.nc ``` ### Output text from multiple input files to one output file ``` -$ compliance-checker --test=cf:1.6 --format text --output=/tmp/combined_output.txt compliance_checker/tests/data/examples/hycom_global.nc compliance_checker/tests/data/examples/ww3.nc +compliance-checker --test=cf:1.6 --format text --output=/tmp/combined_output.txt compliance_checker/tests/data/examples/hycom_global.nc compliance_checker/tests/data/examples/ww3.nc ``` ### Output html and text files from multiple input files (part 1) In this case you'll get 2 files ```/tmp/combined_output.txt``` and ```/tmp/combined_output.html``` that contain cf check results for both input files because you only specified 1 output filename. ``` -$ compliance-checker --test=cf:1.6 --format text --format html --output=/tmp/combined_output.txt compliance_checker/tests/data/examples/hycom_global.nc compliance_checker/tests/data/examples/ww3.nc +compliance-checker --test=cf:1.6 --format text --format html --output=/tmp/combined_output.txt compliance_checker/tests/data/examples/hycom_global.nc compliance_checker/tests/data/examples/ww3.nc ``` ### Output html and text files from multiple input files (part 2) In this case you'll get 4 files ```/tmp/hycom.txt```, ```/tmp/hycom.html```, ```/tmp/ww3.txt```, and ```/tmp/ww3.html``` that contain cf check results because you specified as many output filenames as input filenames. ``` -$ compliance-checker --test=cf:1.6 --format text --format html --output=/tmp/hycom.txt --output=/tmp/ww3.txt compliance_checker/tests/data/examples/hycom_global.nc compliance_checker/tests/data/examples/ww3.nc +compliance-checker --test=cf:1.6 --format text --format html --output=/tmp/hycom.txt --output=/tmp/ww3.txt compliance_checker/tests/data/examples/hycom_global.nc compliance_checker/tests/data/examples/ww3.nc ``` ### Download a particular CF standard names table for use in the test @@ -317,7 +318,8 @@ During the CF test, if a file has a particular version of the cf standard name t try to download the specified version. If it fails, it will fall back to packaged version. ``` -$ compliance-checker -d 35 +compliance-checker -d 35 +``` Downloading cf-standard-names table version 35 from: http://cfconventions.org/Data/cf-standard-names/35/src/cf-standard-name-table.xml ``` From 631ed066a055f681c4163e7a7e574494790b5327 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 20:47:52 +0000 Subject: [PATCH 03/49] [pre-commit.ci] pre-commit autoupdate MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit updates: - [github.com/astral-sh/ruff-pre-commit: v0.4.6 → v0.4.7](https://github.com/astral-sh/ruff-pre-commit/compare/v0.4.6...v0.4.7) --- .pre-commit-config.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 7cbe1199..64ae8fbe 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.4.6 + rev: v0.4.7 hooks: - id: ruff From 059d73d5e641af01796e0c39293c1d9a2f617c97 Mon Sep 17 00:00:00 2001 From: sol1105 Date: Wed, 12 Jun 2024 15:18:45 +0200 Subject: [PATCH 04/49] Adding inputs (i.e. options incl. value) - adding command line option -I/--input - adding inputs to CheckSuite, ComplianceChecker and BaseCheck --- cchecker.py | 47 ++++++++++++++++++++++++++++++++++++ compliance_checker/base.py | 6 ++++- compliance_checker/runner.py | 3 ++- compliance_checker/suite.py | 6 +++-- 4 files changed, 58 insertions(+), 4 deletions(-) diff --git a/cchecker.py b/cchecker.py index 4e5ed5d4..ae9eb8c0 100755 --- a/cchecker.py +++ b/cchecker.py @@ -40,6 +40,30 @@ def parse_options(opts): return options_dict +def parse_inputs(inputs): + """ + Helper function to parse possible inputs. Splits input after the first + and second colon to split into checker/key/value triplets. + + :param inputs: Iterable of strings with inputs + :rtype: dict + :return: Dictionary of dictionaries, with first level keys as checker + type (i.e. "cf", "acdd") and second level keys as options. + """ + inputs_dict = {} + for input_str in inputs: + try: + checker_type, checker_opt, checker_val = input_str.split(":", 2) + except ValueError: + warnings.warn(f"Could not split input {input_str}, ignoring", stacklevel=2) + else: + try: + inputs_dict[checker_type].update({checker_opt: checker_val}) + except KeyError: + inputs_dict[checker_type] = {checker_opt: checker_val} + return inputs_dict + + def main(): # Load all available checker classes check_suite = CheckSuite() @@ -185,6 +209,26 @@ def main(): ), ) + parser.add_argument( + "-I", + "--input", + default=[], + action="append", + help=dedent( + """ + Additional input options to be passed to the + checkers. Multiple input options can be specified + via multiple invocations of this switch. + Input options should be prefixed with a the + checker name followed by the input name and the value , e.g. + '::' + + For now this switch exists to support more complex command line options + for plugins. + """, + ), + ) + parser.add_argument( "-V", "--version", @@ -236,6 +280,7 @@ def main(): sys.exit(0) options_dict = parse_options(args.option) if args.option else defaultdict(set) + inputs_dict = parse_inputs(args.input) if args.input else {} if args.describe_checks: error_stat = 0 @@ -306,6 +351,7 @@ def main(): args.output[0], args.format or ["text"], options=options_dict, + inputs=inputs_dict, ) return_values.append(return_value) had_errors.append(errors) @@ -326,6 +372,7 @@ def main(): output, args.format or ["text"], options=options_dict, + inputs=inputs_dict, ) return_values.append(return_value) had_errors.append(errors) diff --git a/compliance_checker/base.py b/compliance_checker/base.py index 4cf5e751..1d59bc4b 100644 --- a/compliance_checker/base.py +++ b/compliance_checker/base.py @@ -154,12 +154,16 @@ def setup(self, ds): Automatically run when running a CheckSuite. Define this method in your Checker class. """ - def __init__(self, options=None): + def __init__(self, options=None, inputs=None): self._defined_results = defaultdict(lambda: defaultdict(dict)) if options is None: self.options = set() else: self.options = options + if inputs is None: + self.inputs = {} + else: + self.inputs = inputs def get_test_ctx(self, severity, name, variable=None): """ diff --git a/compliance_checker/runner.py b/compliance_checker/runner.py index 536888f5..f8df9436 100644 --- a/compliance_checker/runner.py +++ b/compliance_checker/runner.py @@ -42,6 +42,7 @@ def run_checker( output_filename="-", output_format="text", options=None, + inputs=None, ): """ Static check runner. @@ -58,7 +59,7 @@ def run_checker( @returns If the tests failed (based on the criteria) """ all_groups = [] - cs = CheckSuite(options=options or {}) + cs = CheckSuite(options=options or {}, inputs=inputs or {}) # using OrderedDict is important here to preserve the order # of multiple datasets which may be passed in score_dict = OrderedDict() diff --git a/compliance_checker/suite.py b/compliance_checker/suite.py index a548aae7..4a6c9423 100644 --- a/compliance_checker/suite.py +++ b/compliance_checker/suite.py @@ -62,9 +62,10 @@ class CheckSuite: ) # Base dict of checker names to BaseCheck derived types, override this in your CheckSuite implementation templates_root = "compliance_checker" # modify to load alternative Jinja2 templates - def __init__(self, options=None): + def __init__(self, options=None, inputs=None): self.col_width = 40 self.options = options or {} + self.inputs = inputs or {} @classmethod def _get_generator_plugins(cls): @@ -402,10 +403,11 @@ def run_all(self, ds, checker_names, include_checks=None, skip_checks=None): # version baked in checker_type_name = checker_name.split(":")[0] checker_opts = self.options.get(checker_type_name, set()) + checker_inpts = self.inputs.get(checker_type_name, {}) # instantiate a Checker object try: - checker = checker_class(options=checker_opts) + checker = checker_class(options=checker_opts, inputs=checker_inpts) # hacky fix for no options in constructor except TypeError: checker = checker_class() From 5870e4d5c3cfda0a9c23cf580dcd8f8e037fd3c7 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Tue, 18 Jun 2024 08:54:57 +0200 Subject: [PATCH 05/49] We no longer need this part, np2 was released. --- .github/workflows/default-tests.yml | 12 +----------- 1 file changed, 1 insertion(+), 11 deletions(-) diff --git a/.github/workflows/default-tests.yml b/.github/workflows/default-tests.yml index ff039fe9..08ee54c6 100644 --- a/.github/workflows/default-tests.yml +++ b/.github/workflows/default-tests.yml @@ -9,13 +9,8 @@ jobs: runs-on: ${{ matrix.os }} strategy: matrix: - python-version: ["3.8", "3.9", "3.10", "3.11", "3.12"] + python-version: [ "3.8", "3.9", "3.10", "3.11", "3.12" ] os: [windows-latest, ubuntu-latest, macos-latest] - experimental: [false] - include: - - python-version: "3.12" - os: "ubuntu-latest" - experimental: true fail-fast: false defaults: run: @@ -35,11 +30,6 @@ jobs: --file test_requirements.txt --channel conda-forge - - name: Install unstable dependencies - if: matrix.experimental == true - run: | - micromamba install conda-forge/label/numpy_dev::numpy - - name: Install compliance-checker run: | python -m pip install -e . --no-deps --force-reinstall From e86d8865abffb4d9f5a7662947bd437be9aebc33 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 20 Jun 2024 18:33:06 +0200 Subject: [PATCH 06/49] remove duplicated function --- compliance_checker/cf/cf_1_6.py | 4 ++-- compliance_checker/cf/util.py | 13 ++----------- compliance_checker/tests/test_cf.py | 5 ++--- 3 files changed, 6 insertions(+), 16 deletions(-) diff --git a/compliance_checker/cf/cf_1_6.py b/compliance_checker/cf/cf_1_6.py index d7fa6906..2bf0f74a 100644 --- a/compliance_checker/cf/cf_1_6.py +++ b/compliance_checker/cf/cf_1_6.py @@ -925,7 +925,7 @@ def _check_valid_standard_units(self, ds, variable_name): # IMPLEMENTATION CONFORMANCE 4.4 REQUIRED 1/2 elif standard_name == "time": valid_standard_units.assert_true( - util.units_convertible(units, "seconds since 1970-01-01"), + cfutil.units_convertible(units, "seconds since 1970-01-01"), "time must be in a valid units format since " f"not {units}", ) @@ -1694,7 +1694,7 @@ def check_dimensional_vertical_coordinate( "there is no default", ) - if not util.units_convertible("bar", units): + if not cfutil.units_convertible("bar", units): valid_vertical_coord.assert_true( positive in ("up", "down"), f"{name}: vertical coordinates not defining pressure must include " diff --git a/compliance_checker/cf/util.py b/compliance_checker/cf/util.py index 591a7b3e..9443eb33 100644 --- a/compliance_checker/cf/util.py +++ b/compliance_checker/cf/util.py @@ -9,6 +9,8 @@ from lxml import etree from netCDF4 import Dataset +from compliance_checker.cfutil import units_convertible + # copied from paegan # paegan may depend on these later _possiblet = { @@ -325,17 +327,6 @@ def units_known(units): return True -def units_convertible(units1, units2, reftimeistime=True): - """Return True if a Unit representing the string units1 can be converted - to a Unit representing the string units2, else False.""" - try: - u1 = Unit(units1) - u2 = Unit(units2) - except ValueError: - return False - return u1.is_convertible(u2) - - def units_temporal(units): try: u = Unit(units) diff --git a/compliance_checker/tests/test_cf.py b/compliance_checker/tests/test_cf.py index 5e13bb7b..3db25d2d 100644 --- a/compliance_checker/tests/test_cf.py +++ b/compliance_checker/tests/test_cf.py @@ -29,7 +29,6 @@ download_cf_standard_name_table, is_time_variable, is_vertical_coordinate, - units_convertible, units_temporal, ) from compliance_checker.suite import CheckSuite @@ -1903,8 +1902,8 @@ def test_check_cell_methods(self): # -------------------------------------------------------------------------------- def test_temporal_unit_conversion(self): - self.assertTrue(units_convertible("hours", "seconds")) - self.assertFalse(units_convertible("hours", "hours since 2000-01-01")) + self.assertTrue(cfutil.units_convertible("hours", "seconds")) + self.assertFalse(cfutil.units_convertible("hours", "hours since 2000-01-01")) def test_units_temporal(self): self.assertTrue(units_temporal("hours since 2000-01-01")) From 69fe8ab301dee8e69b6fbc5714d509082b4a73b1 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 1 Jul 2024 21:53:19 +0000 Subject: [PATCH 07/49] [pre-commit.ci] pre-commit autoupdate MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit updates: - [github.com/astral-sh/ruff-pre-commit: v0.4.7 → v0.5.0](https://github.com/astral-sh/ruff-pre-commit/compare/v0.4.7...v0.5.0) --- .pre-commit-config.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 64ae8fbe..3a3ecd05 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.4.7 + rev: v0.5.0 hooks: - id: ruff From c46b541dbff9ce7f6680b2bad3c0223f94323c02 Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Thu, 11 Jul 2024 11:54:31 -0400 Subject: [PATCH 08/49] Add missing CF 1.9 checker to pyproject.toml --- pyproject.toml | 1 + 1 file changed, 1 insertion(+) diff --git a/pyproject.toml b/pyproject.toml index 94920af3..9bb30233 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -48,6 +48,7 @@ entry-points."compliance_checker.suites"."acdd-1.3" = "compliance_checker.acdd:A entry-points."compliance_checker.suites"."cf-1.6" = "compliance_checker.cf.cf:CF1_6Check" entry-points."compliance_checker.suites"."cf-1.7" = "compliance_checker.cf.cf:CF1_7Check" entry-points."compliance_checker.suites"."cf-1.8" = "compliance_checker.cf.cf:CF1_8Check" +entry-points."compliance_checker.suites"."cf-1.9" = "compliance_checker.cf.cf:CF1_9Check" entry-points."compliance_checker.suites"."ioos-0.1" = "compliance_checker.ioos:IOOS0_1Check" entry-points."compliance_checker.suites"."ioos-1.1" = "compliance_checker.ioos:IOOS1_1Check" entry-points."compliance_checker.suites"."ioos-1.2" = "compliance_checker.ioos:IOOS1_2Check" From ee8b8801073b412896074f723bf14bee00e83f81 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 11 Jul 2024 19:56:31 +0200 Subject: [PATCH 09/49] Update the CF version in the README --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index cdbd4edf..93843783 100644 --- a/README.md +++ b/README.md @@ -22,6 +22,7 @@ It currently supports the following sources and standards: | Standard | Source | .nc/OPeNDAP/.cdl | SOS | | ---------------------------------------------------------------------------------------------------- | ----------- | ------ | ------------------------------- | | [ACDD (1.1, 1.3)](https://wiki.esipfed.org/Attribute_Convention_for_Data_Discovery_1-3) | Built-in | X | - | +| [CF (1.9)](http://cfconventions.org/Data/cf-conventions/cf-conventions-1.9/cf-conventions.html) | Built-in | X | - | | [CF (1.8)](http://cfconventions.org/Data/cf-conventions/cf-conventions-1.8/cf-conventions.html) | Built-in | X | - | | [CF (1.7)](http://cfconventions.org/Data/cf-conventions/cf-conventions-1.7/cf-conventions.html) | Built-in | X | - | | [CF (1.6)](http://cfconventions.org/cf-conventions/v1.6.0/cf-conventions.html) | Built-in | X | - | From d5993f761b4591e8344020c301b09245d0c9049a Mon Sep 17 00:00:00 2001 From: sol1105 Date: Fri, 12 Jul 2024 07:16:03 +0200 Subject: [PATCH 10/49] Using option command line argument instead of defining separate input option --- cchecker.py | 70 ++++++----------------------- compliance_checker/base.py | 8 +--- compliance_checker/runner.py | 3 +- compliance_checker/suite.py | 8 ++-- compliance_checker/tests/test_cf.py | 2 +- 5 files changed, 21 insertions(+), 70 deletions(-) diff --git a/cchecker.py b/cchecker.py index ae9eb8c0..1327a3b2 100755 --- a/cchecker.py +++ b/cchecker.py @@ -22,48 +22,28 @@ def _print_checker_name_header(checker_str): def parse_options(opts): """ - Helper function to parse possible options. Splits option after the first - colon to split into key/value pairs. + Helper function to parse possible options. Splits option into key/value + pairs and optionally a value for the checker option. The separator + is a colon. :param opts: Iterable of strings with options :rtype: dict - :return: Dictionary with keys as checker type (i.e. "cf", "acdd") + :return: Dictionary with keys as checker type (i.e. "cf", "acdd"). + Each value is a dictionary where keys are checker options and values + are checker option values or None if not provided. """ - options_dict = defaultdict(set) + options_dict = defaultdict(dict) for opt_str in opts: try: - checker_type, checker_opt = opt_str.split(":", 1) + checker_type, checker_opt, *checker_val = opt_str.split(":", 2) + checker_val = checker_val[0] if checker_val else None except ValueError: warnings.warn(f"Could not split option {opt_str}, ignoring", stacklevel=2) else: - options_dict[checker_type].add(checker_opt) + options_dict[checker_type][checker_opt] = checker_val return options_dict -def parse_inputs(inputs): - """ - Helper function to parse possible inputs. Splits input after the first - and second colon to split into checker/key/value triplets. - - :param inputs: Iterable of strings with inputs - :rtype: dict - :return: Dictionary of dictionaries, with first level keys as checker - type (i.e. "cf", "acdd") and second level keys as options. - """ - inputs_dict = {} - for input_str in inputs: - try: - checker_type, checker_opt, checker_val = input_str.split(":", 2) - except ValueError: - warnings.warn(f"Could not split input {input_str}, ignoring", stacklevel=2) - else: - try: - inputs_dict[checker_type].update({checker_opt: checker_val}) - except KeyError: - inputs_dict[checker_type] = {checker_opt: checker_val} - return inputs_dict - - def main(): # Load all available checker classes check_suite = CheckSuite() @@ -198,8 +178,9 @@ def main(): checkers. Multiple options can be specified via multiple invocations of this switch. Options should be prefixed with a the - checker name followed by the option, e.g. - ':' + checker name followed by the option, + potentially followed by a value, e.g. + ':[:]' Available options: 'cf:enable_appendix_a_checks' - Allow check @@ -209,26 +190,6 @@ def main(): ), ) - parser.add_argument( - "-I", - "--input", - default=[], - action="append", - help=dedent( - """ - Additional input options to be passed to the - checkers. Multiple input options can be specified - via multiple invocations of this switch. - Input options should be prefixed with a the - checker name followed by the input name and the value , e.g. - '::' - - For now this switch exists to support more complex command line options - for plugins. - """, - ), - ) - parser.add_argument( "-V", "--version", @@ -279,8 +240,7 @@ def main(): print(f"IOOS compliance checker version {__version__}") sys.exit(0) - options_dict = parse_options(args.option) if args.option else defaultdict(set) - inputs_dict = parse_inputs(args.input) if args.input else {} + options_dict = parse_options(args.option) if args.option else defaultdict(dict) if args.describe_checks: error_stat = 0 @@ -351,7 +311,6 @@ def main(): args.output[0], args.format or ["text"], options=options_dict, - inputs=inputs_dict, ) return_values.append(return_value) had_errors.append(errors) @@ -372,7 +331,6 @@ def main(): output, args.format or ["text"], options=options_dict, - inputs=inputs_dict, ) return_values.append(return_value) had_errors.append(errors) diff --git a/compliance_checker/base.py b/compliance_checker/base.py index 1d59bc4b..0f276754 100644 --- a/compliance_checker/base.py +++ b/compliance_checker/base.py @@ -154,16 +154,12 @@ def setup(self, ds): Automatically run when running a CheckSuite. Define this method in your Checker class. """ - def __init__(self, options=None, inputs=None): + def __init__(self, options=None): self._defined_results = defaultdict(lambda: defaultdict(dict)) if options is None: - self.options = set() + self.options = {} else: self.options = options - if inputs is None: - self.inputs = {} - else: - self.inputs = inputs def get_test_ctx(self, severity, name, variable=None): """ diff --git a/compliance_checker/runner.py b/compliance_checker/runner.py index f8df9436..536888f5 100644 --- a/compliance_checker/runner.py +++ b/compliance_checker/runner.py @@ -42,7 +42,6 @@ def run_checker( output_filename="-", output_format="text", options=None, - inputs=None, ): """ Static check runner. @@ -59,7 +58,7 @@ def run_checker( @returns If the tests failed (based on the criteria) """ all_groups = [] - cs = CheckSuite(options=options or {}, inputs=inputs or {}) + cs = CheckSuite(options=options or {}) # using OrderedDict is important here to preserve the order # of multiple datasets which may be passed in score_dict = OrderedDict() diff --git a/compliance_checker/suite.py b/compliance_checker/suite.py index 4a6c9423..6b534f5f 100644 --- a/compliance_checker/suite.py +++ b/compliance_checker/suite.py @@ -62,10 +62,9 @@ class CheckSuite: ) # Base dict of checker names to BaseCheck derived types, override this in your CheckSuite implementation templates_root = "compliance_checker" # modify to load alternative Jinja2 templates - def __init__(self, options=None, inputs=None): + def __init__(self, options=None): self.col_width = 40 self.options = options or {} - self.inputs = inputs or {} @classmethod def _get_generator_plugins(cls): @@ -402,12 +401,11 @@ def run_all(self, ds, checker_names, include_checks=None, skip_checks=None): # use some kind of checker object with checker type and # version baked in checker_type_name = checker_name.split(":")[0] - checker_opts = self.options.get(checker_type_name, set()) - checker_inpts = self.inputs.get(checker_type_name, {}) + checker_opts = self.options.get(checker_type_name, {}) # instantiate a Checker object try: - checker = checker_class(options=checker_opts, inputs=checker_inpts) + checker = checker_class(options=checker_opts) # hacky fix for no options in constructor except TypeError: checker = checker_class() diff --git a/compliance_checker/tests/test_cf.py b/compliance_checker/tests/test_cf.py index 3db25d2d..04ca15b2 100644 --- a/compliance_checker/tests/test_cf.py +++ b/compliance_checker/tests/test_cf.py @@ -240,7 +240,7 @@ def test_appendix_a(self): dataset = self.load_dataset(STATIC_FILES["bad_data_type"]) # Ordinarily, options would be specified in the checker constructor, but # we set them manually here so we don't have to monkey patch `setUp` - self.cf.options = {"enable_appendix_a_checks"} + self.cf.options = {"enable_appendix_a_checks": None} new_check = copy.deepcopy(self.cf) self.cf.setup(dataset) aa_results = self.cf.check_appendix_a(dataset) From f28f3214ce001be2ca6012874941738b87108285 Mon Sep 17 00:00:00 2001 From: sol1105 Date: Mon, 15 Jul 2024 10:00:14 +0200 Subject: [PATCH 11/49] Adding test for cchecker.py:parse_options --- compliance_checker/tests/test_cli.py | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 2788ec25..670e5b27 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -7,9 +7,12 @@ import json import os import platform +import shutil import subprocess import sys from argparse import Namespace +from collections import defaultdict +from importlib.machinery import SourceFileLoader import pytest @@ -268,3 +271,23 @@ def test_nczarr_pass_through(self, zarr_url): output_format="text", ) assert not errors + + +def test_parse_options(): + """Test the option parser of cchecker.py""" + # Load cchecker.py + cchecker = SourceFileLoader("cchecker", shutil.which("cchecker.py")).load_module() + # Simple test checker_type:checker_opt + opt_dict = cchecker.parse_options(["cf:enable_appendix_a_checks"]) + assert opt_dict == defaultdict(dict, {"cf": {"enable_appendix_a_checks": None}}) + # Test case checker_type:checker_opt:checker_val + opt_dict = cchecker.parse_options( + ["type:opt:val", "type:opt2:val:2", "cf:enable_appendix_a_checks"], + ) + assert opt_dict == defaultdict( + dict, + { + "type": {"opt": "val", "opt2": "val:2"}, + "cf": {"enable_appendix_a_checks": None}, + }, + ) From c967e857b30e6b2fbb4eb95fe1973923935233cd Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Tue, 16 Jul 2024 12:24:41 -0400 Subject: [PATCH 12/49] Remove deprecated load_module, import from project folder --- compliance_checker/tests/test_cli.py | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 670e5b27..675a2eff 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -12,6 +12,7 @@ import sys from argparse import Namespace from collections import defaultdict +import importlib.util from importlib.machinery import SourceFileLoader import pytest @@ -276,12 +277,17 @@ def test_nczarr_pass_through(self, zarr_url): def test_parse_options(): """Test the option parser of cchecker.py""" # Load cchecker.py - cchecker = SourceFileLoader("cchecker", shutil.which("cchecker.py")).load_module() + cchecker_file_path = os.path.join(os.path.dirname(__file__), "..", "..", + "cchecker.py") + spec = importlib.util.spec_from_file_location("cchecker", + cchecker_file_path) + module = importlib.util.module_from_spec(spec) + SourceFileLoader(spec.name, spec.origin).exec_module(module) # Simple test checker_type:checker_opt - opt_dict = cchecker.parse_options(["cf:enable_appendix_a_checks"]) + opt_dict = module.parse_options(["cf:enable_appendix_a_checks"]) assert opt_dict == defaultdict(dict, {"cf": {"enable_appendix_a_checks": None}}) # Test case checker_type:checker_opt:checker_val - opt_dict = cchecker.parse_options( + opt_dict = module.parse_options( ["type:opt:val", "type:opt2:val:2", "cf:enable_appendix_a_checks"], ) assert opt_dict == defaultdict( From bda8f4eaae8af18e568da7bd0ffc03b92726fcd8 Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Tue, 16 Jul 2024 12:28:31 -0400 Subject: [PATCH 13/49] Remove now unused shutil module import in test_cli.py --- compliance_checker/tests/test_cli.py | 1 - 1 file changed, 1 deletion(-) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 675a2eff..0dc87d93 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -7,7 +7,6 @@ import json import os import platform -import shutil import subprocess import sys from argparse import Namespace From 6255403c73c4a836ee8eb42b964acde675f7d0f3 Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Tue, 16 Jul 2024 15:44:02 -0400 Subject: [PATCH 14/49] Use module from project directory path --- compliance_checker/tests/test_cli.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 0dc87d93..ceaa108e 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -276,10 +276,10 @@ def test_nczarr_pass_through(self, zarr_url): def test_parse_options(): """Test the option parser of cchecker.py""" # Load cchecker.py - cchecker_file_path = os.path.join(os.path.dirname(__file__), "..", "..", - "cchecker.py") - spec = importlib.util.spec_from_file_location("cchecker", - cchecker_file_path) + cchecker_file_path = os.path.join( + os.path.dirname(__file__), "..", "..", "cchecker.py" + ) + spec = importlib.util.spec_from_file_location("cchecker", cchecker_file_path) module = importlib.util.module_from_spec(spec) SourceFileLoader(spec.name, spec.origin).exec_module(module) # Simple test checker_type:checker_opt From e5913faa532ddc51321337f58a2a402c5bbcc2d5 Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Tue, 16 Jul 2024 15:44:37 -0400 Subject: [PATCH 15/49] Add extra check to see if netCDF libs compiled with nczarr support --- compliance_checker/tests/test_cli.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index ceaa108e..9ce35bb1 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -244,6 +244,10 @@ def _check_libnetcdf_version(): platform.system() != "Linux", reason="NCZarr is not officially supported for your OS as of when this API was written", ) + @pytest.mark.skipif( + subprocess.check_output(["nc-config", "--has-nczarr"]) != b"yes\n", + reason="NCZarr support was not built with this netCDF version", + ) @pytest.mark.parametrize( "zarr_url", [ From 2a4ddfb9f749fcd2320b032739dbd2cf56d166e3 Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Tue, 16 Jul 2024 15:52:47 -0400 Subject: [PATCH 16/49] [Ruff] sort imports in test_cli.py --- compliance_checker/tests/test_cli.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 9ce35bb1..e1027db1 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -3,6 +3,7 @@ Tests for command line output and parsing """ +import importlib.util import io import json import os @@ -11,7 +12,6 @@ import sys from argparse import Namespace from collections import defaultdict -import importlib.util from importlib.machinery import SourceFileLoader import pytest @@ -281,7 +281,10 @@ def test_parse_options(): """Test the option parser of cchecker.py""" # Load cchecker.py cchecker_file_path = os.path.join( - os.path.dirname(__file__), "..", "..", "cchecker.py" + os.path.dirname(__file__), + "..", + "..", + "cchecker.py", ) spec = importlib.util.spec_from_file_location("cchecker", cchecker_file_path) module = importlib.util.module_from_spec(spec) From 23bcd72fade1949fd23052d6d249debb32a64eaa Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 14 Aug 2024 17:03:50 +0200 Subject: [PATCH 17/49] update --- .pre-commit-config.yaml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 3a3ecd05..41330f77 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -19,7 +19,7 @@ repos: - test_requirements.txt - repo: https://github.com/psf/black - rev: 24.4.2 + rev: 24.8.0 hooks: - id: black language_version: python3 @@ -31,12 +31,12 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.5.0 + rev: v0.5.7 hooks: - id: ruff - repo: https://github.com/tox-dev/pyproject-fmt - rev: 2.1.3 + rev: 2.2.1 hooks: - id: pyproject-fmt From abc4089138874029a099ad8b754b9c812d5ff463 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 14 Aug 2024 17:03:54 +0200 Subject: [PATCH 18/49] fix windows tests --- compliance_checker/tests/test_cli.py | 27 ++++++++++++++------------- 1 file changed, 14 insertions(+), 13 deletions(-) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index e1027db1..03e63317 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -20,6 +20,11 @@ from .conftest import datadir, static_files +if platform.system() == "Windows": + ncconfig = ["bash", f"{os.environ['CONDA_PREFIX']}\\Library\\bin\\nc-config"] +else: + ncconfig = ["nc-config"] + @pytest.mark.usefixtures("checksuite_setup") class TestCLI: @@ -221,19 +226,15 @@ def test_multi_checker_return_value(self, tmp_txt_file): assert not return_value def _check_libnetcdf_version(): - if platform.system() == "Linux": - # nc-config doesn't work on windows... and neither does NCZarr so this skipif is mutually exclusive to the OS check skipif - return ( - float( - subprocess.check_output( - ["nc-config", "--version"], - encoding="UTF-8", - )[9:12], - ) - < 8.0 + return ( + float( + subprocess.check_output( + ncconfig + ["--version"], + encoding="UTF-8", + )[9:12], ) - else: - return True + < 8.0 + ) # TODO uncomment the third parameter once S3 support is working @pytest.mark.skipif( @@ -241,7 +242,7 @@ def _check_libnetcdf_version(): reason="NCZarr support was not available until netCDF version 4.8.0. Please upgrade to the latest libnetcdf version to test this functionality", ) @pytest.mark.skipif( - platform.system() != "Linux", + subprocess.check_output([ncconfig, "--has-nczarr"]) != b"yes\n", reason="NCZarr is not officially supported for your OS as of when this API was written", ) @pytest.mark.skipif( From 1586401745e5ebe3d78a06d6b434665641fde558 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 14 Aug 2024 17:12:38 +0200 Subject: [PATCH 19/49] fix Windows tests --- compliance_checker/tests/test_cli.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 03e63317..6452eff4 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -242,7 +242,7 @@ def _check_libnetcdf_version(): reason="NCZarr support was not available until netCDF version 4.8.0. Please upgrade to the latest libnetcdf version to test this functionality", ) @pytest.mark.skipif( - subprocess.check_output([ncconfig, "--has-nczarr"]) != b"yes\n", + subprocess.check_output(ncconfig + ["--has-nczarr"]) != b"yes\n", reason="NCZarr is not officially supported for your OS as of when this API was written", ) @pytest.mark.skipif( From 170bd2f30fe98c73502d19d6f37cc13d7cfa2e66 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 14 Aug 2024 17:12:45 +0200 Subject: [PATCH 20/49] fix lints --- compliance_checker/cf/cf_1_6.py | 4 ++-- compliance_checker/cf/cf_1_9.py | 25 ++++++++++++++++++------- compliance_checker/tests/test_cf.py | 12 +++++++----- 3 files changed, 27 insertions(+), 14 deletions(-) diff --git a/compliance_checker/cf/cf_1_6.py b/compliance_checker/cf/cf_1_6.py index 2bf0f74a..c1280d96 100644 --- a/compliance_checker/cf/cf_1_6.py +++ b/compliance_checker/cf/cf_1_6.py @@ -3340,7 +3340,7 @@ def check_packed_data(self, ds): # IMPLEMENTATION CONFORMANCE 8.1 REQUIRED 1/3 # scale_factor and add_offset same type - if type(add_offset) != type(scale_factor): + if not isinstance(add_offset, type(scale_factor)): valid = False reasoning.append( "Attributes add_offset and scale_factor have different data type.", @@ -3350,7 +3350,7 @@ def check_packed_data(self, ds): # if not the same type # FIXME: Check add_offset too. - elif type(scale_factor) != var.dtype.type: + elif not isinstance(scale_factor, var.dtype.type): # Check both attributes are type float or double if not isinstance(scale_factor, (float, np.floating)): valid = False diff --git a/compliance_checker/cf/cf_1_9.py b/compliance_checker/cf/cf_1_9.py index 73211e44..3fcf5dfe 100644 --- a/compliance_checker/cf/cf_1_9.py +++ b/compliance_checker/cf/cf_1_9.py @@ -150,7 +150,10 @@ def check_domain_variables(self, ds: Dataset): continue appendix_a_not_recommended_attrs = [] for attr_name in domain_var.ncattrs(): - if attr_name in self.appendix_a and "D" not in self.appendix_a[attr_name]["attr_loc"]: + if ( + attr_name in self.appendix_a + and "D" not in self.appendix_a[attr_name]["attr_loc"] + ): appendix_a_not_recommended_attrs.append(attr_name) if appendix_a_not_recommended_attrs: @@ -163,10 +166,12 @@ def check_domain_variables(self, ds: Dataset): # no errors occurred domain_valid.score += 1 - # IMPLEMENTATION CONFORMANCE 5.8 REQUIRED 4/4 if hasattr(domain_var, "cell_measures"): - cell_measures_var_names = regex.findall(r"\b(?:area|volume):\s+(\w+)", domain_var.cell_measures) + cell_measures_var_names = regex.findall( + r"\b(?:area|volume):\s+(\w+)", + domain_var.cell_measures, + ) # check exist for var_name in cell_measures_var_names: try: @@ -174,10 +179,16 @@ def check_domain_variables(self, ds: Dataset): except ValueError: # TODO: what to do here? continue - domain_coord_var_names = {var_like.name for var_like in domain_coord_vars} - domain_valid.assert_true(set(cell_measures_variable.dimensions).issubset(domain_coord_var_names), - "Variables named in the cell_measures attributes must have a dimensions attribute with " - "values that are a subset of the referring domain variable's dimension attribute") + domain_coord_var_names = { + var_like.name for var_like in domain_coord_vars + } + domain_valid.assert_true( + set(cell_measures_variable.dimensions).issubset( + domain_coord_var_names, + ), + "Variables named in the cell_measures attributes must have a dimensions attribute with " + "values that are a subset of the referring domain variable's dimension attribute", + ) results.append(domain_valid.to_result()) diff --git a/compliance_checker/tests/test_cf.py b/compliance_checker/tests/test_cf.py index f053f837..65088f84 100644 --- a/compliance_checker/tests/test_cf.py +++ b/compliance_checker/tests/test_cf.py @@ -3249,7 +3249,7 @@ def test_domain(self): dataset.createDimension("lat", 20) dataset.createDimension("depth", 20) domain_var.setncattr("dimensions", "lon lat depth") - cube = dataset.createVariable("cube", "f8", ("lon", "lat", "depth")) + dataset.createVariable("cube", "f8", ("lon", "lat", "depth")) # OK, coordinates in cell_measures are subset of coordinates of # referring domain variable's coordinates attribute results = self.cf.check_domain_variables(dataset) @@ -3258,10 +3258,12 @@ def test_domain(self): domain_var.cell_measures = "volume: cube_bad" dataset.createVariable("cube_bad", "f8", ("lon", "lat", "depth", "time")) results = self.cf.check_domain_variables(dataset) - self.assertTrue("Variables named in the cell_measures attributes must " - "have a dimensions attribute with values that are a " - "subset of the referring domain variable's dimension " - "attribute" in results[0].msgs) + self.assertTrue( + "Variables named in the cell_measures attributes must " + "have a dimensions attribute with values that are a " + "subset of the referring domain variable's dimension " + "attribute" in results[0].msgs, + ) del dataset dataset = MockTimeSeries() # domain should be dimensionless -- currently not an error in From 3213f6d5790c594441e65bbf8df508b792aa37b5 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 14 Aug 2024 17:18:57 +0200 Subject: [PATCH 21/49] fix Windows tests --- compliance_checker/protocols/zarr.py | 15 ++++++++++----- compliance_checker/suite.py | 5 ----- compliance_checker/tests/test_cli.py | 24 +++++++++++------------- 3 files changed, 21 insertions(+), 23 deletions(-) diff --git a/compliance_checker/protocols/zarr.py b/compliance_checker/protocols/zarr.py index f111624c..517ce9e1 100644 --- a/compliance_checker/protocols/zarr.py +++ b/compliance_checker/protocols/zarr.py @@ -1,3 +1,4 @@ +import platform import zipfile from pathlib import Path from urllib.parse import urlparse @@ -6,7 +7,11 @@ from compliance_checker.protocols import netcdf -# + +def _fix_windows_slashes(zarr_url): + if platform.system() == "Windows": + zarr_url = zarr_url.replace("///", "//") + return zarr_url def is_zarr(url): @@ -55,10 +60,9 @@ def as_zarr(url): pr = urlparse(str(url)) if "mode=nczarr" in pr.fragment: - if pr.netloc: - return str(url) # already valid nczarr url - elif pr.scheme == "file": - return str(url) # already valid nczarr url + if pr.netloc or pr.scheme == "file": + url = _fix_windows_slashes(url) + return url zarr_url = Path( url2pathname(pr.path), @@ -78,4 +82,5 @@ def as_zarr(url): url_base = url if mode == "s3" else zarr_url.as_uri() zarr_url = f"{url_base}#mode=nczarr,{mode}" + zarr_url = _fix_windows_slashes(zarr_url) return zarr_url diff --git a/compliance_checker/suite.py b/compliance_checker/suite.py index 6b534f5f..6a4281e2 100644 --- a/compliance_checker/suite.py +++ b/compliance_checker/suite.py @@ -6,7 +6,6 @@ import inspect import itertools import os -import platform import re import subprocess import sys @@ -893,10 +892,6 @@ def load_local_dataset(self, ds_str): ds_str = self.generate_dataset(ds_str) if zarr.is_zarr(ds_str): - if platform.system() != "Linux": - print( - f"WARNING: {platform.system()} OS detected. NCZarr is not officially supported for your OS as of when this API was written. Your mileage may vary.", - ) return Dataset(zarr.as_zarr(ds_str)) if netcdf.is_netcdf(ds_str): diff --git a/compliance_checker/tests/test_cli.py b/compliance_checker/tests/test_cli.py index 6452eff4..b77b565d 100644 --- a/compliance_checker/tests/test_cli.py +++ b/compliance_checker/tests/test_cli.py @@ -20,8 +20,10 @@ from .conftest import datadir, static_files -if platform.system() == "Windows": - ncconfig = ["bash", f"{os.environ['CONDA_PREFIX']}\\Library\\bin\\nc-config"] +on_windows = platform.system() == "Windows" + +if on_windows: + ncconfig = ["sh", f"{os.environ['CONDA_PREFIX']}\\Library\\bin\\nc-config"] else: ncconfig = ["nc-config"] @@ -236,18 +238,9 @@ def _check_libnetcdf_version(): < 8.0 ) - # TODO uncomment the third parameter once S3 support is working - @pytest.mark.skipif( - _check_libnetcdf_version(), - reason="NCZarr support was not available until netCDF version 4.8.0. Please upgrade to the latest libnetcdf version to test this functionality", - ) @pytest.mark.skipif( subprocess.check_output(ncconfig + ["--has-nczarr"]) != b"yes\n", - reason="NCZarr is not officially supported for your OS as of when this API was written", - ) - @pytest.mark.skipif( - subprocess.check_output(["nc-config", "--has-nczarr"]) != b"yes\n", - reason="NCZarr support was not built with this netCDF version", + reason="NCZarr is not available.", ) @pytest.mark.parametrize( "zarr_url", @@ -256,7 +249,12 @@ def _check_libnetcdf_version(): str(datadir / "zip.zarr"), # "s3://hrrrzarr/sfc/20210408/20210408_10z_anl.zarr#mode=nczarr,s3" ], - ids=["local_file", "zip_file"], # ,'s3_url' + ids=[ + "local_file", + "zip_file", + # TODO uncomment once S3 support is working. + # "s3_url", + ], ) def test_nczarr_pass_through(self, zarr_url): """ From 84d2322c6ad7ec7f80cdef019db3e992b8607cce Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 2 Sep 2024 20:54:37 +0000 Subject: [PATCH 22/49] [pre-commit.ci] pre-commit autoupdate MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit updates: - [github.com/astral-sh/ruff-pre-commit: v0.5.7 → v0.6.3](https://github.com/astral-sh/ruff-pre-commit/compare/v0.5.7...v0.6.3) --- .pre-commit-config.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 41330f77..dc28bb0d 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.5.7 + rev: v0.6.3 hooks: - id: ruff From 9c1a8cef4d3811d97758e59f9a14c29c12568436 Mon Sep 17 00:00:00 2001 From: Benjamin Adams Date: Mon, 16 Sep 2024 11:52:38 -0400 Subject: [PATCH 23/49] Remove set check for certain variable attributes due to sorting issues --- compliance_checker/cf/cf_1_6.py | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/compliance_checker/cf/cf_1_6.py b/compliance_checker/cf/cf_1_6.py index c1280d96..49178abf 100644 --- a/compliance_checker/cf/cf_1_6.py +++ b/compliance_checker/cf/cf_1_6.py @@ -116,17 +116,13 @@ def check_child_attr_data_types(self, ds): """ ctx = TestCtx(BaseCheck.MEDIUM, self.section_titles["2.5"]) - special_attrs = { - "actual_range", - "valid_min", - "valid_max", - "valid_range", - "_FillValue", - } + special_attrs = ("actual_range", "valid_min", "valid_max", + "valid_range", "_FillValue") for _var_name, var in ds.variables.items(): - for att_name in special_attrs.intersection(var.ncattrs()): - self._parent_var_attr_type_check(att_name, var, ctx) + for att_name in special_attrs: + if att_name in var.ncattrs(): + self._parent_var_attr_type_check(att_name, var, ctx) return ctx.to_result() # TODO: consider renaming to avoid confusion with non-underscore From b67dfd96dbf21a688bedeb86173dd1d0502c416e Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 16 Sep 2024 17:10:06 +0000 Subject: [PATCH 24/49] [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --- compliance_checker/cf/cf_1_6.py | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/compliance_checker/cf/cf_1_6.py b/compliance_checker/cf/cf_1_6.py index 49178abf..3979777e 100644 --- a/compliance_checker/cf/cf_1_6.py +++ b/compliance_checker/cf/cf_1_6.py @@ -116,8 +116,13 @@ def check_child_attr_data_types(self, ds): """ ctx = TestCtx(BaseCheck.MEDIUM, self.section_titles["2.5"]) - special_attrs = ("actual_range", "valid_min", "valid_max", - "valid_range", "_FillValue") + special_attrs = ( + "actual_range", + "valid_min", + "valid_max", + "valid_range", + "_FillValue", + ) for _var_name, var in ds.variables.items(): for att_name in special_attrs: From 63adf3d2e912c19ebb1deb575ee94ceeafb88063 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Fri, 4 Oct 2024 18:10:10 +0200 Subject: [PATCH 25/49] bump umamba 2 --- .github/workflows/cc-plugin-ncei-test.yml | 2 +- .github/workflows/cc-plugin-og-test.yml | 2 +- .github/workflows/cc-plugin-sgrid-test.yml | 2 +- .github/workflows/cc-plugin-ugrid-test.yml | 2 +- .github/workflows/codecov.yml | 2 +- .github/workflows/default-tests.yml | 2 +- .github/workflows/deploy-docs.yml | 2 +- .github/workflows/integration-tests.yml | 2 +- 8 files changed, 8 insertions(+), 8 deletions(-) diff --git a/.github/workflows/cc-plugin-ncei-test.yml b/.github/workflows/cc-plugin-ncei-test.yml index 9bace25c..6f24366f 100644 --- a/.github/workflows/cc-plugin-ncei-test.yml +++ b/.github/workflows/cc-plugin-ncei-test.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/cc-plugin-og-test.yml b/.github/workflows/cc-plugin-og-test.yml index 3df4d263..f79850b6 100644 --- a/.github/workflows/cc-plugin-og-test.yml +++ b/.github/workflows/cc-plugin-og-test.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/cc-plugin-sgrid-test.yml b/.github/workflows/cc-plugin-sgrid-test.yml index a73e9604..3fe35d3a 100644 --- a/.github/workflows/cc-plugin-sgrid-test.yml +++ b/.github/workflows/cc-plugin-sgrid-test.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/cc-plugin-ugrid-test.yml b/.github/workflows/cc-plugin-ugrid-test.yml index 51522bc2..0cab2d36 100644 --- a/.github/workflows/cc-plugin-ugrid-test.yml +++ b/.github/workflows/cc-plugin-ugrid-test.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/codecov.yml b/.github/workflows/codecov.yml index c32a9fb4..f1007185 100644 --- a/.github/workflows/codecov.yml +++ b/.github/workflows/codecov.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/default-tests.yml b/.github/workflows/default-tests.yml index 08ee54c6..d3b272d8 100644 --- a/.github/workflows/default-tests.yml +++ b/.github/workflows/default-tests.yml @@ -20,7 +20,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba ${{ matrix.python-version }} - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/deploy-docs.yml b/.github/workflows/deploy-docs.yml index 574b648f..6eae9d5d 100644 --- a/.github/workflows/deploy-docs.yml +++ b/.github/workflows/deploy-docs.yml @@ -18,7 +18,7 @@ jobs: fetch-depth: 0 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash diff --git a/.github/workflows/integration-tests.yml b/.github/workflows/integration-tests.yml index 87c46cfe..ea54d31a 100644 --- a/.github/workflows/integration-tests.yml +++ b/.github/workflows/integration-tests.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash From 673acf791819283ab3fbb2d4eb2330d05651da78 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Fri, 4 Oct 2024 18:10:18 +0200 Subject: [PATCH 26/49] do we still need this pin? --- test_requirements.txt | 2 -- 1 file changed, 2 deletions(-) diff --git a/test_requirements.txt b/test_requirements.txt index 3b2c22ad..1ebf0279 100644 --- a/test_requirements.txt +++ b/test_requirements.txt @@ -11,5 +11,3 @@ pytest>=2.9.0 pytest-cov>=3.0.0 pytest-vcr requests-mock>=1.7.0 -# We need this pin until a new version of vcrpy is out. -urllib3<2 From 76c539430f47a0eb43db71c9025830bc060db6c6 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Fri, 4 Oct 2024 18:17:10 +0200 Subject: [PATCH 27/49] remove inline comments too --- requirements.txt | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/requirements.txt b/requirements.txt index 914faa1e..1a61c43b 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,7 +1,7 @@ cf-units>=2 cftime>=1.1.0 -importlib-metadata # drop this when dropping Python 3.8 -importlib-resources # drop this when dropping Python 3.8 +importlib-metadata +importlib-resources isodate>=0.6.1 jinja2>=2.7.3 lxml>=3.2.1 From 755f71fd1b01f86df468a8a2454eccb1c46d15c8 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Fri, 4 Oct 2024 18:17:57 +0200 Subject: [PATCH 28/49] missed a spot --- .github/workflows/cc-plugin-glider-test.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/cc-plugin-glider-test.yml b/.github/workflows/cc-plugin-glider-test.yml index 09f0f025..e147c0ef 100644 --- a/.github/workflows/cc-plugin-glider-test.yml +++ b/.github/workflows/cc-plugin-glider-test.yml @@ -12,7 +12,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup Micromamba - uses: mamba-org/setup-micromamba@v1 + uses: mamba-org/setup-micromamba@v2 with: environment-name: TEST init-shell: bash From ae023d1831a0911f7a69096ef6ddb5d68d56fc2e Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Fri, 4 Oct 2024 18:31:36 +0200 Subject: [PATCH 29/49] new umamba syntax --- .github/workflows/cc-plugin-glider-test.yml | 2 +- .github/workflows/cc-plugin-ncei-test.yml | 2 +- .github/workflows/cc-plugin-og-test.yml | 2 +- .github/workflows/cc-plugin-sgrid-test.yml | 2 +- .github/workflows/cc-plugin-ugrid-test.yml | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.github/workflows/cc-plugin-glider-test.yml b/.github/workflows/cc-plugin-glider-test.yml index e147c0ef..bd9d3058 100644 --- a/.github/workflows/cc-plugin-glider-test.yml +++ b/.github/workflows/cc-plugin-glider-test.yml @@ -33,6 +33,6 @@ jobs: git clone https://github.com/ioos/cc-plugin-glider.git && cd cc-plugin-glider && micromamba install --file requirements.txt --file requirements-dev.txt --channel conda-forge - && micromamba remove compliance-checker --no-prune --force + && micromamba remove compliance-checker --no-prune-deps --force && python -m pip install -e . --no-deps --force-reinstall && python -m pytest -s -rxs -v cc_plugin_glider diff --git a/.github/workflows/cc-plugin-ncei-test.yml b/.github/workflows/cc-plugin-ncei-test.yml index 6f24366f..ae322ad5 100644 --- a/.github/workflows/cc-plugin-ncei-test.yml +++ b/.github/workflows/cc-plugin-ncei-test.yml @@ -33,6 +33,6 @@ jobs: git clone https://github.com/ioos/cc-plugin-ncei.git && cd cc-plugin-ncei && micromamba install --file requirements.txt --file requirements-dev.txt --channel conda-forge - && micromamba remove compliance-checker --no-prune --force + && micromamba remove compliance-checker --no-prune-deps --force && python -m pip install -e . --no-deps --force-reinstall && python -m pytest -s -rxs -v cc_plugin_ncei diff --git a/.github/workflows/cc-plugin-og-test.yml b/.github/workflows/cc-plugin-og-test.yml index f79850b6..c74a033b 100644 --- a/.github/workflows/cc-plugin-og-test.yml +++ b/.github/workflows/cc-plugin-og-test.yml @@ -33,6 +33,6 @@ jobs: git clone https://github.com/uw-farlab/cc-plugin-og.git && cd cc-plugin-og && micromamba install --file requirements.txt --channel conda-forge - && micromamba remove compliance-checker --no-prune --force + && micromamba remove compliance-checker --no-prune-deps --force && python -m pip install -e . --no-deps --force-reinstall && python -m pytest -s -rxs -v cc_plugin_og diff --git a/.github/workflows/cc-plugin-sgrid-test.yml b/.github/workflows/cc-plugin-sgrid-test.yml index 3fe35d3a..153603ff 100644 --- a/.github/workflows/cc-plugin-sgrid-test.yml +++ b/.github/workflows/cc-plugin-sgrid-test.yml @@ -33,6 +33,6 @@ jobs: git clone https://github.com/ioos/cc-plugin-sgrid.git && cd cc-plugin-sgrid && micromamba install --file requirements.txt --file requirements-dev.txt --channel conda-forge - && micromamba remove compliance-checker --no-prune --force + && micromamba remove compliance-checker --no-prune-deps --force && python -m pip install -e . --no-deps --force-reinstall && python -m pytest -s -rxs -v cc_plugin_sgrid diff --git a/.github/workflows/cc-plugin-ugrid-test.yml b/.github/workflows/cc-plugin-ugrid-test.yml index 0cab2d36..1cb7abf0 100644 --- a/.github/workflows/cc-plugin-ugrid-test.yml +++ b/.github/workflows/cc-plugin-ugrid-test.yml @@ -33,6 +33,6 @@ jobs: git clone https://github.com/ioos/cc-checker-ugrid.git && cd cc-checker-ugrid && micromamba install --file requirements.txt --file requirements-dev.txt --channel conda-forge - && micromamba remove compliance-checker --no-prune --force + && micromamba remove compliance-checker --no-prune-deps --force && python -m pip install -e . --no-deps --force-reinstall && python -m pytest -s -rxs -v cc_plugin_ugrid From b063b47451d4cd2bccf7a1b7d092506643c3dbfd Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 7 Oct 2024 22:22:27 +0000 Subject: [PATCH 30/49] [pre-commit.ci] pre-commit autoupdate MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit updates: - [github.com/pre-commit/pre-commit-hooks: v4.6.0 → v5.0.0](https://github.com/pre-commit/pre-commit-hooks/compare/v4.6.0...v5.0.0) - [github.com/psf/black: 24.8.0 → 24.10.0](https://github.com/psf/black/compare/24.8.0...24.10.0) - [github.com/astral-sh/ruff-pre-commit: v0.6.3 → v0.6.9](https://github.com/astral-sh/ruff-pre-commit/compare/v0.6.3...v0.6.9) - [github.com/tox-dev/pyproject-fmt: 2.2.1 → 2.2.4](https://github.com/tox-dev/pyproject-fmt/compare/2.2.1...2.2.4) --- .pre-commit-config.yaml | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index dc28bb0d..f3aa4bf2 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,6 +1,6 @@ repos: - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.6.0 + rev: v5.0.0 hooks: - id: trailing-whitespace exclude: compliance_checker/tests/data @@ -19,7 +19,7 @@ repos: - test_requirements.txt - repo: https://github.com/psf/black - rev: 24.8.0 + rev: 24.10.0 hooks: - id: black language_version: python3 @@ -31,12 +31,12 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.6.3 + rev: v0.6.9 hooks: - id: ruff - repo: https://github.com/tox-dev/pyproject-fmt - rev: 2.2.1 + rev: 2.2.4 hooks: - id: pyproject-fmt From 0706b38dfb5e81b1a20187b858357cc06f59701d Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Mon, 21 Oct 2024 13:25:20 +0200 Subject: [PATCH 31/49] update --- .pre-commit-config.yaml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index f3aa4bf2..e3d15781 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,12 +31,12 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.6.9 + rev: v0.7.0 hooks: - id: ruff - repo: https://github.com/tox-dev/pyproject-fmt - rev: 2.2.4 + rev: 2.4.3 hooks: - id: pyproject-fmt From 8a71bf6bcd3b4655e2eea067a87e36c9704f65da Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Mon, 21 Oct 2024 14:03:53 +0200 Subject: [PATCH 32/49] add py313 --- .github/workflows/default-tests.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/default-tests.yml b/.github/workflows/default-tests.yml index d3b272d8..a68f2f28 100644 --- a/.github/workflows/default-tests.yml +++ b/.github/workflows/default-tests.yml @@ -9,8 +9,8 @@ jobs: runs-on: ${{ matrix.os }} strategy: matrix: - python-version: [ "3.8", "3.9", "3.10", "3.11", "3.12" ] - os: [windows-latest, ubuntu-latest, macos-latest] + python-version: [ "3.9", "3.10", "3.11", "3.12", "3.13" ] + os: [ windows-latest, ubuntu-latest, macos-latest ] fail-fast: false defaults: run: From 54a9db38b916d77b10d6d3cf23b109ab1eba3873 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Mon, 21 Oct 2024 14:04:19 +0200 Subject: [PATCH 33/49] pre-commit fixes --- pyproject.toml | 1 + 1 file changed, 1 insertion(+) diff --git a/pyproject.toml b/pyproject.toml index 9bb30233..17e8b215 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -33,6 +33,7 @@ classifiers = [ "Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.12", + "Programming Language :: Python :: 3.13", "Topic :: Scientific/Engineering", ] dynamic = [ From 9099e7388f5398edf41b3baff11595ebdec0896c Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 23 Oct 2024 09:31:46 +0200 Subject: [PATCH 34/49] don't record this test with VCR --- .../cassettes/test_netcdf_content_type.yaml | 1113 ++++++++++++++--- compliance_checker/tests/test_protocols.py | 1 - 2 files changed, 949 insertions(+), 165 deletions(-) diff --git a/compliance_checker/tests/cassettes/test_netcdf_content_type.yaml b/compliance_checker/tests/cassettes/test_netcdf_content_type.yaml index 1ef82460..d0b06f6c 100644 --- a/compliance_checker/tests/cassettes/test_netcdf_content_type.yaml +++ b/compliance_checker/tests/cassettes/test_netcdf_content_type.yaml @@ -5,11 +5,11 @@ interactions: Accept: - '*/*' Accept-Encoding: - - gzip, deflate, br + - gzip, deflate, br, zstd Connection: - keep-alive User-Agent: - - python-requests/2.32.1 + - python-requests/2.32.3 method: HEAD uri: https://gliders.ioos.us/erddap/tabledap/amelia-20180501T0000.ncCF?&time%3E=max(time)-1%20hour response: @@ -27,11 +27,11 @@ interactions: Content-Type: - application/x-netcdf Date: - - Tue, 21 May 2024 19:48:37 GMT + - Wed, 23 Oct 2024 07:40:09 GMT Last-Modified: - - Tue, 21 May 2024 19:48:37 GMT + - Wed, 23 Oct 2024 07:40:09 GMT Server: - - nginx/1.18.0 + - nginx/1.20.1 Strict-Transport-Security: - max-age=0 X-Cache: @@ -55,170 +55,955 @@ interactions: Accept: - '*/*' Accept-Encoding: - - gzip, deflate, br + - gzip, deflate, br, zstd Connection: - keep-alive User-Agent: - - python-requests/2.32.1 + - python-requests/2.32.3 method: GET uri: https://gliders.ioos.us/erddap/tabledap/amelia-20180501T0000.ncCF?&time%3E=max(time)-1%20hour response: body: string: !!binary | - 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 - nSTsnm9wj1hb9E97nq2b5/8DW8SJJxyTAAA= + Q0RGAQAAAAAAAAAKAAAABQAAAAp0cmFqZWN0b3J5AAAAAAABAAAAB3Byb2ZpbGUAAAAABwAAAANv + YnMAAAAAUwAAABF0cmFqZWN0b3J5X3N0cmxlbgAAAAAAABQAAAANd21vX2lkX3N0cmxlbgAAAAAA + AAcAAAAMAAAAOAAAAA5hY2tub3dsZWRnbWVudAAAAAAAAgAAADdUaGlzIGRlcGxveW1lbnQgc3Vw + cG9ydGVkIGJ5IEhlaXNpbmctU2ltb25zIEZvdW5kYXRpb24uAAAAAA1jZG1fZGF0YV90eXBlAAAA + AAAAAgAAABFUcmFqZWN0b3J5UHJvZmlsZQAAAAAAABVjZG1fcHJvZmlsZV92YXJpYWJsZXMAAAAA + AAACAAAAPHRpbWVfdXYsbGF0X3V2LGxvbl91dix1LHYscHJvZmlsZV9pZCx0aW1lLGxhdGl0dWRl + LGxvbmdpdHVkZQAAABhjZG1fdHJhamVjdG9yeV92YXJpYWJsZXMAAAACAAAAEXRyYWplY3Rvcnks + d21vX2lkAAAAAAAAEGNvbnRyaWJ1dG9yX25hbWUAAAACAAAAakRvbmdsYWkgR29uZywgSmVhbm5h + IEh1ZHNvbiwgTHlkaWEgQmllbmxpZW4sIEhhaXhhbmcgV2FuZywgU2VhbiBGYXRlLCBQRyBSb3Nz + LCBKb2huIEtlcmZvb3QsIExhdXJhIE5henphcm8AAAAAABBjb250cmlidXRvcl9yb2xlAAAAAgAA + AKRQcmluY2lwYWwgSW52ZXN0aWdhdG9yLCBHbGlkZXIgUGlsb3QsIEdsaWRlciBQcmVwYXJhdGlv + biwgRGF0YSBQcm9jZXNzb3IsIEdsaWRlciBEZXBsb3ltZW50IGFuZCBSZWNvdmVyeSwgR2xpZGVy + IERlcGxveW1lbnQgYW5kIFJlY292ZXJ5LCBEYXRhIE1hbmFnZXIsIERhdGEgTWFuYWdlcgAAAAtD + b252ZW50aW9ucwAAAAACAAAAL1VuaWRhdGEgRGF0YXNldCBEaXNjb3ZlcnkgdjEuMCwgQ09BUkRT + LCBDRi0xLjEwAAAAAA1jcmVhdG9yX2VtYWlsAAAAAAAAAgAAABpuYXp6YXJvQG1hcmluZS5ydXRn + ZXJzLmVkdQAAAAAADGNyZWF0b3JfbmFtZQAAAAIAAAANTGF1cmEgTmF6emFybwAAAAAAAAtjcmVh + dG9yX3VybAAAAAACAAAAIGh0dHA6Ly9ydWNvb2wubWFyaW5lLnJ1dGdlcnMuZWR1AAAADGRhdGVf + Y3JlYXRlZAAAAAIAAAAUMjAxOC0wNS0xM1QxNjo0Njo0M1oAAAALZGF0ZV9pc3N1ZWQAAAAAAgAA + ABQyMDE4LTA1LTEzVDE2OjQ2OjQzWgAAABNFYXN0ZXJubW9zdF9FYXN0aW5nAAAAAAYAAAABwFLE + yfsvi/QAAAALZmVhdHVyZVR5cGUAAAAAAgAAABFUcmFqZWN0b3J5UHJvZmlsZQAAAAAAAA5mb3Jt + YXRfdmVyc2lvbgAAAAAAAgAAACYvdG1wL0lPT1NfR2xpZGVyX05ldENERl92Mi4wLXFhcnRvZC5u + YwAAAAAAEmdlb3NwYXRpYWxfbGF0X21heAAAAAAABgAAAAFAQr0YRA4xXAAAABJnZW9zcGF0aWFs + X2xhdF9taW4AAAAAAAYAAAABQEK834IBBxEAAAAUZ2Vvc3BhdGlhbF9sYXRfdW5pdHMAAAACAAAA + DWRlZ3JlZXNfbm9ydGgAAAAAAAASZ2Vvc3BhdGlhbF9sb25fbWF4AAAAAAAGAAAAAcBSxMn7L4v0 + AAAAEmdlb3NwYXRpYWxfbG9uX21pbgAAAAAABgAAAAHAUsUX0uqb9wAAABRnZW9zcGF0aWFsX2xv + bl91bml0cwAAAAIAAAAMZGVncmVlc19lYXN0AAAAF2dlb3NwYXRpYWxfdmVydGljYWxfbWF4AAAA + AAUAAAABQcD1wwAAABdnZW9zcGF0aWFsX3ZlcnRpY2FsX21pbgAAAAAFAAAAAT/mZmYAAAAcZ2Vv + c3BhdGlhbF92ZXJ0aWNhbF9wb3NpdGl2ZQAAAAIAAAAEZG93bgAAABlnZW9zcGF0aWFsX3ZlcnRp + Y2FsX3VuaXRzAAAAAAAAAgAAAAFtAAAAAAAACmd0c19pbmdlc3QAAAAAAAIAAAAEdHJ1ZQAAAAdo + aXN0b3J5AAAAAAIAAAExMjAxOC0wNS0xM1QxNjo0Njo0M1ogL2hvbWUva2VyZm9vdC9zbG9jdW0v + bWF0bGFiL3NwdC9leHBvcnQvbmMvSU9PUy9EQUMvd3JpdGVRYXJ0b2RHbGlkZXJGbGF0TmMubSVz + dGFuZGFyZCBuYW1lcyBmb3IgcWFydG9kIGZsYWdzIGhhdmUgY2hhbmdlZCBvbiAyMDIyLTA3LTA1 + CjIwMjQtMTAtMjNUMDc6Mjg6NTdaIChsb2NhbCBmaWxlcykKMjAyNC0xMC0yM1QwNzoyODo1N1og + aHR0cDovL2dsaWRlcnMuaW9vcy51cy9lcmRkYXAvdGFibGVkYXAvYW1lbGlhLTIwMTgwNTAxVDAw + MDAubmNDRj8mdGltZSUzRT1tYXgodGltZSktMSUyMGhvdXIAAAAAAAACaWQAAAAAAAIAAAAUYW1l + bGlhLTIwMTgwNTEzVDE5NTAAAAAHaW5mb1VybAAAAAACAAAAH2h0dHBzOi8vZ2xpZGVycy5pb29z + LnVzL2VyZGRhcC8AAAAAC2luc3RpdHV0aW9uAAAAAAIAAAA1VmlyZ2luaWEgSW5zdGl0dXRlIG9m + IE1hcmluZSBTY2llbmNlIC0gV2lsbGlhbSAmIE1hcnkAAAAAAAARaW9vc19kYWNfY2hlY2tzdW0A + AAAAAAACAAAAIGMzMTUwODMxMTU4YThjZTA3N2I4MTdjMmYzMWFkNDk4AAAAEmlvb3NfZGFjX2Nv + bXBsZXRlZAAAAAAAAgAAAARUcnVlAAAAGWlvb3NfcmVnaW9uYWxfYXNzb2NpYXRpb24AAAAAAAAC + AAAACE1BUkFDT09TAAAACGtleXdvcmRzAAAAAgAAAbFBVVZTID4gQXV0b25vbW91cyBVbmRlcndh + dGVyIFZlaGljbGVzLCBFYXJ0aCBTY2llbmNlID4gT2NlYW5zID4gT2NlYW4gUHJlc3N1cmUgPiBX + YXRlciBQcmVzc3VyZSwgRWFydGggU2NpZW5jZSA+IE9jZWFucyA+IE9jZWFuIFRlbXBlcmF0dXJl + ID4gV2F0ZXIgVGVtcGVyYXR1cmUsIEVhcnRoIFNjaWVuY2UgPiBPY2VhbnMgPiBTYWxpbml0eS9E + ZW5zaXR5ID4gQ29uZHVjdGl2aXR5LCBFYXJ0aCBTY2llbmNlID4gT2NlYW5zID4gU2FsaW5pdHkv + RGVuc2l0eSA+IERlbnNpdHksIEVhcnRoIFNjaWVuY2UgPiBPY2VhbnMgPiBTYWxpbml0eS9EZW5z + aXR5ID4gU2FsaW5pdHksIGdsaWRlciwgSW4gU2l0dSBPY2Vhbi1iYXNlZCBwbGF0Zm9ybXMgPiBT + ZWFnbGlkZXIsIFNsb2N1bSwgU3ByYXksIHRyYWplY3RvcnksIHVuZGVyd2F0ZXIgZ2xpZGVyLCB3 + YXRlciwgd21vAAAAAAAAE2tleXdvcmRzX3ZvY2FidWxhcnkAAAAAAgAAABVHQ01EIFNjaWVuY2Ug + S2V5d29yZHMAAAAAAAAHbGljZW5zZQAAAAACAAAAoFRoaXMgZGF0YSBtYXkgYmUgcmVkaXN0cmli + dXRlZCBhbmQgdXNlZCB3aXRob3V0IHJlc3RyaWN0aW9uLiAgRGF0YSBwcm92aWRlZCBhcyBpcyB3 + aXRoIG5vIGV4cHJlc3NlZCBvciBpbXBsaWVkIGFzc3VyYW5jZSBvZiBxdWFsaXR5IGFzc3VyYW5j + ZSBvciBxdWFsaXR5IGNvbnRyb2wAAAAUTWV0YWRhdGFfQ29udmVudGlvbnMAAAACAAAALlVuaWRh + dGEgRGF0YXNldCBEaXNjb3ZlcnkgdjEuMCwgQ09BUkRTLCBDRi0xLjYAAAAAABBuYW1pbmdfYXV0 + aG9yaXR5AAAAAgAAABJlZHUucnV0Z2Vycy5tYXJpbmUAAAAAABVOb3J0aGVybm1vc3RfTm9ydGhp + bmcAAAAAAAAGAAAAAUBCvRhEDjFcAAAADXBsYXRmb3JtX3R5cGUAAAAAAAACAAAADVNsb2N1bSBH + bGlkZXIAAAAAAAAHcHJvamVjdAAAAAACAAAAQFRlc3QgRGVwbG95bWVudCBmb3IgTm9ydGh3ZXN0 + IFBhc3NhZ2UgUHJvamVjdCBpbiBNQVJBQ09PUyBEb21haW4AAAAPcHVibGlzaGVyX2VtYWlsAAAA + AAIAAAAaa2VyZm9vdEBtYXJpbmUucnV0Z2Vycy5lZHUAAAAAAA5wdWJsaXNoZXJfbmFtZQAAAAAA + AgAAAAxKb2huIEtlcmZvb3QAAAANcHVibGlzaGVyX3VybAAAAAAAAAIAAAAgaHR0cDovL3J1Y29v + bC5tYXJpbmUucnV0Z2Vycy5lZHUAAAAKcmVmZXJlbmNlcwAAAAAAAgAAAFpodHRwczovL2lvb3Mu + bm9hYS5nb3Yvd3AtY29udGVudC91cGxvYWRzLzIwMTUvMTAvTWFudWFsLWZvci1RQy1vZi1HbGlk + ZXItRGF0YV8wNV8wOV8xNi5wZGYAAAAAAAhzZWFfbmFtZQAAAAIAAAASTWlkLUF0bGFudGljIEJp + Z2h0AAAAAAAGc291cmNlAAAAAAACAAAAKk9ic2VydmF0aW9uYWwgZGF0YSBmcm9tIGEgcHJvZmls + aW5nIGdsaWRlcgAAAAAACXNvdXJjZVVybAAAAAAAAAIAAAANKGxvY2FsIGZpbGVzKQAAAAAAABVT + b3V0aGVybm1vc3RfTm9ydGhpbmcAAAAAAAAGAAAAAUBCvN+CAQcRAAAAGHN0YW5kYXJkX25hbWVf + dm9jYWJ1bGFyeQAAAAIAAAAGQ0YtdjI1AAAAAAAPc3Vic2V0VmFyaWFibGVzAAAAAAIAAAA0d21v + X2lkLHRyYWplY3RvcnkscHJvZmlsZV9pZCx0aW1lLGxhdGl0dWRlLGxvbmdpdHVkZQAAAAdzdW1t + YXJ5AAAAAAIAAAMdVGhpcyBpcyBhIHRlc3QgbWlzc2lvbiBmb3IgdGhlIE5vcnRod2VzdCBQYXNz + YWdlIFByb2plY3QuIFRoaXMgZ2xpZGVyIHdpbGwgYmUgZGVwbG95ZWQgaW4gdGhlIHNvdXRoZXJu + IE1pZC1BdGxhbnRpYyBCaWdodC4gVGhlIGdvYWwgb2YgdGhlIG1pc3Npb24gaXMgdG8gY29uZHVj + dCBDVEQgYW5kIERPIE9wdG9kZSBjcm9zcy1jYWxpYnJhdGlvbiB3aXRoIGEgcmVjZW50bHkgY2Fs + aWJyYXRlZCBNQVJBQ09PUyBnbGlkZXIgU3lsdmlhIGluIHRoZSBNQVJBQ09PUyBkb21haW4uIEl0 + IHdpbGwgcHJvdmlkZSBhZGRpdGlvbmFsIGh5ZHJvZ3JhcGhpYyBkYXRhIGluIHRoZSBNaWQtQXRs + YW50aWMgcmVnaW9uIGV2ZW4gdGhvdWdoIGl0cyBDVEQgaGFzIG5vdCBiZWVuIHJlY2VudGx5IGNh + bGlicmF0ZWQuIFRoZSBnbGlkZXIgaXMgc2NoZWR1bGVkIHRvIGJlIGRlcGxveWVkIG9uIFR1ZXNk + YXksIE1heSAxLCAyMDE4IGFuZCByZWNvdmVyZWQgb24gTW9uZGF5IE1heSAxNCwgMjAxOC4gVGhl + IGdsaWRlciB3aWxsIGJlIGRlcGxveWVkIG5lYXIgdGhlIDM1IG0gaXNvYmF0aCBhdCBtaWQtc2hl + bGYgb2ZmIFdhY2hlcHJlYWd1ZSBWQSBhbmQgZmx5IG9mZnNob3JlIHRvd2FyZCBXYXNoaW5ndG9u + IENhbnlvbi4gSXQgd2lsbCBjb25kdWN0IHZpcnR1YWwgbW9vcmluZyBwcm9maWxpbmcgZmxpZ2h0 + IGF0IHRoZSBoZWFkIG9mIFdhc2hpbmd0b24gQ2FueW9uIGRvd24gdG8gMzUwIG0gZm9yIGFib3V0 + IGEgd2VlaywgdGhlbiBoZWFkIGJhY2sgaW5zaG9yZSB0b3dhcmQgV2FjaGVwcmVhZ3VlIFZBIGZv + ciByZWNvdmVyeS4AAAAAAAARdGltZV9jb3ZlcmFnZV9lbmQAAAAAAAACAAAAFDIwMTgtMDUtMTRU + MTI6Mjk6MTJaAAAAE3RpbWVfY292ZXJhZ2Vfc3RhcnQAAAAAAgAAABQyMDE4LTA1LTE0VDExOjMw + OjIwWgAAAAV0aXRsZQAAAAAAAAIAAAAUYW1lbGlhLTIwMTgwNTAxVDAwMDAAAAATV2VzdGVybm1v + c3RfRWFzdGluZwAAAAAGAAAAAcBSxRfS6pv3AAAACwAAAEoAAAAKdHJhamVjdG9yeQAAAAAAAgAA + AAAAAAADAAAADAAAAAYAAAALX0NodW5rU2l6ZXMAAAAABAAAAAEAAAAUAAAACV9FbmNvZGluZwAA + AAAAAAIAAAAKSVNPLTg4NTktMQAAAAAAB2NmX3JvbGUAAAAAAgAAAA10cmFqZWN0b3J5X2lkAAAA + AAAAB2NvbW1lbnQAAAAAAgAAACtBIHRyYWplY3RvcnkgaXMgb25lIGRlcGxveW1lbnQgb2YgYSBn + bGlkZXIuAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAApJZGVudGlmaWVyAAAAAAAJbG9uZ19u + YW1lAAAAAAAAAgAAAA9UcmFqZWN0b3J5IE5hbWUAAAAAAgAAABQAAK40AAAABndtb19pZAAAAAAA + AgAAAAAAAAAEAAAADAAAAAMAAAAJX0VuY29kaW5nAAAAAAAAAgAAAApJU08tODg1OS0xAAAAAAAN + aW9vc19jYXRlZ29yeQAAAAAAAAIAAAAKSWRlbnRpZmllcgAAAAAACWxvbmdfbmFtZQAAAAAAAAIA + AAAGV01PIElEAAAAAAACAAAACAAArkgAAAAKcHJvZmlsZV9pZAAAAAAAAQAAAAEAAAAMAAAACAAA + AApfRmlsbFZhbHVlAAAAAAAEAAAAAf///BkAAAAMYWN0dWFsX3JhbmdlAAAABAAAAAIAAAVvAAAF + dQAAAAdjZl9yb2xlAAAAAAIAAAAKcHJvZmlsZV9pZAAAAAAAB2NvbW1lbnQAAAAAAgAAAINTZXF1 + ZW50aWFsIHByb2ZpbGUgbnVtYmVyIHdpdGhpbiB0aGUgdHJhamVjdG9yeS4gIFRoaXMgdmFsdWUg + aXMgdW5pcXVlIGluIGVhY2ggZmlsZSB0aGF0IGlzIHBhcnQgb2YgYSBzaW5nbGUgdHJhamVjdG9y + eS9kZXBsb3ltZW50LgAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAKSWRlbnRpZmllcgAAAAAA + CWxvbmdfbmFtZQAAAAAAAAIAAAAKUHJvZmlsZSBJRAAAAAAACXZhbGlkX21heAAAAAAAAAQAAAAB + f////wAAAAl2YWxpZF9taW4AAAAAAAAEAAAAAQAAAAEAAAAEAAAAHAAArlAAAAAEdGltZQAAAAEA + AAABAAAADAAAAAwAAAATX0Nvb3JkaW5hdGVBeGlzVHlwZQAAAAACAAAABFRpbWUAAAAMYWN0dWFs + X3JhbmdlAAAABgAAAAJB1r5c0x+TgEHWvmBGPKK7AAAABGF4aXMAAAACAAAAAVQAAAAAAAAIY2Fs + ZW5kYXIAAAACAAAACWdyZWdvcmlhbgAAAAAAAAdjb21tZW50AAAAAAIAAAA4VGltZXN0YW1wIGNv + cnJlc3BvbmRpbmcgdG8gdGhlIG1pZC1wb2ludCBvZiB0aGUgcHJvZmlsZS4AAAANaW9vc19jYXRl + Z29yeQAAAAAAAAIAAAAEVGltZQAAAAlsb25nX25hbWUAAAAAAAACAAAADFByb2ZpbGUgVGltZQAA + ABBvYnNlcnZhdGlvbl90eXBlAAAAAgAAAApjYWxjdWxhdGVkAAAAAAAIcGxhdGZvcm0AAAACAAAA + CHBsYXRmb3JtAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAABHRpbWUAAAALdGltZV9vcmlnaW4A + AAAAAgAAABQwMS1KQU4tMTk3MCAwMDowMDowMAAAAAV1bml0cwAAAAAAAAIAAAAic2Vjb25kcyBz + aW5jZSAxOTcwLTAxLTAxVDAwOjAwOjAwWgAAAAAABgAAADgAAK5sAAAACGxhdGl0dWRlAAAAAQAA + AAEAAAAMAAAADwAAABNfQ29vcmRpbmF0ZUF4aXNUeXBlAAAAAAIAAAADTGF0AAAAAApfRmlsbFZh + bHVlAAAAAAAGAAAAAcCPOAAAAAAAAAAADGFjdHVhbF9yYW5nZQAAAAYAAAACQEK834IBBxFAQr0Y + RA4xXAAAAARheGlzAAAAAgAAAAFZAAAAAAAAD2NvbG9yQmFyTWF4aW11bQAAAAAGAAAAAUBWgAAA + AAAAAAAAD2NvbG9yQmFyTWluaW11bQAAAAAGAAAAAcBWgAAAAAAAAAAAB2NvbW1lbnQAAAAAAgAA + AF1WYWx1ZSBpcyBpbnRlcnBvbGF0ZWQgdG8gcHJvdmlkZSBhbiBlc3RpbWF0ZSBvZiB0aGUgbGF0 + aXR1ZGUgYXQgdGhlIG1pZC1wb2ludCBvZiB0aGUgcHJvZmlsZS4AAAAAAAANaW9vc19jYXRlZ29y + eQAAAAAAAAIAAAAITG9jYXRpb24AAAAJbG9uZ19uYW1lAAAAAAAAAgAAABBQcm9maWxlIExhdGl0 + dWRlAAAAEG9ic2VydmF0aW9uX3R5cGUAAAACAAAACmNhbGN1bGF0ZWQAAAAAAAhwbGF0Zm9ybQAA + AAIAAAAIcGxhdGZvcm0AAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAIbGF0aXR1ZGUAAAAFdW5p + dHMAAAAAAAACAAAADWRlZ3JlZXNfbm9ydGgAAAAAAAAJdmFsaWRfbWF4AAAAAAAABgAAAAFAVoAA + AAAAAAAAAAl2YWxpZF9taW4AAAAAAAAGAAAAAcBWgAAAAAAAAAAABgAAADgAAK6kAAAACWxvbmdp + dHVkZQAAAAAAAAEAAAABAAAADAAAAA8AAAATX0Nvb3JkaW5hdGVBeGlzVHlwZQAAAAACAAAAA0xv + bgAAAAAKX0ZpbGxWYWx1ZQAAAAAABgAAAAHAjzgAAAAAAAAAAAxhY3R1YWxfcmFuZ2UAAAAGAAAA + AsBSxRfS6pv3wFLEyfsvi/QAAAAEYXhpcwAAAAIAAAABWAAAAAAAAA9jb2xvckJhck1heGltdW0A + AAAABgAAAAFAZoAAAAAAAAAAAA9jb2xvckJhck1pbmltdW0AAAAABgAAAAHAZoAAAAAAAAAAAAdj + b21tZW50AAAAAAIAAABeVmFsdWUgaXMgaW50ZXJwb2xhdGVkIHRvIHByb3ZpZGUgYW4gZXN0aW1h + dGUgb2YgdGhlIGxvbmdpdHVkZSBhdCB0aGUgbWlkLXBvaW50IG9mIHRoZSBwcm9maWxlLgAAAAAA + DWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAACExvY2F0aW9uAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAR + UHJvZmlsZSBMb25naXR1ZGUAAAAAAAAQb2JzZXJ2YXRpb25fdHlwZQAAAAIAAAAKY2FsY3VsYXRl + ZAAAAAAACHBsYXRmb3JtAAAAAgAAAAhwbGF0Zm9ybQAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAA + AAlsb25naXR1ZGUAAAAAAAAFdW5pdHMAAAAAAAACAAAADGRlZ3JlZXNfZWFzdAAAAAl2YWxpZF9t + YXgAAAAAAAAGAAAAAUBmgAAAAAAAAAAACXZhbGlkX21pbgAAAAAAAAYAAAABwGaAAAAAAAAAAAAG + AAAAOAAArtwAAAAGbGF0X3V2AAAAAAABAAAAAQAAAAwAAAAMAAAACl9GaWxsVmFsdWUAAAAAAAYA + AAABwI84AAAAAAAAAAAPY29sb3JCYXJNYXhpbXVtAAAAAAYAAAABQFaAAAAAAAAAAAAPY29sb3JC + YXJNaW5pbXVtAAAAAAYAAAABwFaAAAAAAAAAAAAHY29tbWVudAAAAAACAAAAyFRoZSBkZXB0aC1h + dmVyYWdlZCBjdXJyZW50IGlzIGFuIGVzdGltYXRlIG9mIHRoZSBuZXQgY3VycmVudCBtZWFzdXJl + ZCB3aGlsZSB0aGUgZ2xpZGVyIGlzIHVuZGVyd2F0ZXIuICBUaGUgdmFsdWUgaXMgY2FsY3VsYXRl + ZCBvdmVyIHRoZSBlbnRpcmUgdW5kZXJ3YXRlciBzZWdtZW50LCB3aGljaCBtYXkgY29uc2lzdCBv + ZiAxIG9yIG1vcmUgZGl2ZXMuAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAACExvY2F0aW9uAAAA + CWxvbmdfbmFtZQAAAAAAAAIAAAAXRGVwdGgtYXZlcmFnZWQgTGF0aXR1ZGUAAAAAEG9ic2VydmF0 + aW9uX3R5cGUAAAACAAAACmNhbGN1bGF0ZWQAAAAAAAhwbGF0Zm9ybQAAAAIAAAAIcGxhdGZvcm0A + AAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAIbGF0aXR1ZGUAAAAFdW5pdHMAAAAAAAACAAAADWRl + Z3JlZXNfbm9ydGgAAAAAAAAJdmFsaWRfbWF4AAAAAAAABgAAAAFAVoAAAAAAAAAAAAl2YWxpZF9t + aW4AAAAAAAAGAAAAAcBWgAAAAAAAAAAABgAAADgAAK8UAAAABmxvbl91dgAAAAAAAQAAAAEAAAAM + AAAADAAAAApfRmlsbFZhbHVlAAAAAAAGAAAAAcCPOAAAAAAAAAAAD2NvbG9yQmFyTWF4aW11bQAA + AAAGAAAAAUBmgAAAAAAAAAAAD2NvbG9yQmFyTWluaW11bQAAAAAGAAAAAcBmgAAAAAAAAAAAB2Nv + bW1lbnQAAAAAAgAAAMhUaGUgZGVwdGgtYXZlcmFnZWQgY3VycmVudCBpcyBhbiBlc3RpbWF0ZSBv + ZiB0aGUgbmV0IGN1cnJlbnQgbWVhc3VyZWQgd2hpbGUgdGhlIGdsaWRlciBpcyB1bmRlcndhdGVy + LiAgVGhlIHZhbHVlIGlzIGNhbGN1bGF0ZWQgb3ZlciB0aGUgZW50aXJlIHVuZGVyd2F0ZXIgc2Vn + bWVudCwgd2hpY2ggbWF5IGNvbnNpc3Qgb2YgMSBvciBtb3JlIGRpdmVzLgAAAA1pb29zX2NhdGVn + b3J5AAAAAAAAAgAAAAhMb2NhdGlvbgAAAAlsb25nX25hbWUAAAAAAAACAAAAGERlcHRoLWF2ZXJh + Z2VkIExvbmdpdHVkZQAAABBvYnNlcnZhdGlvbl90eXBlAAAAAgAAAApjYWxjdWxhdGVkAAAAAAAI + cGxhdGZvcm0AAAACAAAACHBsYXRmb3JtAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAACWxvbmdp + dHVkZQAAAAAAAAV1bml0cwAAAAAAAAIAAAAMZGVncmVlc19lYXN0AAAACXZhbGlkX21heAAAAAAA + AAYAAAABQGaAAAAAAAAAAAAJdmFsaWRfbWluAAAAAAAABgAAAAHAZoAAAAAAAAAAAAYAAAA4AACv + TAAAAAd0aW1lX3V2AAAAAAEAAAABAAAADAAAAAgAAAAIY2FsZW5kYXIAAAACAAAACWdyZWdvcmlh + bgAAAAAAAAdjb21tZW50AAAAAAIAAADIVGhlIGRlcHRoLWF2ZXJhZ2VkIGN1cnJlbnQgaXMgYW4g + ZXN0aW1hdGUgb2YgdGhlIG5ldCBjdXJyZW50IG1lYXN1cmVkIHdoaWxlIHRoZSBnbGlkZXIgaXMg + dW5kZXJ3YXRlci4gIFRoZSB2YWx1ZSBpcyBjYWxjdWxhdGVkIG92ZXIgdGhlIGVudGlyZSB1bmRl + cndhdGVyIHNlZ21lbnQsIHdoaWNoIG1heSBjb25zaXN0IG9mIDEgb3IgbW9yZSBkaXZlcy4AAAAN + aW9vc19jYXRlZ29yeQAAAAAAAAIAAAAEVGltZQAAAAlsb25nX25hbWUAAAAAAAACAAAAE0RlcHRo + LWF2ZXJhZ2VkIFRpbWUAAAAAEG9ic2VydmF0aW9uX3R5cGUAAAACAAAACmNhbGN1bGF0ZWQAAAAA + AA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAAR0aW1lAAAAC3RpbWVfb3JpZ2luAAAAAAIAAAAUMDEt + SkFOLTE5NzAgMDA6MDA6MDAAAAAFdW5pdHMAAAAAAAACAAAAInNlY29uZHMgc2luY2UgMTk3MC0w + MS0wMVQwMDowMDowMFoAAAAAAAYAAAA4AACvhAAAAAF1AAAAAAAAAQAAAAEAAAAMAAAADQAAAApf + RmlsbFZhbHVlAAAAAAAGAAAAAcCPOAAAAAAAAAAAD2NvbG9yQmFyTWF4aW11bQAAAAAGAAAAAT/g + AAAAAAAAAAAAD2NvbG9yQmFyTWluaW11bQAAAAAGAAAAAb/gAAAAAAAAAAAAB2NvbW1lbnQAAAAA + AgAAAMhUaGUgZGVwdGgtYXZlcmFnZWQgY3VycmVudCBpcyBhbiBlc3RpbWF0ZSBvZiB0aGUgbmV0 + IGN1cnJlbnQgbWVhc3VyZWQgd2hpbGUgdGhlIGdsaWRlciBpcyB1bmRlcndhdGVyLiAgVGhlIHZh + bHVlIGlzIGNhbGN1bGF0ZWQgb3ZlciB0aGUgZW50aXJlIHVuZGVyd2F0ZXIgc2VnbWVudCwgd2hp + Y2ggbWF5IGNvbnNpc3Qgb2YgMSBvciBtb3JlIGRpdmVzLgAAAAtjb29yZGluYXRlcwAAAAACAAAA + FWxvbl91diBsYXRfdXYgdGltZV91dgAAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAhDdXJy + ZW50cwAAAAlsb25nX25hbWUAAAAAAAACAAAAKkRlcHRoLWF2ZXJhZ2VkIEVhc3R3YXJkIFNlYSBX + YXRlciBWZWxvY2l0eQAAAAAAEG9ic2VydmF0aW9uX3R5cGUAAAACAAAACmNhbGN1bGF0ZWQAAAAA + AAhwbGF0Zm9ybQAAAAIAAAAIcGxhdGZvcm0AAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAbZWFz + dHdhcmRfc2VhX3dhdGVyX3ZlbG9jaXR5AAAAAAV1bml0cwAAAAAAAAIAAAAFbSBzLTEAAAAAAAAJ + dmFsaWRfbWF4AAAAAAAABgAAAAFAJAAAAAAAAAAAAAl2YWxpZF9taW4AAAAAAAAGAAAAAcAkAAAA + AAAAAAAABgAAADgAAK+8AAAAAXYAAAAAAAABAAAAAQAAAAwAAAANAAAACl9GaWxsVmFsdWUAAAAA + AAYAAAABwI84AAAAAAAAAAAPY29sb3JCYXJNYXhpbXVtAAAAAAYAAAABP+AAAAAAAAAAAAAPY29s + b3JCYXJNaW5pbXVtAAAAAAYAAAABv+AAAAAAAAAAAAAHY29tbWVudAAAAAACAAAAyFRoZSBkZXB0 + aC1hdmVyYWdlZCBjdXJyZW50IGlzIGFuIGVzdGltYXRlIG9mIHRoZSBuZXQgY3VycmVudCBtZWFz + dXJlZCB3aGlsZSB0aGUgZ2xpZGVyIGlzIHVuZGVyd2F0ZXIuICBUaGUgdmFsdWUgaXMgY2FsY3Vs + YXRlZCBvdmVyIHRoZSBlbnRpcmUgdW5kZXJ3YXRlciBzZWdtZW50LCB3aGljaCBtYXkgY29uc2lz + dCBvZiAxIG9yIG1vcmUgZGl2ZXMuAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAVbG9uX3V2IGxhdF91 + diB0aW1lX3V2AAAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAACEN1cnJlbnRzAAAACWxvbmdf + bmFtZQAAAAAAAAIAAAArRGVwdGgtYXZlcmFnZWQgTm9ydGh3YXJkIFNlYSBXYXRlciBWZWxvY2l0 + eQAAAAAQb2JzZXJ2YXRpb25fdHlwZQAAAAIAAAAKY2FsY3VsYXRlZAAAAAAACHBsYXRmb3JtAAAA + AgAAAAhwbGF0Zm9ybQAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAABxub3J0aHdhcmRfc2VhX3dh + dGVyX3ZlbG9jaXR5AAAABXVuaXRzAAAAAAAAAgAAAAVtIHMtMQAAAAAAAAl2YWxpZF9tYXgAAAAA + AAAGAAAAAUAkAAAAAAAAAAAACXZhbGlkX21pbgAAAAAAAAYAAAABwCQAAAAAAAAAAAAGAAAAOAAA + r/QAAAAPdHJhamVjdG9yeUluZGV4AAAAAAEAAAABAAAADAAAAAMAAAASaW5zdGFuY2VfZGltZW5z + aW9uAAAAAAACAAAACnRyYWplY3RvcnkAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAApJZGVu + dGlmaWVyAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAADNUaGUgdHJhamVjdG9yeSB0byB3aGljaCB0 + aGlzIHByb2ZpbGUgaXMgYXNzb2NpYXRlZC4AAAAABAAAABwAALAsAAAAB3Jvd1NpemUAAAAAAQAA + AAEAAAAMAAAAAwAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAApJZGVudGlmaWVyAAAAAAAJbG9u + Z19uYW1lAAAAAAAAAgAAACdOdW1iZXIgb2YgT2JzZXJ2YXRpb25zIGZvciB0aGlzIFByb2ZpbGUA + AAAAEHNhbXBsZV9kaW1lbnNpb24AAAACAAAAA29icwAAAAAEAAAAHAAAsEgAAAAFZGVwdGgAAAAA + AAABAAAAAgAAAAwAAAAVAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAACgAAABNfQ29vcmRpbmF0 + ZUF4aXNUeXBlAAAAAAIAAAAGSGVpZ2h0AAAAAAAWX0Nvb3JkaW5hdGVaaXNQb3NpdGl2ZQAAAAAA + AgAAAARkb3duAAAACl9GaWxsVmFsdWUAAAAAAAUAAAABxHnAAAAAAAxhY3R1YWxfcmFuZ2UAAAAF + AAAAAj/mZmZBwPXDAAAAE2FuY2lsbGFyeV92YXJpYWJsZXMAAAAAAgAAAAhkZXB0aF9xYwAAAARh + eGlzAAAAAgAAAAFaAAAAAAAAD2NvbG9yQmFyTWF4aW11bQAAAAAGAAAAAUCfQAAAAAAAAAAAD2Nv + bG9yQmFyTWluaW11bQAAAAAGAAAAAQAAAAAAAAAAAAAAD2NvbG9yQmFyUGFsZXR0ZQAAAAACAAAA + Ck9jZWFuRGVwdGgAAAAAAAppbnN0cnVtZW50AAAAAAACAAAADmluc3RydW1lbnRfY3RkAAAAAAAN + aW9vc19jYXRlZ29yeQAAAAAAAAIAAAAITG9jYXRpb24AAAAJbG9uZ19uYW1lAAAAAAAAAgAAAAVE + ZXB0aAAAAAAAABBvYnNlcnZhdGlvbl90eXBlAAAAAgAAAApjYWxjdWxhdGVkAAAAAAAIcGxhdGZv + cm0AAAACAAAACHBsYXRmb3JtAAAACHBvc2l0aXZlAAAAAgAAAARkb3duAAAAD3JlZmVyZW5jZV9k + YXR1bQAAAAACAAAAC3NlYS1zdXJmYWNlAAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAAVkZXB0 + aAAAAAAAAAV1bml0cwAAAAAAAAIAAAABbQAAAAAAAAl2YWxpZF9tYXgAAAAAAAAFAAAAAUT6AAAA + AAAJdmFsaWRfbWluAAAAAAAABQAAAAEAAAAAAAAABQAAAUwAALBkAAAADGNvbmR1Y3Rpdml0eQAA + AAEAAAACAAAADAAAABAAAAALX0NodW5rU2l6ZXMAAAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUA + AAAAAAUAAAABxHnAAAAAAAxhY3R1YWxfcmFuZ2UAAAAFAAAAAkBeBMBAeSmzAAAAE2FuY2lsbGFy + eV92YXJpYWJsZXMAAAAAAgAAAMZxYXJ0b2RfY29uZHVjdGl2aXR5X2NsaW1hdG9sb2dpY2FsX2Zs + YWcgcWFydG9kX2NvbmR1Y3Rpdml0eV9mbGF0X2xpbmVfZmxhZyBxYXJ0b2RfY29uZHVjdGl2aXR5 + X2dyb3NzX3JhbmdlX2ZsYWcgcWFydG9kX2NvbmR1Y3Rpdml0eV9yYXRlX29mX2NoYW5nZV9mbGFn + IHFhcnRvZF9jb25kdWN0aXZpdHlfc3Bpa2VfZmxhZyBjb25kdWN0aXZpdHlfcWMAAAAAAA9jb2xv + ckJhck1heGltdW0AAAAABgAAAAFAIgAAAAAAAAAAAA9jb2xvckJhck1pbmltdW0AAAAABgAAAAEA + AAAAAAAAAAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRl + cHRoAAAAAAAACmluc3RydW1lbnQAAAAAAAIAAAAOaW5zdHJ1bWVudF9jdGQAAAAAAA1pb29zX2Nh + dGVnb3J5AAAAAAAAAgAAAAhTYWxpbml0eQAAAAlsb25nX25hbWUAAAAAAAACAAAAIVNlYSBXYXRl + ciBFbGVjdHJpY2FsIENvbmR1Y3Rpdml0eQAAAAAAABBvYnNlcnZhdGlvbl90eXBlAAAAAgAAAAht + ZWFzdXJlZAAAAAhwbGF0Zm9ybQAAAAIAAAAIcGxhdGZvcm0AAAANc3RhbmRhcmRfbmFtZQAAAAAA + AAIAAAAhc2VhX3dhdGVyX2VsZWN0cmljYWxfY29uZHVjdGl2aXR5AAAAAAAABXVuaXRzAAAAAAAA + AgAAAAVTIG0tMQAAAAAAAAl2YWxpZF9tYXgAAAAAAAAFAAAAAUEgAAAAAAAJdmFsaWRfbWluAAAA + AAAABQAAAAEAAAAAAAAABQAAAUwAALGwAAAAD2NvbmR1Y3Rpdml0eV9xYwAAAAABAAAAAgAAAAwA + AAAKAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAACgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAYEA + AAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAA + AAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAAKBub19xY19wZXJmb3JtZWQgZ29vZF9kYXRhIHBy + b2JhYmx5X2dvb2RfZGF0YSBiYWRfZGF0YV90aGF0X2FyZV9wb3RlbnRpYWxseV9jb3JyZWN0YWJs + ZSBiYWRfZGF0YSB2YWx1ZV9jaGFuZ2VkIG5vdF91c2VkIG5vdF91c2VkIGludGVycG9sYXRlZF92 + YWx1ZSBtaXNzaW5nX3ZhbHVlAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAKAAECAwQFBgcICQAAAAAA + DWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAZ + Y29uZHVjdGl2aXR5IFF1YWxpdHkgRmxhZwAAAAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAC1z + ZWFfd2F0ZXJfZWxlY3RyaWNhbF9jb25kdWN0aXZpdHkgc3RhdHVzX2ZsYWcAAAAAAAAJdmFsaWRf + bWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAAAAAAAAAAEAAABUAACy + /AAAAAdkZW5zaXR5AAAAAAEAAAACAAAADAAAABAAAAALX0NodW5rU2l6ZXMAAAAABAAAAAEAAAAK + AAAACl9GaWxsVmFsdWUAAAAAAAUAAAABxHnAAAAAAAxhY3R1YWxfcmFuZ2UAAAAFAAAAAkR/8KRE + gDIlAAAAE2FuY2lsbGFyeV92YXJpYWJsZXMAAAAAAgAAAIhxYXJ0b2RfZGVuc2l0eV9jbGltYXRv + bG9naWNhbF9mbGFnIHFhcnRvZF9kZW5zaXR5X2ZsYXRfbGluZV9mbGFnIHFhcnRvZF9kZW5zaXR5 + X3JhdGVfb2ZfY2hhbmdlX2ZsYWcgcWFydG9kX2RlbnNpdHlfc3Bpa2VfZmxhZyBkZW5zaXR5X3Fj + AAAAD2NvbG9yQmFyTWF4aW11bQAAAAAGAAAAAUCQIAAAAAAAAAAAD2NvbG9yQmFyTWluaW11bQAA + AAAGAAAAAUCP4AAAAAAAAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25n + aXR1ZGUgZGVwdGgAAAAAAAAKaW5zdHJ1bWVudAAAAAAAAgAAAA5pbnN0cnVtZW50X2N0ZAAAAAAA + DWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAR + U2VhIFdhdGVyIERlbnNpdHkAAAAAAAAQb2JzZXJ2YXRpb25fdHlwZQAAAAIAAAAKY2FsY3VsYXRl + ZAAAAAAACHBsYXRmb3JtAAAAAgAAAAhwbGF0Zm9ybQAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAA + ABFzZWFfd2F0ZXJfZGVuc2l0eQAAAAAAAAV1bml0cwAAAAAAAAIAAAAGa2cgbS0zAAAAAAAJdmFs + aWRfbWF4AAAAAAAABQAAAAFEggAAAAAACXZhbGlkX21pbgAAAAAAAAUAAAABRH3AAAAAAAUAAAFM + AACzUAAAAApkZW5zaXR5X3FjAAAAAAABAAAAAgAAAAwAAAAKAAAAC19DaHVua1NpemVzAAAAAAQA + AAABAAAACgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAYEAAAAAAAALY29vcmRpbmF0ZXMAAAAAAgAA + AB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAA + AgAAAKBub19xY19wZXJmb3JtZWQgZ29vZF9kYXRhIHByb2JhYmx5X2dvb2RfZGF0YSBiYWRfZGF0 + YV90aGF0X2FyZV9wb3RlbnRpYWxseV9jb3JyZWN0YWJsZSBiYWRfZGF0YSB2YWx1ZV9jaGFuZ2Vk + IG5vdF91c2VkIG5vdF91c2VkIGludGVycG9sYXRlZF92YWx1ZSBtaXNzaW5nX3ZhbHVlAAAAC2Zs + YWdfdmFsdWVzAAAAAAEAAAAKAAECAwQFBgcICQAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAA + BU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAUZGVuc2l0eSBRdWFsaXR5IEZsYWcAAAAN + c3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAdc2VhX3dhdGVyX2RlbnNpdHkgc3RhdHVzX2ZsYWcAAAAA + AAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAAAAAAAA + AAEAAABUAAC0nAAAAAhkZXB0aF9xYwAAAAEAAAACAAAADAAAAAoAAAALX0NodW5rU2l6ZXMAAAAA + BAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAAAAAAAAtjb29yZGluYXRlcwAAAAAC + AAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MAAAAA + AAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJvYmFibHlfZ29vZF9kYXRhIGJhZF9k + YXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxlIGJhZF9kYXRhIHZhbHVlX2NoYW5n + ZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3ZhbHVlIG1pc3NpbmdfdmFsdWUAAAAL + ZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIA + AAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABJkZXB0aCBRdWFsaXR5IEZsYWcAAAAA + AA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAABFkZXB0aCBzdGF0dXNfZmxhZwAAAAAAAAl2YWxpZF9t + YXgAAAAAAAABAAAAAQkAAAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEAAAAAAAAAAQAAAFQAALTw + AAAADmluc3RydW1lbnRfY3RkAAAAAAABAAAAAgAAAAwAAAAMAAAACl9GaWxsVmFsdWUAAAAAAAEA + AAABfwAAAAAAABBjYWxpYnJhdGlvbl9kYXRlAAAAAgAAAAoyMDEzLzA2LzEzAAAAAAAHY29tbWVu + dAAAAAACAAAACnB1bXBlZCBDVEQAAAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1 + ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAAEmZhY3RvcnlfY2FsaWJyYXRlZAAAAAAAAgAAAAoyMDEz + LzA2LzEzAAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAKSWRlbnRpZmllcgAAAAAACWxvbmdf + bmFtZQAAAAAAAAIAAAAMQ1REIE1ldGFkYXRhAAAACm1ha2VfbW9kZWwAAAAAAAIAAAANU2VhYmly + ZCBHUENURAAAAAAAAAhwbGF0Zm9ybQAAAAIAAAAIcGxhdGZvcm0AAAANc2VyaWFsX251bWJlcgAA + AAAAAAIAAAAEOTA5MgAAAAR0eXBlAAAAAgAAAAhwbGF0Zm9ybQAAAAV1bml0cwAAAAAAAAIAAAAB + MQAAAAAAAAEAAABUAAC1RAAAAAZsYXRfcWMAAAAAAAEAAAACAAAADAAAAAoAAAALX0NodW5rU2l6 + ZXMAAAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAAAAAAAAtjb29yZGluYXRl + cwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmlu + Z3MAAAAAAAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJvYmFibHlfZ29vZF9kYXRh + IGJhZF9kYXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxlIGJhZF9kYXRhIHZhbHVl + X2NoYW5nZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3ZhbHVlIG1pc3NpbmdfdmFs + dWUAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAANaW9vc19jYXRlZ29yeQAA + AAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABBsYXQgUXVhbGl0eSBGbGFn + AAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAFGxhdGl0dWRlIHN0YXR1c19mbGFnAAAACXZhbGlk + X21heAAAAAAAAAEAAAABCQAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAAAQAAAAAAAAABAAAAVAAA + tZgAAAAJbGF0X3V2X3FjAAAAAAAAAQAAAAIAAAAMAAAACQAAAApfRmlsbFZhbHVlAAAAAAABAAAA + AYEAAAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0 + aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAAKBub19xY19wZXJmb3JtZWQgZ29vZF9kYXRh + IHByb2JhYmx5X2dvb2RfZGF0YSBiYWRfZGF0YV90aGF0X2FyZV9wb3RlbnRpYWxseV9jb3JyZWN0 + YWJsZSBiYWRfZGF0YSB2YWx1ZV9jaGFuZ2VkIG5vdF91c2VkIG5vdF91c2VkIGludGVycG9sYXRl + ZF92YWx1ZSBtaXNzaW5nX3ZhbHVlAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAKAAECAwQFBgcICQAA + AAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIA + AAATbGF0X3V2IFF1YWxpdHkgRmxhZwAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAUbGF0aXR1 + ZGUgc3RhdHVzX2ZsYWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAA + AAAAAAEAAAABAAAAAAAAAAEAAABUAAC17AAAAAZsb25fcWMAAAAAAAEAAAACAAAADAAAAAoAAAAL + X0NodW5rU2l6ZXMAAAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAAAAAAAAtj + b29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZs + YWdfbWVhbmluZ3MAAAAAAAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJvYmFibHlf + Z29vZF9kYXRhIGJhZF9kYXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxlIGJhZF9k + YXRhIHZhbHVlX2NoYW5nZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3ZhbHVlIG1p + c3NpbmdfdmFsdWUAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAANaW9vc19j + YXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABBsb24gUXVh + bGl0eSBGbGFnAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAFWxvbmdpdHVkZSBzdGF0dXNfZmxh + ZwAAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQkAAAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEA + AAAAAAAAAQAAAFQAALZAAAAACWxvbl91dl9xYwAAAAAAAAEAAAACAAAADAAAAAkAAAAKX0ZpbGxW + YWx1ZQAAAAAAAQAAAAGBAAAAAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBs + b25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5ncwAAAAAAAAIAAACgbm9fcWNfcGVyZm9y + bWVkIGdvb2RfZGF0YSBwcm9iYWJseV9nb29kX2RhdGEgYmFkX2RhdGFfdGhhdF9hcmVfcG90ZW50 + aWFsbHlfY29ycmVjdGFibGUgYmFkX2RhdGEgdmFsdWVfY2hhbmdlZCBub3RfdXNlZCBub3RfdXNl + ZCBpbnRlcnBvbGF0ZWRfdmFsdWUgbWlzc2luZ192YWx1ZQAAAAtmbGFnX3ZhbHVlcwAAAAABAAAA + CgABAgMEBQYHCAkAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25n + X25hbWUAAAAAAAACAAAAE2xvbl91diBRdWFsaXR5IEZsYWcAAAAADXN0YW5kYXJkX25hbWUAAAAA + AAACAAAAFWxvbmdpdHVkZSBzdGF0dXNfZmxhZwAAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQkA + AAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEAAAAAAAAAAQAAAFQAALaUAAAADXBsYXRmb3JtX21l + dGEAAAAAAAABAAAAAgAAAAwAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABfwAAAAAAAAdjb21t + ZW50AAAAAAIAAAAZVklNUyBTbG9jdW0gR2xpZGVyIEFtZWxpYQAAAAAAAAtjb29yZGluYXRlcwAA + AAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAAAmlkAAAAAAACAAAABmFt + ZWxpYQAAAAAACmluc3RydW1lbnQAAAAAAAIAAAAOaW5zdHJ1bWVudF9jdGQAAAAAAA1pb29zX2Nh + dGVnb3J5AAAAAAAAAgAAAApJZGVudGlmaWVyAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABFQbGF0 + Zm9ybSBNZXRhZGF0YQAAAAAAAAR0eXBlAAAAAgAAAAhwbGF0Zm9ybQAAAAV1bml0cwAAAAAAAAIA + AAABMQAAAAAAAAZ3bW9faWQAAAAAAAIAAAAHNDgwMTkzNAAAAAABAAAAVAAAtugAAAALcHJlY2lz + ZV9sYXQAAAAAAQAAAAIAAAAMAAAAEgAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0Zp + bGxWYWx1ZQAAAAAABgAAAAHAjzgAAAAAAAAAAAxhY3R1YWxfcmFuZ2UAAAAGAAAAAkBCvN8pLOwC + QEK9GJSCvGQAAAATYW5jaWxsYXJ5X3ZhcmlhYmxlcwAAAAACAAAAG3FhcnRvZF9sb2NhdGlvbl9m + bGFnIGxhdF9xYwAAAAAPY29sb3JCYXJNYXhpbXVtAAAAAAYAAAABQFaAAAAAAAAAAAAPY29sb3JC + YXJNaW5pbXVtAAAAAAYAAAABwFaAAAAAAAAAAAAHY29tbWVudAAAAAACAAAANkludGVycG9sYXRl + ZCBsYXRpdHVkZSBhdCBlYWNoIHBvaW50IGluIHRoZSB0aW1lLXNlcmllcwAAAAAAGmNvb3JkaW5h + dGVfcmVmZXJlbmNlX2ZyYW1lAAAAAAACAAAAFnVybjpvZ2M6Y3JzOkVQU0c6OjQzMjYAAAAAAAtj + b29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWlv + b3NfY2F0ZWdvcnkAAAAAAAACAAAACExvY2F0aW9uAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAQUHJl + Y2lzZSBMYXRpdHVkZQAAABBvYnNlcnZhdGlvbl90eXBlAAAAAgAAAAhtZWFzdXJlZAAAAAhwbGF0 + Zm9ybQAAAAIAAAAIcGxhdGZvcm0AAAAJcmVmZXJlbmNlAAAAAAAAAgAAAAVXR1M4NAAAAAAAAA1z + dGFuZGFyZF9uYW1lAAAAAAAAAgAAAAhsYXRpdHVkZQAAAAV1bml0cwAAAAAAAAIAAAANZGVncmVl + c19ub3J0aAAAAAAAAAl2YWxpZF9tYXgAAAAAAAAGAAAAAUBWgAAAAAAAAAAACXZhbGlkX21pbgAA + AAAAAAYAAAABwFaAAAAAAAAAAAAGAAACmAAAtzwAAAALcHJlY2lzZV9sb24AAAAAAQAAAAIAAAAM + AAAAEgAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAABgAAAAHA + jzgAAAAAAAAAAAxhY3R1YWxfcmFuZ2UAAAAGAAAAAsBSxRiuF1kswFLEx//gH04AAAATYW5jaWxs + YXJ5X3ZhcmlhYmxlcwAAAAACAAAAG3FhcnRvZF9sb2NhdGlvbl9mbGFnIGxvbl9xYwAAAAAPY29s + b3JCYXJNYXhpbXVtAAAAAAYAAAABQGaAAAAAAAAAAAAPY29sb3JCYXJNaW5pbXVtAAAAAAYAAAAB + wGaAAAAAAAAAAAAHY29tbWVudAAAAAACAAAAN0ludGVycG9sYXRlZCBsb25naXR1ZGUgYXQgZWFj + aCBwb2ludCBpbiB0aGUgdGltZS1zZXJpZXMAAAAAGmNvb3JkaW5hdGVfcmVmZXJlbmNlX2ZyYW1l + AAAAAAACAAAAFnVybjpvZ2M6Y3JzOkVQU0c6OjQzMjYAAAAAAAtjb29yZGluYXRlcwAAAAACAAAA + HXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAAC + AAAACExvY2F0aW9uAAAACWxvbmdfbmFtZQAAAAAAAAIAAAARUHJlY2lzZSBMb25naXR1ZGUAAAAA + AAAQb2JzZXJ2YXRpb25fdHlwZQAAAAIAAAAIbWVhc3VyZWQAAAAIcGxhdGZvcm0AAAACAAAACHBs + YXRmb3JtAAAACXJlZmVyZW5jZQAAAAAAAAIAAAAFV0dTODQAAAAAAAANc3RhbmRhcmRfbmFtZQAA + AAAAAAIAAAAJbG9uZ2l0dWRlAAAAAAAABXVuaXRzAAAAAAAAAgAAAAxkZWdyZWVzX2Vhc3QAAAAJ + dmFsaWRfbWF4AAAAAAAABgAAAAFAZoAAAAAAAAAAAAl2YWxpZF9taW4AAAAAAAAGAAAAAcBmgAAA + AAAAAAAABgAAApgAALnUAAAADHByZWNpc2VfdGltZQAAAAEAAAACAAAADAAAAAwAAAALX0NodW5r + U2l6ZXMAAAAABAAAAAEAAAAKAAAADGFjdHVhbF9yYW5nZQAAAAYAAAACQda+XMk55OhB1r5gTB8Z + 8gAAABNhbmNpbGxhcnlfdmFyaWFibGVzAAAAAAIAAAAHdGltZV9xYwAAAAAIY2FsZW5kYXIAAAAC + AAAACWdyZWdvcmlhbgAAAAAAAAdjb21tZW50AAAAAAIAAAAqVGltZXN0YW1wIGF0IGVhY2ggcG9p + bnQgaW4gdGhlIHRpbWUtc2VyaWVzAAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0 + dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAARUaW1lAAAA + CWxvbmdfbmFtZQAAAAAAAAIAAAAMUHJlY2lzZSBUaW1lAAAAEG9ic2VydmF0aW9uX3R5cGUAAAAC + AAAACG1lYXN1cmVkAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAABHRpbWUAAAALdGltZV9vcmln + aW4AAAAAAgAAABQwMS1KQU4tMTk3MCAwMDowMDowMAAAAAV1bml0cwAAAAAAAAIAAAAic2Vjb25k + cyBzaW5jZSAxOTcwLTAxLTAxVDAwOjAwOjAwWgAAAAAABgAAApgAALxsAAAACHByZXNzdXJlAAAA + AQAAAAIAAAAMAAAAEgAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAA + AAAABQAAAAHEecAAAAAADGFjdHVhbF9yYW5nZQAAAAUAAAACP+ZmZkHA9cMAAAATYW5jaWxsYXJ5 + X3ZhcmlhYmxlcwAAAAACAAAAqXFhcnRvZF9tb25vdG9uaWNfcHJlc3N1cmVfZmxhZyBxYXJ0b2Rf + cHJlc3N1cmVfZmxhdF9saW5lX2ZsYWcgcWFydG9kX3ByZXNzdXJlX2dyb3NzX3JhbmdlX2ZsYWcg + cWFydG9kX3ByZXNzdXJlX3JhdGVfb2ZfY2hhbmdlX2ZsYWcgcWFydG9kX3ByZXNzdXJlX3NwaWtl + X2ZsYWcgcHJlc3N1cmVfcWMAAAAAAAAPY29sb3JCYXJNYXhpbXVtAAAAAAYAAAABQJ9AAAAAAAAA + AAAPY29sb3JCYXJNaW5pbXVtAAAAAAYAAAABAAAAAAAAAAAAAAALY29vcmRpbmF0ZXMAAAAAAgAA + AB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAAppbnN0cnVtZW50AAAAAAACAAAA + Dmluc3RydW1lbnRfY3RkAAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAIUHJlc3N1cmUAAAAJ + bG9uZ19uYW1lAAAAAAAAAgAAABJTZWEgV2F0ZXIgUHJlc3N1cmUAAAAAABBvYnNlcnZhdGlvbl90 + eXBlAAAAAgAAAAhtZWFzdXJlZAAAAAhwbGF0Zm9ybQAAAAIAAAAIcGxhdGZvcm0AAAAIcG9zaXRp + dmUAAAACAAAABGRvd24AAAAPcmVmZXJlbmNlX2RhdHVtAAAAAAIAAAALc2VhLXN1cmZhY2UAAAAA + DXN0YW5kYXJkX25hbWUAAAAAAAACAAAAEnNlYV93YXRlcl9wcmVzc3VyZQAAAAAABXVuaXRzAAAA + AAAAAgAAAARkYmFyAAAACXZhbGlkX21heAAAAAAAAAUAAAABRPoAAAAAAAl2YWxpZF9taW4AAAAA + AAAFAAAAAQAAAAAAAAAFAAABTAAAvwQAAAALcHJlc3N1cmVfcWMAAAAAAQAAAAIAAAAMAAAACgAA + AAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAGBAAAAAAAA + C2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAAN + ZmxhZ19tZWFuaW5ncwAAAAAAAAIAAACgbm9fcWNfcGVyZm9ybWVkIGdvb2RfZGF0YSBwcm9iYWJs + eV9nb29kX2RhdGEgYmFkX2RhdGFfdGhhdF9hcmVfcG90ZW50aWFsbHlfY29ycmVjdGFibGUgYmFk + X2RhdGEgdmFsdWVfY2hhbmdlZCBub3RfdXNlZCBub3RfdXNlZCBpbnRlcnBvbGF0ZWRfdmFsdWUg + bWlzc2luZ192YWx1ZQAAAAtmbGFnX3ZhbHVlcwAAAAABAAAACgABAgMEBQYHCAkAAAAAAA1pb29z + X2NhdGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAFXByZXNz + dXJlIFF1YWxpdHkgRmxhZwAAAAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAB5zZWFfd2F0ZXJf + cHJlc3N1cmUgc3RhdHVzX2ZsYWcAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQkAAAAAAAAJdmFs + aWRfbWluAAAAAAAAAQAAAAEAAAAAAAAAAQAAAFQAAMBQAAAADnByb2ZpbGVfbGF0X3FjAAAAAAAB + AAAAAgAAAAwAAAAJAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAAAAAAAAtjb29yZGluYXRlcwAA + AAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MA + AAAAAAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJvYmFibHlfZ29vZF9kYXRhIGJh + ZF9kYXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxlIGJhZF9kYXRhIHZhbHVlX2No + YW5nZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3ZhbHVlIG1pc3NpbmdfdmFsdWUA + AAALZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAANaW9vc19jYXRlZ29yeQAAAAAA + AAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABhwcm9maWxlX2xhdCBRdWFsaXR5 + IEZsYWcAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAUbGF0aXR1ZGUgc3RhdHVzX2ZsYWcAAAAJ + dmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAAAAAAAAAAEA + AABUAADApAAAAA5wcm9maWxlX2xvbl9xYwAAAAAAAQAAAAIAAAAMAAAACQAAAApfRmlsbFZhbHVl + AAAAAAABAAAAAYEAAAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdp + dHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAAKBub19xY19wZXJmb3JtZWQg + Z29vZF9kYXRhIHByb2JhYmx5X2dvb2RfZGF0YSBiYWRfZGF0YV90aGF0X2FyZV9wb3RlbnRpYWxs + eV9jb3JyZWN0YWJsZSBiYWRfZGF0YSB2YWx1ZV9jaGFuZ2VkIG5vdF91c2VkIG5vdF91c2VkIGlu + dGVycG9sYXRlZF92YWx1ZSBtaXNzaW5nX3ZhbHVlAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAKAAEC + AwQFBgcICQAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFt + ZQAAAAAAAAIAAAAYcHJvZmlsZV9sb24gUXVhbGl0eSBGbGFnAAAADXN0YW5kYXJkX25hbWUAAAAA + AAACAAAAFWxvbmdpdHVkZSBzdGF0dXNfZmxhZwAAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQkA + AAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEAAAAAAAAAAQAAAFQAAMD4AAAAD3Byb2ZpbGVfdGlt + ZV9xYwAAAAABAAAAAgAAAAwAAAAJAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAAAAAAAAtjb29y + ZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdf + bWVhbmluZ3MAAAAAAAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJvYmFibHlfZ29v + ZF9kYXRhIGJhZF9kYXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxlIGJhZF9kYXRh + IHZhbHVlX2NoYW5nZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3ZhbHVlIG1pc3Np + bmdfdmFsdWUAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAANaW9vc19jYXRl + Z29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABlwcm9maWxlX3Rp + bWUgUXVhbGl0eSBGbGFnAAAAAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAEHRpbWUgc3RhdHVz + X2ZsYWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAAB + AAAAAAAAAAEAAABUAADBTAAAACdxYXJ0b2RfY29uZHVjdGl2aXR5X2NsaW1hdG9sb2dpY2FsX2Zs + YWcAAAAAAQAAAAIAAAAMAAAACwAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxW + YWx1ZQAAAAAAAQAAAAEJAAAAAAAAB2NvbW1lbnQAAAAAAgAAACdRQVJUT0QgY29uZHVjdGl2aXR5 + IGNsaW1hdG9sb2dpY2FsIHRlc3QAAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVk + ZSBsb25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5ncwAAAAAAAAIAAAARUGFzcyBTdXNw + ZWN0IEZhaWwAAAAAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAMBAwQAAAAADWlvb3NfY2F0ZWdvcnkA + AAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAnUUFSVE9EIGNvbmR1Y3Rp + dml0eSBjbGltYXRvbG9naWNhbCB0ZXN0AAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAB1jbGlt + YXRvbG9neV90ZXN0X3F1YWxpdHlfZmxhZwAAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQQAAAAA + AAAJdmFsaWRfbWluAAAAAAAAAQAAAAEBAAAAAAAAAQAAAFQAAMGgAAAAInFhcnRvZF9jb25kdWN0 + aXZpdHlfZmxhdF9saW5lX2ZsYWcAAAAAAAEAAAACAAAADAAAAAsAAAALX0NodW5rU2l6ZXMAAAAA + BAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAdjb21tZW50AAAAAAIAAAAi + UUFSVE9EIGNvbmR1Y3Rpdml0eSBmbGF0IGxpbmUgdGVzdAAAAAAAC2Nvb3JkaW5hdGVzAAAAAAIA + AAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5ncwAAAAAA + AAIAAAARUGFzcyBTdXNwZWN0IEZhaWwAAAAAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAMBAwQAAAAA + DWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAi + UUFSVE9EIGNvbmR1Y3Rpdml0eSBmbGF0IGxpbmUgdGVzdAAAAAAADXN0YW5kYXJkX25hbWUAAAAA + AAACAAAAG2ZsYXRfbGluZV90ZXN0X3F1YWxpdHlfZmxhZwAAAAAJdmFsaWRfbWF4AAAAAAAAAQAA + AAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADB9AAAACRxYXJ0b2Rf + Y29uZHVjdGl2aXR5X2dyb3NzX3JhbmdlX2ZsYWcAAAABAAAAAgAAAAwAAAAMAAAAC19DaHVua1Np + emVzAAAAAAQAAAABAAAACgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAQkAAAAAAAAMYWN0dWFsX3Jh + bmdlAAAAAQAAAAIBAQAAAAAAB2NvbW1lbnQAAAAAAgAAACRRQVJUT0QgY29uZHVjdGl2aXR5IGdy + b3NzIHJhbmdlIHRlc3QAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdp + dHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFp + bAAAAAAAAAtmbGFnX3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIA + AAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAACRRQVJUT0QgY29uZHVjdGl2aXR5IGdy + b3NzIHJhbmdlIHRlc3QAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAdZ3Jvc3NfcmFuZ2VfdGVz + dF9xdWFsaXR5X2ZsYWcAAAAAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlkX21p + bgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADCSAAAACBxYXJ0b2RfY29uZHVjdGl2aXR5X3ByaW1h + cnlfZmxhZwAAAAEAAAACAAAADAAAAAcAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0 + dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAAtkYWNfY29tbWVudAAAAAACAAAAFWlvb3NfcWNfbW9k + dWxlX3FhcnRvZAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAACdQQVNTIE5PVF9FVkFMVUFU + RUQgU1VTUEVDVCBGQUlMIE1JU1NJTkcAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAFAQIDBAkAAAAA + AAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAHUXVhbGl0eQAAAAAJdmFsaWRfbWF4AAAAAAAAAQAA + AAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADCnAAAACdxYXJ0b2Rf + Y29uZHVjdGl2aXR5X3JhdGVfb2ZfY2hhbmdlX2ZsYWcAAAAAAQAAAAIAAAAMAAAADAAAAAtfQ2h1 + bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAAAAAADGFjdHVh + bF9yYW5nZQAAAAEAAAACAQMAAAAAAAdjb21tZW50AAAAAAIAAAAnUUFSVE9EIGNvbmR1Y3Rpdml0 + eSByYXRlIG9mIGNoYW5nZSB0ZXN0AAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1 + ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAAEVBhc3MgU3Vz + cGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQMEAAAAAA1pb29zX2NhdGVnb3J5 + AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAJ1FBUlRPRCBjb25kdWN0 + aXZpdHkgcmF0ZSBvZiBjaGFuZ2UgdGVzdAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAgcmF0 + ZV9vZl9jaGFuZ2VfdGVzdF9xdWFsaXR5X2ZsYWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAA + AAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADC8AAAAB5xYXJ0b2RfY29uZHVj + dGl2aXR5X3NwaWtlX2ZsYWcAAAAAAAEAAAACAAAADAAAAAwAAAALX0NodW5rU2l6ZXMAAAAABAAA + AAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UAAAABAAAA + AgECAAAAAAAHY29tbWVudAAAAAACAAAAHlFBUlRPRCBjb25kdWN0aXZpdHkgc3Bpa2UgdGVzdAAA + AAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAA + AAANZmxhZ19tZWFuaW5ncwAAAAAAAAIAAAARUGFzcyBTdXNwZWN0IEZhaWwAAAAAAAALZmxhZ192 + YWx1ZXMAAAAAAQAAAAMBAwQAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAA + CWxvbmdfbmFtZQAAAAAAAAIAAAAeUUFSVE9EIGNvbmR1Y3Rpdml0eSBzcGlrZSB0ZXN0AAAAAAAN + c3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAXc3Bpa2VfdGVzdF9xdWFsaXR5X2ZsYWcAAAAACXZhbGlk + X21heAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAAAQEAAAAAAAABAAAAVAAA + w0QAAAAicWFydG9kX2RlbnNpdHlfY2xpbWF0b2xvZ2ljYWxfZmxhZwAAAAAAAQAAAAIAAAAMAAAA + DAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAA + AAAADGFjdHVhbF9yYW5nZQAAAAEAAAACAQEAAAAAAAdjb21tZW50AAAAAAIAAAAfUUFSVE9EIGRl + bnNpdHkgY2xpbWF0b2xvZ3kgdGVzdAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0 + dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1 + c3BlY3QgRmFpbAAAAAAAAAtmbGFnX3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29y + eQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAAB9RQVJUT0QgZGVuc2l0 + eSBjbGltYXRvbG9neSB0ZXN0AAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAB1jbGltYXRvbG9n + eV90ZXN0X3F1YWxpdHlfZmxhZwAAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQQAAAAAAAAJdmFs + aWRfbWluAAAAAAAAAQAAAAEBAAAAAAAAAQAAAFQAAMOYAAAAHXFhcnRvZF9kZW5zaXR5X2ZsYXRf + bGluZV9mbGFnAAAAAAAAAQAAAAIAAAAMAAAADAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoA + AAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAAAAAADGFjdHVhbF9yYW5nZQAAAAEAAAACAQIAAAAA + AAdjb21tZW50AAAAAAIAAAAdUUFSVE9EIGRlbnNpdHkgZmxhdCBsaW5lIHRlc3QAAAAAAAALY29v + cmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFn + X21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFpbAAAAAAAAAtmbGFnX3ZhbHVlcwAA + AAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19u + YW1lAAAAAAAAAgAAAB1RQVJUT0QgZGVuc2l0eSBmbGF0IGxpbmUgdGVzdAAAAAAAAA1zdGFuZGFy + ZF9uYW1lAAAAAAAAAgAAABtmbGF0X2xpbmVfdGVzdF9xdWFsaXR5X2ZsYWcAAAAACXZhbGlkX21h + eAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAAAQEAAAAAAAABAAAAVAAAw+wA + AAAfcWFydG9kX2RlbnNpdHlfZ3Jvc3NfcmFuZ2VfZmxhZwAAAAABAAAAAgAAAAwAAAAHAAAAC2Nv + b3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAALZGFj + X2NvbW1lbnQAAAAAAgAAABVpb29zX3FjX21vZHVsZV9xYXJ0b2QAAAAAAAANZmxhZ19tZWFuaW5n + cwAAAAAAAAIAAAAnUEFTUyBOT1RfRVZBTFVBVEVEIFNVU1BFQ1QgRkFJTCBNSVNTSU5HAAAAAAtm + bGFnX3ZhbHVlcwAAAAABAAAABQECAwQJAAAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAAB1F1 + YWxpdHkAAAAACXZhbGlkX21heAAAAAAAAAEAAAABCQAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAA + AQEAAAAAAAABAAAAVAAAxEAAAAAbcWFydG9kX2RlbnNpdHlfcHJpbWFyeV9mbGFnAAAAAAEAAAAC + AAAADAAAAAcAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBk + ZXB0aAAAAAAAAAtkYWNfY29tbWVudAAAAAACAAAAFWlvb3NfcWNfbW9kdWxlX3FhcnRvZAAAAAAA + AA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAACdQQVNTIE5PVF9FVkFMVUFURUQgU1VTUEVDVCBGQUlM + IE1JU1NJTkcAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAFAQIDBAkAAAAAAAANaW9vc19jYXRlZ29y + eQAAAAAAAAIAAAAHUXVhbGl0eQAAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlk + X21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADElAAAACJxYXJ0b2RfZGVuc2l0eV9yYXRlX29m + X2NoYW5nZV9mbGFnAAAAAAABAAAAAgAAAAwAAAAMAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAA + CgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAQkAAAAAAAAMYWN0dWFsX3JhbmdlAAAAAQAAAAIBAwAA + AAAAB2NvbW1lbnQAAAAAAgAAACJRQVJUT0QgZGVuc2l0eSByYXRlIG9mIGNoYW5nZSB0ZXN0AAAA + AAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAA + AA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFpbAAAAAAAAAtmbGFnX3Zh + bHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJ + bG9uZ19uYW1lAAAAAAAAAgAAACJRQVJUT0QgZGVuc2l0eSByYXRlIG9mIGNoYW5nZSB0ZXN0AAAA + AAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAgcmF0ZV9vZl9jaGFuZ2VfdGVzdF9xdWFsaXR5X2Zs + YWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAA + AAAAAAEAAABUAADE6AAAABlxYXJ0b2RfZGVuc2l0eV9zcGlrZV9mbGFnAAAAAAAAAQAAAAIAAAAM + AAAADAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJ + AAAAAAAADGFjdHVhbF9yYW5nZQAAAAEAAAACAQIAAAAAAAdjb21tZW50AAAAAAIAAAAZUUFSVE9E + IGRlbnNpdHkgc3Bpa2UgdGVzdAAAAAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1 + ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAAEVBhc3MgU3Vz + cGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQMEAAAAAA1pb29zX2NhdGVnb3J5 + AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAGVFBUlRPRCBkZW5zaXR5 + IHNwaWtlIHRlc3QAAAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAXc3Bpa2VfdGVzdF9xdWFs + aXR5X2ZsYWcAAAAACXZhbGlkX21heAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAAB + AAAAAQEAAAAAAAABAAAAVAAAxTwAAAAUcWFydG9kX2xvY2F0aW9uX2ZsYWcAAAABAAAAAgAAAAwA + AAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UAAAABAAAAAgEB + AAAAAAAHY29tbWVudAAAAAACAAAAFFFBUlRPRCBsb2NhdGlvbiB0ZXN0AAAAC2Nvb3JkaW5hdGVz + AAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5n + cwAAAAAAAAIAAAARUGFzcyBTdXNwZWN0IEZhaWwAAAAAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAMB + AwQAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAA + AAIAAAAUUUFSVE9EIGxvY2F0aW9uIHRlc3QAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAAAAAA + CXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADFkAAAAB5xYXJ0b2RfbW9ub3Rvbmlj + X3ByZXNzdXJlX2ZsYWcAAAAAAAEAAAACAAAADAAAAAsAAAALX0NodW5rU2l6ZXMAAAAABAAAAAEA + AAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAdjb21tZW50AAAAAAIAAAAeUUFSVE9E + IG1vbm90b25pYyBwcmVzc3VyZSB0ZXN0AAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxh + dGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNz + IFN1c3BlY3QgRmFpbAAAAAAAAAtmbGFnX3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRl + Z29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAAB5RQVJUT0QgbW9u + b3RvbmljIHByZXNzdXJlIHRlc3QAAAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAB5zZWFfd2F0 + ZXJfcHJlc3N1cmUgc3RhdHVzX2ZsYWcAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQQAAAAAAAAJ + dmFsaWRfbWluAAAAAAAAAQAAAAEBAAAAAAAAAQAAAFQAAMXkAAAAHnFhcnRvZF9wcmVzc3VyZV9m + bGF0X2xpbmVfZmxhZwAAAAAAAQAAAAIAAAAMAAAACwAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAA + AAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAAAAAAB2NvbW1lbnQAAAAAAgAAAB5RQVJUT0Qg + cHJlc3N1cmUgZmxhdCBsaW5lIHRlc3QAAAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0 + aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAAEVBhc3Mg + U3VzcGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQMEAAAAAA1pb29zX2NhdGVn + b3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAHlFBUlRPRCBwcmVz + c3VyZSBmbGF0IGxpbmUgdGVzdAAAAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAG2ZsYXRfbGlu + ZV90ZXN0X3F1YWxpdHlfZmxhZwAAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlk + X21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADGOAAAACBxYXJ0b2RfcHJlc3N1cmVfZ3Jvc3Nf + cmFuZ2VfZmxhZwAAAAEAAAACAAAADAAAAAwAAAALX0NodW5rU2l6ZXMAAAAABAAAAAEAAAAKAAAA + Cl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UAAAABAAAAAgEBAAAAAAAH + Y29tbWVudAAAAAACAAAAIFFBUlRPRCBwcmVzc3VyZSBncm9zcyByYW5nZSB0ZXN0AAAAC2Nvb3Jk + aW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19t + ZWFuaW5ncwAAAAAAAAIAAAARUGFzcyBTdXNwZWN0IEZhaWwAAAAAAAALZmxhZ192YWx1ZXMAAAAA + AQAAAAMBAwQAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFt + ZQAAAAAAAAIAAAAgUUFSVE9EIHByZXNzdXJlIGdyb3NzIHJhbmdlIHRlc3QAAAANc3RhbmRhcmRf + bmFtZQAAAAAAAAIAAAAdZ3Jvc3NfcmFuZ2VfdGVzdF9xdWFsaXR5X2ZsYWcAAAAAAAAJdmFsaWRf + bWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADG + jAAAABxxYXJ0b2RfcHJlc3N1cmVfcHJpbWFyeV9mbGFnAAAAAQAAAAIAAAAMAAAABwAAAAtjb29y + ZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAAC2RhY19j + b21tZW50AAAAAAIAAAAVaW9vc19xY19tb2R1bGVfcWFydG9kAAAAAAAADWZsYWdfbWVhbmluZ3MA + AAAAAAACAAAAJ1BBU1MgTk9UX0VWQUxVQVRFRCBTVVNQRUNUIEZBSUwgTUlTU0lORwAAAAALZmxh + Z192YWx1ZXMAAAAAAQAAAAUBAgMECQAAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAdRdWFs + aXR5AAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQkAAAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEB + AAAAAAAAAQAAAFQAAMbgAAAAI3FhcnRvZF9wcmVzc3VyZV9yYXRlX29mX2NoYW5nZV9mbGFnAAAA + AAEAAAACAAAADAAAAAwAAAALX0NodW5rU2l6ZXMAAAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUA + AAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UAAAABAAAAAgEBAAAAAAAHY29tbWVudAAAAAAC + AAAAI1FBUlRPRCBwcmVzc3VyZSByYXRlIG9mIGNoYW5nZSB0ZXN0AAAAAAtjb29yZGluYXRlcwAA + AAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MA + AAAAAAACAAAAEVBhc3MgU3VzcGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQME + AAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAAC + AAAAHlFBUlRPRCBwcmVzc3VyZSByYXRlIG9mIGNoYW5nZQAAAAAADXN0YW5kYXJkX25hbWUAAAAA + AAACAAAAIHJhdGVfb2ZfY2hhbmdlX3Rlc3RfcXVhbGl0eV9mbGFnAAAACXZhbGlkX21heAAAAAAA + AAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAAAQEAAAAAAAABAAAAVAAAxzQAAAAacWFy + dG9kX3ByZXNzdXJlX3NwaWtlX2ZsYWcAAAAAAAEAAAACAAAADAAAAAwAAAALX0NodW5rU2l6ZXMA + AAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UA + AAABAAAAAgECAAAAAAAHY29tbWVudAAAAAACAAAAGlFBUlRPRCBwcmVzc3VyZSBzcGlrZSB0ZXN0 + AAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAA + AAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFpbAAAAAAAAAtmbGFn + X3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAA + AAAJbG9uZ19uYW1lAAAAAAAAAgAAABpRQVJUT0QgcHJlc3N1cmUgc3Bpa2UgdGVzdAAAAAAADXN0 + YW5kYXJkX25hbWUAAAAAAAACAAAAF3NwaWtlX3Rlc3RfcXVhbGl0eV9mbGFnAAAAAAl2YWxpZF9t + YXgAAAAAAAABAAAAAQQAAAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEBAAAAAAAAAQAAAFQAAMeI + AAAAI3FhcnRvZF9zYWxpbml0eV9jbGltYXRvbG9naWNhbF9mbGFnAAAAAAEAAAACAAAADAAAAAwA + AAALX0NodW5rU2l6ZXMAAAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAA + AAxhY3R1YWxfcmFuZ2UAAAABAAAAAgEBAAAAAAAHY29tbWVudAAAAAACAAAAI1FBUlRPRCBzYWxp + bml0eSBjbGltYXRvbG9naWNhbCB0ZXN0AAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0 + aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAAEVBhc3Mg + U3VzcGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQMEAAAAAA1pb29zX2NhdGVn + b3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAI1FBUlRPRCBzYWxp + bml0eSBjbGltYXRvbG9naWNhbCB0ZXN0AAAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAAB1jbGlt + YXRvbG9neV90ZXN0X3F1YWxpdHlfZmxhZwAAAAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQQAAAAA + AAAJdmFsaWRfbWluAAAAAAAAAQAAAAEBAAAAAAAAAQAAAFQAAMfcAAAAHnFhcnRvZF9zYWxpbml0 + eV9mbGF0X2xpbmVfZmxhZwAAAAAAAQAAAAIAAAAMAAAADAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAA + AQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAAAAAADGFjdHVhbF9yYW5nZQAAAAEAAAAC + AQIAAAAAAAdjb21tZW50AAAAAAIAAAAeUUFSVE9EIHNhbGluaXR5IGZsYXQgbGluZSB0ZXN0AAAA + AAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAA + AA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFpbAAAAAAAAAtmbGFnX3Zh + bHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJ + bG9uZ19uYW1lAAAAAAAAAgAAAB5RQVJUT0Qgc2FsaW5pdHkgZmxhdCBsaW5lIHRlc3QAAAAAAA1z + dGFuZGFyZF9uYW1lAAAAAAAAAgAAABtmbGF0X2xpbmVfdGVzdF9xdWFsaXR5X2ZsYWcAAAAACXZh + bGlkX21heAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAAAQEAAAAAAAABAAAA + VAAAyDAAAAAgcWFydG9kX3NhbGluaXR5X2dyb3NzX3JhbmdlX2ZsYWcAAAABAAAAAgAAAAwAAAAH + AAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAA + AAALZGFjX2NvbW1lbnQAAAAAAgAAABVpb29zX3FjX21vZHVsZV9xYXJ0b2QAAAAAAAANZmxhZ19t + ZWFuaW5ncwAAAAAAAAIAAAAnUEFTUyBOT1RfRVZBTFVBVEVEIFNVU1BFQ1QgRkFJTCBNSVNTSU5H + AAAAAAtmbGFnX3ZhbHVlcwAAAAABAAAABQECAwQJAAAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAAC + AAAAB1F1YWxpdHkAAAAACXZhbGlkX21heAAAAAAAAAEAAAABCQAAAAAAAAl2YWxpZF9taW4AAAAA + AAABAAAAAQEAAAAAAAABAAAAVAAAyIQAAAAccWFydG9kX3NhbGluaXR5X3ByaW1hcnlfZmxhZwAA + AAEAAAACAAAADAAAAAcAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdp + dHVkZSBkZXB0aAAAAAAAAAtkYWNfY29tbWVudAAAAAACAAAAFWlvb3NfcWNfbW9kdWxlX3FhcnRv + ZAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAACdQQVNTIE5PVF9FVkFMVUFURUQgU1VTUEVD + VCBGQUlMIE1JU1NJTkcAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAFAQIDBAkAAAAAAAANaW9vc19j + YXRlZ29yeQAAAAAAAAIAAAAHUXVhbGl0eQAAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAA + CXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADI2AAAACNxYXJ0b2Rfc2FsaW5pdHlf + cmF0ZV9vZl9jaGFuZ2VfZmxhZwAAAAABAAAAAgAAAAwAAAAMAAAAC19DaHVua1NpemVzAAAAAAQA + AAABAAAACgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAQkAAAAAAAAMYWN0dWFsX3JhbmdlAAAAAQAA + AAIBAwAAAAAAB2NvbW1lbnQAAAAAAgAAACNRQVJUT0Qgc2FsaW5pdHkgcmF0ZSBvZiBjaGFuZ2Ug + dGVzdAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0 + aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFpbAAAAAAAAAtm + bGFnX3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIA + AAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAACNRQVJUT0Qgc2FsaW5pdHkgcmF0ZSBvZiBjaGFuZ2Ug + dGVzdAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAgcmF0ZV9vZl9jaGFuZ2VfdGVzdF9xdWFs + aXR5X2ZsYWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEA + AAABAQAAAAAAAAEAAABUAADJLAAAABpxYXJ0b2Rfc2FsaW5pdHlfc3Bpa2VfZmxhZwAAAAAAAQAA + AAIAAAAMAAAADAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAA + AQAAAAEJAAAAAAAADGFjdHVhbF9yYW5nZQAAAAEAAAACAQMAAAAAAAdjb21tZW50AAAAAAIAAAAa + UUFSVE9EIHNhbGluaXR5IHNwaWtlIHRlc3QAAAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUg + bGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAAEVBh + c3MgU3VzcGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQMEAAAAAA1pb29zX2Nh + dGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAGlFBUlRPRCBz + YWxpbml0eSBzcGlrZSB0ZXN0AAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAXc3Bpa2VfdGVz + dF9xdWFsaXR5X2ZsYWcAAAAACXZhbGlkX21heAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4A + AAAAAAABAAAAAQEAAAAAAAABAAAAVAAAyYAAAAAScWFydG9kX3N5bnRheF9mbGFnAAAAAAABAAAA + AgAAAAwAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UAAAAB + AAAAAgEBAAAAAAAHY29tbWVudAAAAAACAAAAElFBUlRPRCBzeW50YXggdGVzdAAAAAAAC2Nvb3Jk + aW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19t + ZWFuaW5ncwAAAAAAAAIAAAAJUGFzcyBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAACAQQA + AAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAAC + AAAAElFBUlRPRCBzeW50YXggdGVzdAAAAAAACXZhbGlkX21heAAAAAAAAAEAAAABBAAAAAAAAAl2 + YWxpZF9taW4AAAAAAAABAAAAAQEAAAAAAAABAAAAVAAAydQAAAAmcWFydG9kX3RlbXBlcmF0dXJl + X2NsaW1hdG9sb2dpY2FsX2ZsYWcAAAAAAAEAAAACAAAADAAAAAwAAAALX0NodW5rU2l6ZXMAAAAA + BAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxhY3R1YWxfcmFuZ2UAAAAB + AAAAAgEBAAAAAAAHY29tbWVudAAAAAACAAAAJlFBUlRPRCB0ZW1wZXJhdHVyZSBjbGltYXRvbG9n + aWNhbCB0ZXN0AAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVk + ZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAABFQYXNzIFN1c3BlY3QgRmFpbAAA + AAAAAAtmbGFnX3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAF + T3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAACZRQVJUT0QgdGVtcGVyYXR1cmUgY2xpbWF0 + b2xvZ2ljYWwgdGVzdAAAAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAHWNsaW1hdG9sb2d5X3Rl + c3RfcXVhbGl0eV9mbGFnAAAAAAAACXZhbGlkX21heAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9t + aW4AAAAAAAABAAAAAQEAAAAAAAABAAAAVAAAyigAAAAhcWFydG9kX3RlbXBlcmF0dXJlX2ZsYXRf + bGluZV9mbGFnAAAAAAAAAQAAAAIAAAAMAAAADAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoA + AAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAAAAAADGFjdHVhbF9yYW5nZQAAAAEAAAACAQIAAAAA + AAdjb21tZW50AAAAAAIAAAAhUUFSVE9EIHRlbXBlcmF0dXJlIGZsYXQgbGluZSB0ZXN0AAAAAAAA + C2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAAN + ZmxhZ19tZWFuaW5ncwAAAAAAAAIAAAARUGFzcyBTdXNwZWN0IEZhaWwAAAAAAAALZmxhZ192YWx1 + ZXMAAAAAAQAAAAMBAwQAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxv + bmdfbmFtZQAAAAAAAAIAAAAhUUFSVE9EIHRlbXBlcmF0dXJlIGZsYXQgbGluZSB0ZXN0AAAAAAAA + DXN0YW5kYXJkX25hbWUAAAAAAAACAAAAG2ZsYXRfbGluZV90ZXN0X3F1YWxpdHlfZmxhZwAAAAAJ + dmFsaWRfbWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEA + AABUAADKfAAAACNxYXJ0b2RfdGVtcGVyYXR1cmVfZ3Jvc3NfcmFuZ2VfZmxhZwAAAAABAAAAAgAA + AAwAAAAMAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAACgAAAApfRmlsbFZhbHVlAAAAAAABAAAA + AQkAAAAAAAAMYWN0dWFsX3JhbmdlAAAAAQAAAAIBAQAAAAAAB2NvbW1lbnQAAAAAAgAAACNRQVJU + T0QgdGVtcGVyYXR1cmUgZ3Jvc3MgcmFuZ2UgdGVzdAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10 + aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAA + ABFQYXNzIFN1c3BlY3QgRmFpbAAAAAAAAAtmbGFnX3ZhbHVlcwAAAAABAAAAAwEDBAAAAAANaW9v + c19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAACNRQVJU + T0QgdGVtcGVyYXR1cmUgZ3Jvc3MgcmFuZ2UgdGVzdAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIA + AAAdZ3Jvc3NfcmFuZ2VfdGVzdF9xdWFsaXR5X2ZsYWcAAAAAAAAJdmFsaWRfbWF4AAAAAAAAAQAA + AAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADK0AAAAB9xYXJ0b2Rf + dGVtcGVyYXR1cmVfcHJpbWFyeV9mbGFnAAAAAAEAAAACAAAADAAAAAcAAAALY29vcmRpbmF0ZXMA + AAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAAtkYWNfY29tbWVudAAA + AAACAAAAFWlvb3NfcWNfbW9kdWxlX3FhcnRvZAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAA + ACdQQVNTIE5PVF9FVkFMVUFURUQgU1VTUEVDVCBGQUlMIE1JU1NJTkcAAAAAC2ZsYWdfdmFsdWVz + AAAAAAEAAAAFAQIDBAkAAAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAHUXVhbGl0eQAAAAAJ + dmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEA + AABUAADLJAAAACZxYXJ0b2RfdGVtcGVyYXR1cmVfcmF0ZV9vZl9jaGFuZ2VfZmxhZwAAAAAAAQAA + AAIAAAAMAAAADAAAAAtfQ2h1bmtTaXplcwAAAAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAA + AQAAAAEJAAAAAAAADGFjdHVhbF9yYW5nZQAAAAEAAAACAQMAAAAAAAdjb21tZW50AAAAAAIAAAAm + UUFSVE9EIHRlbXBlcmF0dXJlIHJhdGUgb2YgY2hhbmdlIHRlc3QAAAAAAAtjb29yZGluYXRlcwAA + AAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmluZ3MA + AAAAAAACAAAAEVBhc3MgU3VzcGVjdCBGYWlsAAAAAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAADAQME + AAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAAC + AAAAJlFBUlRPRCB0ZW1wZXJhdHVyZSByYXRlIG9mIGNoYW5nZSB0ZXN0AAAAAAANc3RhbmRhcmRf + bmFtZQAAAAAAAAIAAAAgcmF0ZV9vZl9jaGFuZ2VfdGVzdF9xdWFsaXR5X2ZsYWcAAAAJdmFsaWRf + bWF4AAAAAAAAAQAAAAEEAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAQAAAAAAAAEAAABUAADL + eAAAAB1xYXJ0b2RfdGVtcGVyYXR1cmVfc3Bpa2VfZmxhZwAAAAAAAAEAAAACAAAADAAAAAwAAAAL + X0NodW5rU2l6ZXMAAAAABAAAAAEAAAAKAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABCQAAAAAAAAxh + Y3R1YWxfcmFuZ2UAAAABAAAAAgECAAAAAAAHY29tbWVudAAAAAACAAAAHVFBUlRPRCB0ZW1wZXJh + dHVyZSBzcGlrZSB0ZXN0AAAAAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBs + b25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5ncwAAAAAAAAIAAAARUGFzcyBTdXNwZWN0 + IEZhaWwAAAAAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAMBAwQAAAAADWlvb3NfY2F0ZWdvcnkAAAAA + AAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAdUUFSVE9EIHRlbXBlcmF0dXJl + IHNwaWtlIHRlc3QAAAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAXc3Bpa2VfdGVzdF9xdWFs + aXR5X2ZsYWcAAAAACXZhbGlkX21heAAAAAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAAB + AAAAAQEAAAAAAAABAAAAVAAAy8wAAAAacWFydG9kX3RpbWluZ19hbmRfZ2FwX2ZsYWcAAAAAAAEA + AAACAAAADAAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAEJAAAAAAAADGFjdHVhbF9yYW5nZQAA + AAEAAAACAQEAAAAAAAdjb21tZW50AAAAAAIAAAAaUUFSVE9EIHRpbWluZyBhbmQgZ2FwIHRlc3QA + AAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAA + AAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAACVBhc3MgRmFpbAAAAAAAAAtmbGFnX3ZhbHVlcwAA + AAABAAAAAgEEAAAAAAANaW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19u + YW1lAAAAAAAAAgAAABpRQVJUT0QgdGltaW5nIGFuZCBnYXAgdGVzdAAAAAAACXZhbGlkX21heAAA + AAAAAAEAAAABBAAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAAAQEAAAAAAAABAAAAVAAAzCAAAAAI + c2FsaW5pdHkAAAABAAAAAgAAAAwAAAAQAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAACgAAAApf + RmlsbFZhbHVlAAAAAAAFAAAAAcR5wAAAAAAMYWN0dWFsX3JhbmdlAAAABQAAAAJB/wtoQgM8rAAA + ABNhbmNpbGxhcnlfdmFyaWFibGVzAAAAAAIAAACNcWFydG9kX3NhbGluaXR5X2NsaW1hdG9sb2dp + Y2FsX2ZsYWcgcWFydG9kX3NhbGluaXR5X2ZsYXRfbGluZV9mbGFnIHFhcnRvZF9zYWxpbml0eV9y + YXRlX29mX2NoYW5nZV9mbGFnIHFhcnRvZF9zYWxpbml0eV9zcGlrZV9mbGFnIHNhbGluaXR5X3Fj + AAAAAAAAD2NvbG9yQmFyTWF4aW11bQAAAAAGAAAAAUBCgAAAAAAAAAAAD2NvbG9yQmFyTWluaW11 + bQAAAAAGAAAAAUA+AAAAAAAAAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBs + b25naXR1ZGUgZGVwdGgAAAAAAAAKaW5zdHJ1bWVudAAAAAAAAgAAAA5pbnN0cnVtZW50X2N0ZAAA + AAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAACFNhbGluaXR5AAAACWxvbmdfbmFtZQAAAAAAAAIA + AAAcU2VhIFdhdGVyIFByYWN0aWNhbCBTYWxpbml0eQAAABBvYnNlcnZhdGlvbl90eXBlAAAAAgAA + AApjYWxjdWxhdGVkAAAAAAAIcGxhdGZvcm0AAAACAAAACHBsYXRmb3JtAAAADXN0YW5kYXJkX25h + bWUAAAAAAAACAAAAHHNlYV93YXRlcl9wcmFjdGljYWxfc2FsaW5pdHkAAAAFdW5pdHMAAAAAAAAE + AAAAAQAAAAEAAAAJdmFsaWRfbWF4AAAAAAAABQAAAAFCIAAAAAAACXZhbGlkX21pbgAAAAAAAAUA + AAABAAAAAAAAAAUAAAFMAADMdAAAAAtzYWxpbml0eV9xYwAAAAABAAAAAgAAAAwAAAAKAAAAC19D + aHVua1NpemVzAAAAAAQAAAABAAAACgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAYEAAAAAAAALY29v + cmRpbmF0ZXMAAAAAAgAAAB10aW1lIGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFn + X21lYW5pbmdzAAAAAAAAAgAAAKBub19xY19wZXJmb3JtZWQgZ29vZF9kYXRhIHByb2JhYmx5X2dv + b2RfZGF0YSBiYWRfZGF0YV90aGF0X2FyZV9wb3RlbnRpYWxseV9jb3JyZWN0YWJsZSBiYWRfZGF0 + YSB2YWx1ZV9jaGFuZ2VkIG5vdF91c2VkIG5vdF91c2VkIGludGVycG9sYXRlZF92YWx1ZSBtaXNz + aW5nX3ZhbHVlAAAAC2ZsYWdfdmFsdWVzAAAAAAEAAAAKAAECAwQFBgcICQAAAAAADWlvb3NfY2F0 + ZWdvcnkAAAAAAAACAAAABU90aGVyAAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAVc2FsaW5pdHkg + UXVhbGl0eSBGbGFnAAAAAAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAKHNlYV93YXRlcl9wcmFj + dGljYWxfc2FsaW5pdHkgc3RhdHVzX2ZsYWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAA + CXZhbGlkX21pbgAAAAAAAAEAAAABAAAAAAAAAAEAAABUAADNwAAAAAt0ZW1wZXJhdHVyZQAAAAAB + AAAAAgAAAAwAAAAQAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAACgAAAApfRmlsbFZhbHVlAAAA + AAAFAAAAAcR5wAAAAAAMYWN0dWFsX3JhbmdlAAAABQAAAAJBCY7zQWUznAAAABNhbmNpbGxhcnlf + dmFyaWFibGVzAAAAAAIAAADAcWFydG9kX3RlbXBlcmF0dXJlX2NsaW1hdG9sb2dpY2FsX2ZsYWcg + cWFydG9kX3RlbXBlcmF0dXJlX2ZsYXRfbGluZV9mbGFnIHFhcnRvZF90ZW1wZXJhdHVyZV9ncm9z + c19yYW5nZV9mbGFnIHFhcnRvZF90ZW1wZXJhdHVyZV9yYXRlX29mX2NoYW5nZV9mbGFnIHFhcnRv + ZF90ZW1wZXJhdHVyZV9zcGlrZV9mbGFnIHRlbXBlcmF0dXJlX3FjAAAAD2NvbG9yQmFyTWF4aW11 + bQAAAAAGAAAAAUBAAAAAAAAAAAAAD2NvbG9yQmFyTWluaW11bQAAAAAGAAAAAQAAAAAAAAAAAAAA + C2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAAK + aW5zdHJ1bWVudAAAAAAAAgAAAA5pbnN0cnVtZW50X2N0ZAAAAAAADWlvb3NfY2F0ZWdvcnkAAAAA + AAACAAAAC1RlbXBlcmF0dXJlAAAAAAlsb25nX25hbWUAAAAAAAACAAAAFVNlYSBXYXRlciBUZW1w + ZXJhdHVyZQAAAAAAABBvYnNlcnZhdGlvbl90eXBlAAAAAgAAAAhtZWFzdXJlZAAAAAhwbGF0Zm9y + bQAAAAIAAAAIcGxhdGZvcm0AAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAVc2VhX3dhdGVyX3Rl + bXBlcmF0dXJlAAAAAAAABXVuaXRzAAAAAAAAAgAAAAdDZWxzaXVzAAAAAAl2YWxpZF9tYXgAAAAA + AAAFAAAAAUIgAAAAAAAJdmFsaWRfbWluAAAAAAAABQAAAAHAoAAAAAAABQAAAUwAAM4UAAAADnRl + bXBlcmF0dXJlX3FjAAAAAAABAAAAAgAAAAwAAAAKAAAAC19DaHVua1NpemVzAAAAAAQAAAABAAAA + CgAAAApfRmlsbFZhbHVlAAAAAAABAAAAAYEAAAAAAAALY29vcmRpbmF0ZXMAAAAAAgAAAB10aW1l + IGxhdGl0dWRlIGxvbmdpdHVkZSBkZXB0aAAAAAAAAA1mbGFnX21lYW5pbmdzAAAAAAAAAgAAAKBu + b19xY19wZXJmb3JtZWQgZ29vZF9kYXRhIHByb2JhYmx5X2dvb2RfZGF0YSBiYWRfZGF0YV90aGF0 + X2FyZV9wb3RlbnRpYWxseV9jb3JyZWN0YWJsZSBiYWRfZGF0YSB2YWx1ZV9jaGFuZ2VkIG5vdF91 + c2VkIG5vdF91c2VkIGludGVycG9sYXRlZF92YWx1ZSBtaXNzaW5nX3ZhbHVlAAAAC2ZsYWdfdmFs + dWVzAAAAAAEAAAAKAAECAwQFBgcICQAAAAAADWlvb3NfY2F0ZWdvcnkAAAAAAAACAAAABU90aGVy + AAAAAAAACWxvbmdfbmFtZQAAAAAAAAIAAAAYdGVtcGVyYXR1cmUgUXVhbGl0eSBGbGFnAAAADXN0 + YW5kYXJkX25hbWUAAAAAAAACAAAAIXNlYV93YXRlcl90ZW1wZXJhdHVyZSBzdGF0dXNfZmxhZwAA + AAAAAAl2YWxpZF9tYXgAAAAAAAABAAAAAQkAAAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEAAAAA + AAAAAQAAAFQAAM9gAAAAB3RpbWVfcWMAAAAAAQAAAAIAAAAMAAAACgAAAAtfQ2h1bmtTaXplcwAA + AAAEAAAAAQAAAAoAAAAKX0ZpbGxWYWx1ZQAAAAAAAQAAAAGBAAAAAAAAC2Nvb3JkaW5hdGVzAAAA + AAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5ncwAA + AAAAAAIAAACgbm9fcWNfcGVyZm9ybWVkIGdvb2RfZGF0YSBwcm9iYWJseV9nb29kX2RhdGEgYmFk + X2RhdGFfdGhhdF9hcmVfcG90ZW50aWFsbHlfY29ycmVjdGFibGUgYmFkX2RhdGEgdmFsdWVfY2hh + bmdlZCBub3RfdXNlZCBub3RfdXNlZCBpbnRlcnBvbGF0ZWRfdmFsdWUgbWlzc2luZ192YWx1ZQAA + AAtmbGFnX3ZhbHVlcwAAAAABAAAACgABAgMEBQYHCAkAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAA + AgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUAAAAAAAACAAAAEXRpbWUgUXVhbGl0eSBGbGFnAAAA + AAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAEHRpbWUgc3RhdHVzX2ZsYWcAAAAJdmFsaWRfbWF4 + AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAAAAEAAAABAAAAAAAAAAEAAABUAADPtAAA + AAp0aW1lX3V2X3FjAAAAAAABAAAAAgAAAAwAAAAJAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAA + AAAAAAtjb29yZGluYXRlcwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAA + AAAADWZsYWdfbWVhbmluZ3MAAAAAAAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJv + YmFibHlfZ29vZF9kYXRhIGJhZF9kYXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxl + IGJhZF9kYXRhIHZhbHVlX2NoYW5nZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3Zh + bHVlIG1pc3NpbmdfdmFsdWUAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAAN + aW9vc19jYXRlZ29yeQAAAAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAABR0 + aW1lX3V2IFF1YWxpdHkgRmxhZwAAAA1zdGFuZGFyZF9uYW1lAAAAAAAAAgAAABB0aW1lIHN0YXR1 + c19mbGFnAAAACXZhbGlkX21heAAAAAAAAAEAAAABCQAAAAAAAAl2YWxpZF9taW4AAAAAAAABAAAA + AQAAAAAAAAABAAAAVAAA0AgAAAAEdV9xYwAAAAEAAAACAAAADAAAAAkAAAAKX0ZpbGxWYWx1ZQAA + AAAAAQAAAAGBAAAAAAAAC2Nvb3JkaW5hdGVzAAAAAAIAAAAddGltZSBsYXRpdHVkZSBsb25naXR1 + ZGUgZGVwdGgAAAAAAAANZmxhZ19tZWFuaW5ncwAAAAAAAAIAAACgbm9fcWNfcGVyZm9ybWVkIGdv + b2RfZGF0YSBwcm9iYWJseV9nb29kX2RhdGEgYmFkX2RhdGFfdGhhdF9hcmVfcG90ZW50aWFsbHlf + Y29ycmVjdGFibGUgYmFkX2RhdGEgdmFsdWVfY2hhbmdlZCBub3RfdXNlZCBub3RfdXNlZCBpbnRl + cnBvbGF0ZWRfdmFsdWUgbWlzc2luZ192YWx1ZQAAAAtmbGFnX3ZhbHVlcwAAAAABAAAACgABAgME + BQYHCAkAAAAAAA1pb29zX2NhdGVnb3J5AAAAAAAAAgAAAAVPdGhlcgAAAAAAAAlsb25nX25hbWUA + AAAAAAACAAAADnUgUXVhbGl0eSBGbGFnAAAAAAANc3RhbmRhcmRfbmFtZQAAAAAAAAIAAAAnZWFz + dHdhcmRfc2VhX3dhdGVyX3ZlbG9jaXR5IHN0YXR1c19mbGFnAAAAAAl2YWxpZF9tYXgAAAAAAAAB + AAAAAQkAAAAAAAAJdmFsaWRfbWluAAAAAAAAAQAAAAEAAAAAAAAAAQAAAFQAANBcAAAABHZfcWMA + AAABAAAAAgAAAAwAAAAJAAAACl9GaWxsVmFsdWUAAAAAAAEAAAABgQAAAAAAAAtjb29yZGluYXRl + cwAAAAACAAAAHXRpbWUgbGF0aXR1ZGUgbG9uZ2l0dWRlIGRlcHRoAAAAAAAADWZsYWdfbWVhbmlu + Z3MAAAAAAAACAAAAoG5vX3FjX3BlcmZvcm1lZCBnb29kX2RhdGEgcHJvYmFibHlfZ29vZF9kYXRh + IGJhZF9kYXRhX3RoYXRfYXJlX3BvdGVudGlhbGx5X2NvcnJlY3RhYmxlIGJhZF9kYXRhIHZhbHVl + X2NoYW5nZWQgbm90X3VzZWQgbm90X3VzZWQgaW50ZXJwb2xhdGVkX3ZhbHVlIG1pc3NpbmdfdmFs + dWUAAAALZmxhZ192YWx1ZXMAAAAAAQAAAAoAAQIDBAUGBwgJAAAAAAANaW9vc19jYXRlZ29yeQAA + AAAAAAIAAAAFT3RoZXIAAAAAAAAJbG9uZ19uYW1lAAAAAAAAAgAAAA52IFF1YWxpdHkgRmxhZwAA + AAAADXN0YW5kYXJkX25hbWUAAAAAAAACAAAAKG5vcnRod2FyZF9zZWFfd2F0ZXJfdmVsb2NpdHkg + c3RhdHVzX2ZsYWcAAAAJdmFsaWRfbWF4AAAAAAAAAQAAAAEJAAAAAAAACXZhbGlkX21pbgAAAAAA + AAEAAAABAAAAAAAAAAEAAABUAADQsGFtZWxpYS0yMDE4MDUwMVQwMDAwNDgwMTkzNAAAAAVvAAAF + cAAABXEAAAVyAAAFcwAABXQAAAV1Qda+XNMfk4BB1r5dQ5c7fEHWvl2suFV9Qda+XvS4uZpB1r5f + aVtBpUHWvl/XxxU+Qda+YEY8ortAQr0YRA4xXEBCvRcAt34BQEK9FH3kevdAQrzxE9u0F0BCvO2y + 7KTgQEK86Rwmml1AQrzfggEHEcBSxMn7L4v0wFLE3X+lwdjAUsTtoYSNkcBSxO/clojywFLE+7ZJ + RsbAUsUQVYztr8BSxRfS6pv3wI84AAAAAADAjzgAAAAAAMCPOAAAAAAAwI84AAAAAADAjzgAAAAA + AMCPOAAAAAAAwI84AAAAAADAjzgAAAAAAMCPOAAAAAAAwI84AAAAAADAjzgAAAAAAMCPOAAAAAAA + wI84AAAAAADAjzgAAAAAAH/4AAAAAAAAf/gAAAAAAAB/+AAAAAAAAH/4AAAAAAAAf/gAAAAAAAB/ + +AAAAAAAAH/4AAAAAAAAwI84AAAAAADAjzgAAAAAAMCPOAAAAAAAwI84AAAAAADAjzgAAAAAAMCP + OAAAAAAAwI84AAAAAADAjzgAAAAAAMCPOAAAAAAAwI84AAAAAADAjzgAAAAAAMCPOAAAAAAAwI84 + AAAAAADAjzgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAANAAAADQAAAAwAAAAO + AAAADAAAAAsAAAAIQar1w0GfrhRBkD1xQYThSEFz1wpBVPXDQTo9cUEhHrhA/64UQM5mZkCgUexA + Y9cKQB9cKUGrmZpBoFHsQY+ZmkGDmZpBbzMzQUz1w0EyZmZBGzMzQP4Ue0DUKPZAnXCkQG4Ue0A2 + ZmZBwPXDQbbMzUGoj1xBngAAQY5mZkGC9cNBb1wpQU3rhUEtcKRBB64UQKlHrkA5mZpBua4UQauF + H0Ggo9dBlLhSQYVHrkF0zM1BXwo9QUAo9kEnhR9BCUeuQOTMzUCo9cNAeZmaQDeuFEGouFJBmKPX + QYgo9kF31wpBWPXDQUAo9kEmPXFBBR64QNvXCkCczM1AYo9cQCmZmkGbHrhBj0euQYKj10FlHrhB + TPXDQTOFH0ESPXFA89cKQLOFH0CEzM1AMKPXQZZ64UF6FHtBVCj2QTAo9kED1wpAyUeuQG1wpD/m + ZmZAXhhAQF4UUUBeD5FAXg6/QF4EwEBh+5RAbVtXQHcmwEB4lJpAeMCDQHjdWUB48dRAeQDnQF4Y + akBeGGpAXhNVQF4PE0BeqZNAa1ynQG7Ss0B4ZBtAeLy+QHje00B4+bFAeQrlQHkWsUBeLeBAXi2M + QF4lhUBeGrVAXhYeQF4RXkBenUlAactoQG8GeUB4WSFAeL80QHjgdkBeJtVAXitBQF4pdEBeJYVA + XhTPQF4QjEBeL9dAZnMuQGzIYEB4YVJAeLsbQHjwMEB5C41AeRzrQF4bCUBeI2RAXiHqQF4ai0Be + EYhAYqwyQGxPykB3/wRAeLIYQHjuY0B5DoBAeSeRQF4kX0BeHk9AXiKSQF4mgUBlJ2dAbD+SQHfl + yUB4t4BAePboQHkTQEB5KbNAXiWvQF4rQUBh/tpAbD+8QHhlQUB4w/RAeP7FQHklxIGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBAESAMgBEgDG1RIAw+ESAL/FEgC3uRIAqykSAB4BEf/YqRH/zp0R/ + 8zdEf/KvRH/yj0R/8kVEgDHgRIAxmESAMQZEgC+cRIAsIUSADjtEgAfMRH/yKUR/83dEf/LeRH/y + mkR/8jtEf/JKRIAyIUSAMfJEgDGERIAxNUSAMNpEgC9uRIAr6kSAEl9EgAduRH/xNER/8bdEf/Gi + RIAx8kSAMZlEgDFERIAxAESAMGhEgC+vRIAtCESAHpBEgAh2RH/xnER/8lhEf/J7RH/yC0R/8WxE + gDFwRIAw8ESAMIVEgC/lRIAtyUSAJwlEgAjHRH/yGUR/8bFEf/IpRH/yIER/8TZEgDDgRIAwj0SA + MANEgC1wRIAfXESACbBEf/Q0RH/yU0R/8WxEf/I8RH/ykUSAMLREgC9kRIAp7USACd9Ef/CpRH/x + 6kR/8bZEf/DhgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEAgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYEAf39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f38AgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYEAgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEAgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEA + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEAf39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f38AQEK9 + F/MgJtVAQr0YAW/1LUBCvRgPt90UQEK9GB3/y7NAQr0YKj1Yv0BCvRg4hJGtQEK9GEbL4qdAQr0Y + UwkGJ0BCvRhhUhafQEK9GG+b73tAQr0Ye9m180BCvRiIKy8QQEK9GJSCvGRAQr0XC4iR+UBCvRcJ + +cUBQEK9FwgrcDtAQr0XBl3SxEBCvRcE0jaFQEK9FwMEqphAQr0XATcQgEBCvRb/q05RQEK9Fv3d + wbdAQr0W/BAqUkBCvRb6Qr9HQEK9FvhzhHpAQr0W9uYGmUBCvRRE3VXWQEK9FE8s1AZAQr0UWXjk + zkBCvRRjxJNUQEK9FG4QCBpAQr0UeFyWFEBCvRSBNFIYQEK9FIuHgRtAQr0UlHIgBEBCvRSewkbA + QEK9FKkNvW5AQr0Us1mBLEBCvPEVi918QEK88RVIyShAQrzxFQXXo0BCvPEUzHRnQEK88RSJgOBA + QrzxFEaGxUBCvPEUDSXWQEK88RPTTtNAQrzxE5BbqkBCvPETTVlXQEK88RMKZeNAQrzxEsd4YUBC + vPESjdvVQEK88RJUGgFAQrzt+5axTkBCvO3u/R/sQEK87eJpljFAQrzt1eUr40BCvO3JYYKzQEK8 + 7bzdcVJAQrztsiP8BEBCvO2loZ4AQEK87ZkfI+tAQrztjJw0yEBCvO2B0iFeQEK87XU92gpAQrzp + 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 + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgQCBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgQCBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgQCBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgQAJCQkJCQkJCQkJCQkJCQkJCQkJCQkJ + CQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJ + CQkJCQAJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJ + CQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQB/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/fwABAQEBAwMDAwEBAQEBAQEBAQMDAwMBAQEBAQEB + AQEBAQMDAwMBAQEBAQEBAQMDAwMBAQEBAQEBAQMDAwMBAQEBAQEBAwMDAwEBAQEBAwMDAwEBAQAC + AQEBAQEBAQEBAQECAgEBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAQECAgEBAQEB + AQEBAQECAgEBAQEBAQEBAQICAQEBAQEBAgABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQABAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQICAgEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQB/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/fwB/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/fwABAQEBAQMDAwEBAQEBAQEBAwMDAwMBAQEBAQEBAQEBAwMDAwMB + AQEBAQEBAQMDAwMBAQEBAQEBAQMDAwMBAQEBAQEBAwMDAwEBAQEBAwMDAwEBAQACAQEBAQEBAQEB + AQECAgEBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAQECAgEBAQEBAQEBAQECAgEB + AQEBAQEBAQICAQEBAQEBAgABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQAJCQkJCQkJCQkJCQkJ + CQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJ + CQkJCQkJCQkJCQkJCQAJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJ + CQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQABAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQB/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/fwABAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQACAQEBAQEBAQEBAQECAgEBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAgIBAQEBAQEBAQEB + AQECAgEBAQEBAQEBAQECAgEBAQEBAQEBAQICAQEBAQEBAgABAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQICAgEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQB/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + fwB/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/fwABAQEBAQMDAQEBAQEBAQEBAwMDAwMBAQEBAQEB + AQEBAwMDAwMBAQEBAQEBAwMDAwEBAQEBAQEBAQEDAwEBAQEBAQEBAwMDAQEBAQEBAQMDAwEBAQAC + AQEBAQEBAQEBAQECAgEBAQEBAQMBAQEBAgIBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAQECAgEBAQEB + AQEBAQECAgEBAQEBAQEBAQICAQEBAwEBAgABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQABAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQICAgEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQABAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQB/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/ + f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/fwABAQEBAwMDAwEB + AQEBAQEBAQMDAwMBAQEBAQEBAQEBAQMDAwMBAQEBAQEBAQMDAwMBAQEBAQEBAQMDAwMBAQEBAQEB + AwMDAwEBAQEBAwMDAwEBAQACAQEBAQEBAQEBAQECAgEBAQEBAQEBAQEBAgIBAQEBAQEBAQEBAgIB + AQEBAQEBAQEBAQECAgEBAQEBAQEBAQECAgEBAQEBAQEBAQICAQEBAQEBAgABAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEB + AQEBAQEBAQEBAQEBAQBCAzysQgM5a0IDKYRCAw3oQgLPs0IC5YRB/6oIQf9OCEH/eJhB/4VFQf+J + 7EH/mQBB/54JQgM4FUIDNZpCAyxPQgMDeUICp1NCAGWtQgAStUH/LKFB/35vQf+BXkH/kCpB/5Ra + Qf+hgEIDNbNCAzXLQgMvRUIDKjhCAyd1QgL+L0ICnr9CALgTQgAPUkH/FrZB/2ChQf99MkIDMzdC + AzDdQgMsO0IDKkFCAx4DQgMMAEICvOdCAd/TQf/MVEH/JOlB/2IZQf+GX0H/jQFB/4fwQgMrbUID + JX5CAyF/QgMTUUIC1MZCAoN4Qf/BqUH/GuxB/02yQf9/XUH/k7RB/4Y8QgMh6EIDHfVCAxPpQgLH + pEIBzW1B//AiQf9OUEH/XEFB/2QsQf+S50H/sFtCAx8kQgMDiUICyVBB//4IQf8LakH/XwRB/4C/ + Qf+DJoGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBAEEJjvNBCZSvQQnPq0EKR0VBCzqTQRYP+UFD + 6XlBYBysQWOvuEFkDYRBZFTKQWRrukFki0RBCaMFQQmwikEJznBBCnWOQQ2800E7q59BRuVgQWPe + AUFkEMtBZGp/QWSURkFkvAJBZL53QQnkj0EJ5clBCe7MQQnop0EJ64VBCpMMQQ2/SEE1yLRBR4bC + QWP2/UFkZMNBZIA0QQndL0EJ92ZBCgkDQQoJbEEKEtdBClc/QQwImkEnJ7tBQf7FQWPrHEFkUlRB + ZJEAQWTPQkFlDYRBCeGxQQoW8EEKKCRBClOPQQtMMEEZscRBQMUFQWLzTUFka1FBZJ2yQWTH40Fl + L4NBCiiNQQora0EKZf5BC77gQSPTw0FAKV9BYi+DQWRVm0Fk9cNBZNXQQWTPQkEKOVhBCsaoQRaU + r0FACdVBZDRuQWRz60Fkye9BZTOcgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEAgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYEAgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEAgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYEAgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGB + gYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEA headers: Access-Control-Allow-Origin: - '*' @@ -231,11 +1016,11 @@ interactions: Content-Type: - application/x-netcdf Date: - - Tue, 21 May 2024 19:48:37 GMT + - Wed, 23 Oct 2024 07:40:09 GMT Last-Modified: - - Tue, 21 May 2024 19:48:37 GMT + - Wed, 23 Oct 2024 07:40:09 GMT Server: - - nginx/1.18.0 + - nginx/1.20.1 Strict-Transport-Security: - max-age=0 Transfer-Encoding: diff --git a/compliance_checker/tests/test_protocols.py b/compliance_checker/tests/test_protocols.py index 850b9b33..8864bbde 100644 --- a/compliance_checker/tests/test_protocols.py +++ b/compliance_checker/tests/test_protocols.py @@ -32,7 +32,6 @@ def test_netcdf_content_type(cs): assert ds is not None -@pytest.mark.vcr() def test_erddap(cs): """ Tests that a connection can be made to ERDDAP's GridDAP From 3a2637e92f7cc5274189c2e4d6762c745c672db3 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 4 Nov 2024 21:32:36 +0000 Subject: [PATCH 35/49] [pre-commit.ci] pre-commit autoupdate MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit updates: - [github.com/astral-sh/ruff-pre-commit: v0.7.0 → v0.7.2](https://github.com/astral-sh/ruff-pre-commit/compare/v0.7.0...v0.7.2) - [github.com/tox-dev/pyproject-fmt: 2.4.3 → v2.5.0](https://github.com/tox-dev/pyproject-fmt/compare/2.4.3...v2.5.0) --- .pre-commit-config.yaml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index e3d15781..de9c3fda 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,12 +31,12 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.7.0 + rev: v0.7.2 hooks: - id: ruff - repo: https://github.com/tox-dev/pyproject-fmt - rev: 2.4.3 + rev: v2.5.0 hooks: - id: pyproject-fmt From 089878376be9496c2bef2be223548c582e23094d Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Tue, 5 Nov 2024 13:17:36 +0100 Subject: [PATCH 36/49] add ruff fix --- .pre-commit-config.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index de9c3fda..f8a40cd9 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -34,6 +34,7 @@ repos: rev: v0.7.2 hooks: - id: ruff + args: ["--fix", "--show-fixes"] - repo: https://github.com/tox-dev/pyproject-fmt rev: v2.5.0 From 11da3f893e4dea1d1bccaff1ed316d9320863397 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Tue, 5 Nov 2024 13:17:39 +0100 Subject: [PATCH 37/49] fix Python deprecations --- compliance_checker/__init__.py | 3 ++- compliance_checker/base.py | 4 ++-- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/compliance_checker/__init__.py b/compliance_checker/__init__.py index f5975758..ca60116c 100644 --- a/compliance_checker/__init__.py +++ b/compliance_checker/__init__.py @@ -1,6 +1,7 @@ +from collections.abc import Generator from contextlib import contextmanager from tempfile import NamedTemporaryFile -from typing import BinaryIO, Generator +from typing import BinaryIO try: from ._version import __version__ diff --git a/compliance_checker/base.py b/compliance_checker/base.py index 0f276754..1fea8c43 100644 --- a/compliance_checker/base.py +++ b/compliance_checker/base.py @@ -8,7 +8,7 @@ import pprint import re import warnings -from collections import defaultdict +from collections.abc import defaultdict from functools import wraps from io import StringIO @@ -24,7 +24,7 @@ # Python 3.5+ should work, also have a fallback try: - from typing import Pattern + from re import Pattern re_pattern_type = Pattern except ImportError: From be976c6580d3be83d6aed44d4990ba26e00561b9 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Tue, 5 Nov 2024 13:17:55 +0100 Subject: [PATCH 38/49] py39 compat --- compliance_checker/base.py | 2 +- pyproject.toml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/compliance_checker/base.py b/compliance_checker/base.py index 1fea8c43..f7189be5 100644 --- a/compliance_checker/base.py +++ b/compliance_checker/base.py @@ -8,7 +8,7 @@ import pprint import re import warnings -from collections.abc import defaultdict +from collections import defaultdict from functools import wraps from io import StringIO diff --git a/pyproject.toml b/pyproject.toml index 17e8b215..696cf98f 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -89,7 +89,7 @@ write_to_template = "__version__ = '{version}'" tag_regex = "^(?Pv)?(?P[^\\+]+)(?P.*)?$" [tool.ruff] -target-version = "py38" +target-version = "py39" line-length = 200 exclude = [ From 7a82c8c6e9c044e801d835ff5ee3ec6496511f04 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 7 Nov 2024 13:24:06 +0100 Subject: [PATCH 39/49] remove compare_unit_types --- compliance_checker/cf/util.py | 30 ------------------------------ 1 file changed, 30 deletions(-) diff --git a/compliance_checker/cf/util.py b/compliance_checker/cf/util.py index 9443eb33..1b592829 100644 --- a/compliance_checker/cf/util.py +++ b/compliance_checker/cf/util.py @@ -392,36 +392,6 @@ def is_vertical_coordinate(var_name, var): return satisfied -def compare_unit_types(specified, reference): - """ - Compares two unit strings via UDUnits - - :param str specified: The specified unit - :param str reference: The reference unit which to compare against - - """ - msgs = [] - err_flag = False - try: - specified_unit = Unit(specified) - except ValueError: - msgs.append(f"Specified conversion unit f{specified} may not be valid UDUnits") - err_flag = True - - try: - reference_unit = Unit(reference) - except ValueError: - msgs.append(f"Specified conversion unit f{reference} may not be valid UDUnits") - err_flag = True - - if err_flag: - return msgs - - unit_convertible = specified_unit.is_convertible(reference_unit) - fail_msg = [f'Units "{specified}" are not convertible to "{reference}"'] - return msgs if unit_convertible else fail_msg - - def string_from_var_type(variable): if isinstance(variable, str): return variable[:] From bf6c6ffeac4b8e751b0fb6890b7d797d8071ba52 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Tue, 12 Nov 2024 16:09:54 +0100 Subject: [PATCH 40/49] update pre-commits --- .pre-commit-config.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index f8a40cd9..7f1c47c6 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.7.2 + rev: v0.7.3 hooks: - id: ruff args: ["--fix", "--show-fixes"] From 066a8268d440f83810df042e3686493532a6b012 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Wed, 13 Nov 2024 10:28:00 +0100 Subject: [PATCH 41/49] .is_long_time_interval() is deprecated and will be removed --- compliance_checker/cf/cf_1_6.py | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/compliance_checker/cf/cf_1_6.py b/compliance_checker/cf/cf_1_6.py index 3979777e..c77d5ea4 100644 --- a/compliance_checker/cf/cf_1_6.py +++ b/compliance_checker/cf/cf_1_6.py @@ -1864,8 +1864,7 @@ def check_time_coordinate(self, ds): ret_val.append(result) # IMPLEMENTATION CONFORMANCE 4.4 RECOMMENDED 2/2 # catch non-recommended months or years time interval - unit = Unit(variable.units) - if unit.is_long_time_interval(): + if any(unit in variable.units for unit in ("months", "years")): message = f"Using relative time interval of months or years is not recommended for coordinate variable {variable.name}" result = Result( BaseCheck.MEDIUM, From 37ef414d0bfed80cc95fd2038110afb72f4f5291 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 14 Nov 2024 21:08:52 +0100 Subject: [PATCH 42/49] remove unused argumente --- compliance_checker/cfutil.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/compliance_checker/cfutil.py b/compliance_checker/cfutil.py index 8bcce331..22336140 100644 --- a/compliance_checker/cfutil.py +++ b/compliance_checker/cfutil.py @@ -2028,7 +2028,7 @@ def guess_feature_type(nc, variable): return "reduced-grid" -def units_convertible(units1, units2, reftimeistime=True): +def units_convertible(units1, units2): """ Return True if a Unit representing the string units1 can be converted to a Unit representing the string units2, else False. From 8bf1e351c8fb6a2f7990d397a26d7d53845a8cb8 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 15 Nov 2024 02:35:54 +0000 Subject: [PATCH 43/49] Bump codecov/codecov-action from 4 to 5 Bumps [codecov/codecov-action](https://github.com/codecov/codecov-action) from 4 to 5. - [Release notes](https://github.com/codecov/codecov-action/releases) - [Changelog](https://github.com/codecov/codecov-action/blob/main/CHANGELOG.md) - [Commits](https://github.com/codecov/codecov-action/compare/v4...v5) --- updated-dependencies: - dependency-name: codecov/codecov-action dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/codecov.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/codecov.yml b/.github/workflows/codecov.yml index f1007185..bdeda77b 100644 --- a/.github/workflows/codecov.yml +++ b/.github/workflows/codecov.yml @@ -37,6 +37,6 @@ jobs: continue-on-error: true - name: Upload to codecov - uses: codecov/codecov-action@v4 + uses: codecov/codecov-action@v5 with: files: coverage.xml From 5c56994bbe69e0f234037273a377a29d23e22031 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 21 Nov 2024 15:31:34 +0100 Subject: [PATCH 44/49] replace httpretty with mocket --- .pre-commit-config.yaml | 2 +- compliance_checker/tests/test_ioos_sos.py | 37 ++++++++++++----------- test_requirements.txt | 2 +- 3 files changed, 21 insertions(+), 20 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 7f1c47c6..b9fd970c 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.7.3 + rev: v0.7.4 hooks: - id: ruff args: ["--fix", "--show-fixes"] diff --git a/compliance_checker/tests/test_ioos_sos.py b/compliance_checker/tests/test_ioos_sos.py index 7d852689..3b98e1ee 100644 --- a/compliance_checker/tests/test_ioos_sos.py +++ b/compliance_checker/tests/test_ioos_sos.py @@ -1,7 +1,8 @@ import os import unittest -import httpretty +from mocket.mocket import mocketize +from mocket.mockhttp import Entry from compliance_checker.runner import ComplianceChecker from compliance_checker.suite import CheckSuite @@ -21,20 +22,20 @@ def setUp(self): # classes will show up CheckSuite().load_all_available_checkers() - @httpretty.activate + @mocketize def test_retrieve_getcaps(self): """Method that simulates retrieving SOS GetCapabilities""" url = "http://data.oceansmap.com/thredds/sos/caricoos_ag/VIA/VIA.ncml" - httpretty.register_uri( - httpretty.GET, - url, - content_type="text/xml", + Entry.single_register( + method=Entry.GET, + uri=url, body=self.resp, + headers={"content-type": "text/xml"}, ) - httpretty.register_uri( - httpretty.HEAD, + Entry.single_register( + Entry.HEAD, url, - content_type="text/xml", + headers={"content-type": "text/xml"}, body="HTTP/1.1 200", ) ComplianceChecker.run_checker(url, ["ioos_sos"], 1, "normal") @@ -53,7 +54,7 @@ def setUp(self): # classes will show up CheckSuite().load_all_available_checkers() - @httpretty.activate + @mocketize def test_retrieve_describesensor(self): """Method that simulates retrieving SOS DescribeSensor""" url = ( @@ -64,17 +65,17 @@ def test_retrieve_describesensor(self): "&outputFormat=text/xml%3Bsubtype%3D%22sensorML/1.0.1/profiles/ioos_sos/1.0%22" "&version=1.0.0" ) - httpretty.register_uri( - httpretty.GET, - url, - content_type="text/xml", + Entry.single_register( + method=Entry.GET, + uri=url, body=self.resp, + headers={"content-type": "text/xml"}, ) - httpretty.register_uri( - httpretty.HEAD, - url, - content_type="text/xml", + Entry.single_register( + method=Entry.HEAD, + uri=url, body="HTTP/1.1 200", + headers={"content-type": "text/xml"}, ) # need to mock out the HEAD response so that compliance checker # recognizes this as some sort of XML doc instead of an OPeNDAP diff --git a/test_requirements.txt b/test_requirements.txt index 1ebf0279..5774e2e3 100644 --- a/test_requirements.txt +++ b/test_requirements.txt @@ -1,7 +1,7 @@ codecov codespell flake8 -httpretty +mocket mypy myst-parser numpydoc From 29f17b3a4a78d30ffc1e4bc8874586d6993050c4 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 14 Nov 2024 18:06:53 +0100 Subject: [PATCH 45/49] py310 --- compliance_checker/cf/util.py | 2 +- compliance_checker/cfutil.py | 2 +- compliance_checker/suite.py | 6 +++--- compliance_checker/tests/conftest.py | 2 +- compliance_checker/tests/resources.py | 3 +-- compliance_checker/tests/test_suite.py | 2 +- pyproject.toml | 4 +--- 7 files changed, 9 insertions(+), 12 deletions(-) diff --git a/compliance_checker/cf/util.py b/compliance_checker/cf/util.py index 1b592829..67cba708 100644 --- a/compliance_checker/cf/util.py +++ b/compliance_checker/cf/util.py @@ -1,11 +1,11 @@ import itertools import os import sys +from importlib.resources import files from pkgutil import get_data import requests from cf_units import Unit -from importlib_resources import files from lxml import etree from netCDF4 import Dataset diff --git a/compliance_checker/cfutil.py b/compliance_checker/cfutil.py index 22336140..202970c2 100644 --- a/compliance_checker/cfutil.py +++ b/compliance_checker/cfutil.py @@ -7,9 +7,9 @@ import warnings from collections import defaultdict from functools import lru_cache, partial +from importlib.resources import files from cf_units import Unit -from importlib_resources import files _UNITLESS_DB = None _SEA_NAMES = None diff --git a/compliance_checker/suite.py b/compliance_checker/suite.py index 6a4281e2..a8a80bce 100644 --- a/compliance_checker/suite.py +++ b/compliance_checker/suite.py @@ -13,11 +13,11 @@ import warnings from collections import defaultdict from datetime import datetime, timezone +from importlib.metadata import entry_points from operator import itemgetter from pathlib import Path from urllib.parse import urlparse -import importlib_metadata import requests from lxml import etree as ET from netCDF4 import Dataset @@ -73,7 +73,7 @@ def _get_generator_plugins(cls): """ if not hasattr(cls, "suite_generators"): - gens = importlib_metadata.entry_points( + gens = entry_points( group="compliance_checker.generators", ) cls.suite_generators = [x.load() for x in gens] @@ -139,7 +139,7 @@ def load_all_available_checkers(cls): base classes. """ cls._load_checkers( - importlib_metadata.entry_points(group="compliance_checker.suites"), + entry_points(group="compliance_checker.suites"), ) @classmethod diff --git a/compliance_checker/tests/conftest.py b/compliance_checker/tests/conftest.py index a320df71..87137100 100644 --- a/compliance_checker/tests/conftest.py +++ b/compliance_checker/tests/conftest.py @@ -1,9 +1,9 @@ import os import subprocess +from importlib.resources import files from itertools import chain import pytest -from importlib_resources import files from netCDF4 import Dataset from compliance_checker.cf import util diff --git a/compliance_checker/tests/resources.py b/compliance_checker/tests/resources.py index 7bbca9d6..8b489150 100644 --- a/compliance_checker/tests/resources.py +++ b/compliance_checker/tests/resources.py @@ -1,6 +1,5 @@ import subprocess - -from importlib_resources import files +from importlib.resources import files def get_filename(path): diff --git a/compliance_checker/tests/test_suite.py b/compliance_checker/tests/test_suite.py index 61915ee9..b855b4e4 100644 --- a/compliance_checker/tests/test_suite.py +++ b/compliance_checker/tests/test_suite.py @@ -1,9 +1,9 @@ import os import unittest +from importlib.resources import files from pathlib import Path import numpy as np -from importlib_resources import files from compliance_checker.acdd import ACDDBaseCheck from compliance_checker.base import BaseCheck, GenericFile, Result diff --git a/pyproject.toml b/pyproject.toml index 696cf98f..41c0c4a2 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -17,7 +17,7 @@ maintainers = [ { name = "Luke Campbell" }, { name = "Filipe Fernandes" }, ] -requires-python = ">=3.8" +requires-python = ">=3.10" classifiers = [ "Development Status :: 5 - Production/Stable", "Intended Audience :: Developers", @@ -28,8 +28,6 @@ classifiers = [ "Operating System :: POSIX :: Linux", "Programming Language :: Python", "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", "Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.12", From 52e9eb68faa678a71c3292af752cd8c81bd71788 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 14 Nov 2024 18:11:53 +0100 Subject: [PATCH 46/49] remove py39 --- .github/workflows/default-tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/default-tests.yml b/.github/workflows/default-tests.yml index a68f2f28..be52e663 100644 --- a/.github/workflows/default-tests.yml +++ b/.github/workflows/default-tests.yml @@ -9,7 +9,7 @@ jobs: runs-on: ${{ matrix.os }} strategy: matrix: - python-version: [ "3.9", "3.10", "3.11", "3.12", "3.13" ] + python-version: [ "3.10", "3.11", "3.12", "3.13" ] os: [ windows-latest, ubuntu-latest, macos-latest ] fail-fast: false defaults: From e3226743ff1050244b005ccc93ec847f2d24ca80 Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 21 Nov 2024 18:43:54 +0100 Subject: [PATCH 47/49] no longer needed --- requirements.txt | 2 -- 1 file changed, 2 deletions(-) diff --git a/requirements.txt b/requirements.txt index 1a61c43b..b73eeef3 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,7 +1,5 @@ cf-units>=2 cftime>=1.1.0 -importlib-metadata -importlib-resources isodate>=0.6.1 jinja2>=2.7.3 lxml>=3.2.1 From 61d4b0a47c004898a5499df3b04dc6a58954515f Mon Sep 17 00:00:00 2001 From: Filipe Fernandes Date: Thu, 21 Nov 2024 22:00:33 +0100 Subject: [PATCH 48/49] reduce the development requirements --- test_requirements.txt | 6 ------ 1 file changed, 6 deletions(-) diff --git a/test_requirements.txt b/test_requirements.txt index 5774e2e3..5aa09bc0 100644 --- a/test_requirements.txt +++ b/test_requirements.txt @@ -1,13 +1,7 @@ -codecov -codespell -flake8 mocket -mypy myst-parser numpydoc -pre-commit pydata-sphinx-theme pytest>=2.9.0 -pytest-cov>=3.0.0 pytest-vcr requests-mock>=1.7.0 From 29cee409fafc2f21382bda9f7cd7e16c905ace39 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Mon, 2 Dec 2024 21:35:27 +0000 Subject: [PATCH 49/49] [pre-commit.ci] pre-commit autoupdate MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit updates: - [github.com/astral-sh/ruff-pre-commit: v0.7.4 → v0.8.1](https://github.com/astral-sh/ruff-pre-commit/compare/v0.7.4...v0.8.1) --- .pre-commit-config.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index b9fd970c..55e8ebed 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -31,7 +31,7 @@ repos: - repo: https://github.com/astral-sh/ruff-pre-commit - rev: v0.7.4 + rev: v0.8.1 hooks: - id: ruff args: ["--fix", "--show-fixes"]