diff --git a/.dockerignore b/.dockerignore index 603386e55e3b..88f241c5275e 100644 --- a/.dockerignore +++ b/.dockerignore @@ -24,6 +24,7 @@ keys/setup !Cargo.toml !contracts/ !setup_2\^26.key +!setup_2\^24.key # It's required to remove .git from contracts, # otherwise yarn tries to use .git parent directory that # doesn't exist. diff --git a/.github/release-please/manifest.json b/.github/release-please/manifest.json index 6041978263ff..cbe9d9da0843 100644 --- a/.github/release-please/manifest.json +++ b/.github/release-please/manifest.json @@ -1,4 +1,4 @@ { - "core": "24.3.0", - "prover": "14.2.0" + "core": "24.4.0", + "prover": "14.3.0" } diff --git a/.github/workflows/build-contract-verifier-template.yml b/.github/workflows/build-contract-verifier-template.yml index fab6a6f18a58..52f03243b414 100644 --- a/.github/workflows/build-contract-verifier-template.yml +++ b/.github/workflows/build-contract-verifier-template.yml @@ -31,11 +31,11 @@ jobs: runs-on: ${{ fromJSON('["matterlabs-ci-runner", "matterlabs-ci-runner-arm"]')[contains(matrix.platforms, 'arm')] }} strategy: matrix: - components: - - contract-verifier - - verified-sources-fetcher - platforms: - - linux/amd64 + components: + - contract-verifier + - verified-sources-fetcher + platforms: + - linux/amd64 steps: - uses: actions/checkout@a5ac7e51b41094c92402da3b24376905380afc29 # v4 @@ -143,8 +143,8 @@ jobs: - name: Show sccache stats if: always() run: | - ci_run sccache --show-stats - ci_run cat /tmp/sccache_log.txt + ci_run sccache --show-stats || true + ci_run cat /tmp/sccache_log.txt || true create_manifest: name: Create release manifest diff --git a/.github/workflows/build-core-template.yml b/.github/workflows/build-core-template.yml index 29b66d991f01..e19b644a512c 100644 --- a/.github/workflows/build-core-template.yml +++ b/.github/workflows/build-core-template.yml @@ -36,15 +36,15 @@ jobs: runs-on: ${{ fromJSON('["matterlabs-ci-runner", "matterlabs-ci-runner-arm"]')[contains(matrix.platforms, 'arm')] }} strategy: matrix: - components: - - server-v2 - - external-node - - snapshots-creator - platforms: - - linux/amd64 - include: - - components: external-node - platforms: linux/arm64 + components: + - server-v2 + - external-node + - snapshots-creator + platforms: + - linux/amd64 + include: + - components: external-node + platforms: linux/arm64 steps: - uses: actions/checkout@a5ac7e51b41094c92402da3b24376905380afc29 # v4 @@ -152,8 +152,8 @@ jobs: - name: Show sccache stats if: always() run: | - ci_run sccache --show-stats - ci_run cat /tmp/sccache_log.txt + ci_run sccache --show-stats || true + ci_run cat /tmp/sccache_log.txt || true create_manifest: name: Create release manifest diff --git a/.github/workflows/build-prover-template.yml b/.github/workflows/build-prover-template.yml index 4da79fccb40a..c2762245bc03 100644 --- a/.github/workflows/build-prover-template.yml +++ b/.github/workflows/build-prover-template.yml @@ -41,7 +41,7 @@ jobs: RUNNER_COMPOSE_FILE: "docker-compose-runner-nightly.yml" ERA_BELLMAN_CUDA_RELEASE: ${{ inputs.ERA_BELLMAN_CUDA_RELEASE }} CUDA_ARCH: ${{ inputs.CUDA_ARCH }} - runs-on: [matterlabs-ci-runner] + runs-on: [ matterlabs-ci-runner ] strategy: matrix: component: @@ -51,6 +51,7 @@ jobs: - witness-vector-generator - prover-fri-gateway - proof-fri-compressor + - proof-fri-gpu-compressor steps: - uses: actions/checkout@a5ac7e51b41094c92402da3b24376905380afc29 # v4 with: @@ -80,11 +81,17 @@ jobs: ci_run zk # We need the CRS only for the fri compressor. - - name: download CRS + - name: download CRS for CPU compressor if: matrix.component == 'proof-fri-compressor' run: | ci_run curl --retry 5 -LO https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2\^26.key + - name: download CRS for GPU compressor + if: matrix.component == 'proof-fri-gpu-compressor' + run: | + ci_run curl --retry 5 -LO https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2\^24.key + + - name: login to Docker registries if: github.event_name != 'pull_request' && (github.ref == 'refs/heads/main' || startsWith(github.ref, 'refs/tags/')) run: | @@ -138,15 +145,15 @@ jobs: env: DOCKER_ACTION: ${{ inputs.action }} COMPONENT: ${{ matrix.component }} - run: | + run: | PASSED_ENV_VARS="ERA_BELLMAN_CUDA_RELEASE,CUDA_ARCH" \ ci_run zk docker $DOCKER_ACTION $COMPONENT - name: Show sccache stats if: always() run: | - ci_run sccache --show-stats - ci_run cat /tmp/sccache_log.txt + ci_run sccache --show-stats || true + ci_run cat /tmp/sccache_log.txt || true copy-images: name: Copy images between docker registries diff --git a/.github/workflows/ci-core-reusable.yml b/.github/workflows/ci-core-reusable.yml index 39b389ef94ed..02069c4259f4 100644 --- a/.github/workflows/ci-core-reusable.yml +++ b/.github/workflows/ci-core-reusable.yml @@ -120,8 +120,8 @@ jobs: - name: Show sccache logs if: always() run: | - ci_run sccache --show-stats - ci_run cat /tmp/sccache_log.txt + ci_run sccache --show-stats || true + ci_run cat /tmp/sccache_log.txt || true integration: name: Integration (consensus=${{ matrix.consensus }}, base_token=${{ matrix.base_token }}, deployment_mode=${{ matrix.deployment_mode }}) @@ -202,7 +202,7 @@ jobs: # `sleep 5` because we need to wait until server started properly - name: Run server run: | - ci_run zk server --components=$SERVER_COMPONENTS &>server.log & + ci_run zk server --use-node-framework --components=$SERVER_COMPONENTS &>server.log & ci_run sleep 5 - name: Run contract verifier @@ -217,12 +217,28 @@ jobs: # We use `yarn` directly because the test launches `zk` commands in both server and EN envs. # An empty topmost environment helps avoid a mess when redefining env vars shared between both envs # (e.g., DATABASE_URL). + # + # Since `base_token` doesn't meaningfully influence the test, we use it as a flag for + # enabling / disabling tree during pruning. run: | if [[ "${{ matrix.deployment_mode }}" == "Validium" ]]; then ci_run zk config compile ext-node-validium ci_run zk config compile ext-node-validium-docker fi - ENABLE_CONSENSUS=${{ matrix.consensus }} DEPLOYMENT_MODE=${{ matrix.deployment_mode }} PASSED_ENV_VARS="ENABLE_CONSENSUS,DEPLOYMENT_MODE" ci_run yarn snapshot-recovery-test snapshot-recovery-test + ENABLE_CONSENSUS=${{ matrix.consensus }} \ + DEPLOYMENT_MODE=${{ matrix.deployment_mode }} \ + DISABLE_TREE_DURING_PRUNING=${{ matrix.base_token == 'Eth' }} \ + ETH_CLIENT_WEB3_URL="http://reth:8545" \ + PASSED_ENV_VARS="ENABLE_CONSENSUS,DEPLOYMENT_MODE,DISABLE_TREE_DURING_PRUNING,ETH_CLIENT_WEB3_URL" \ + ci_run yarn recovery-test snapshot-recovery-test + + - name: Genesis recovery test + run: | + ENABLE_CONSENSUS=${{ matrix.consensus }} \ + DEPLOYMENT_MODE=${{ matrix.deployment_mode }} \ + ETH_CLIENT_WEB3_URL="http://reth:8545" \ + PASSED_ENV_VARS="ENABLE_CONSENSUS,DEPLOYMENT_MODE,ETH_CLIENT_WEB3_URL" \ + ci_run yarn recovery-test genesis-recovery-test - name: Fee projection tests run: ci_run zk test i fees @@ -252,10 +268,13 @@ jobs: - name: Show snapshot-creator.log logs if: always() - run: ci_run cat core/tests/snapshot-recovery-test/snapshot-creator.log || true + run: ci_run cat core/tests/recovery-test/snapshot-creator.log || true - name: Show snapshot-recovery.log logs if: always() - run: ci_run cat core/tests/snapshot-recovery-test/snapshot-recovery.log || true + run: ci_run cat core/tests/recovery-test/snapshot-recovery.log || true + - name: Show genesis-recovery.log logs + if: always() + run: ci_run cat core/tests/recovery-test/genesis-recovery.log || true - name: Show revert.log logs if: always() @@ -268,8 +287,8 @@ jobs: - name: Show sccache logs if: always() run: | - ci_run sccache --show-stats - ci_run cat /tmp/sccache_log.txt + ci_run sccache --show-stats || true + ci_run cat /tmp/sccache_log.txt || true external-node: name: External node (consensus=${{ matrix.consensus }}, base_token=${{ matrix.base_token }}, deployment_mode=${{ matrix.deployment_mode }}) @@ -389,5 +408,5 @@ jobs: - name: Show sccache logs if: always() run: | - ci_run sccache --show-stats - ci_run cat /tmp/sccache_log.txt + ci_run sccache --show-stats || true + ci_run cat /tmp/sccache_log.txt || true diff --git a/.gitignore b/.gitignore index d6658ac4df42..13bc2d3470b4 100644 --- a/.gitignore +++ b/.gitignore @@ -10,7 +10,7 @@ node_modules *.log target -a.out +a.out .gitconfig cobertura.xml tags @@ -30,6 +30,7 @@ Cargo.lock !/Cargo.lock !/infrastructure/zksync-crypto/Cargo.lock !/prover/Cargo.lock +!/zk_toolbox/Cargo.lock /etc/env/target/* /etc/env/.current @@ -96,3 +97,7 @@ hyperchain-*.yml # Prover keys that should not be commited prover/vk_setup_data_generator_server_fri/data/setup_* + +# Zk Toolbox +chains/era/configs/* +configs/* diff --git a/Cargo.lock b/Cargo.lock index 4ad620eef791..2853035f34b7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1015,7 +1015,7 @@ dependencies = [ [[package]] name = "circuit_encodings" version = "0.1.50" -source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#05502ec874bd1dbfd8a72cd7df340a2fe3f6d3a0" +source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#a9b1c3a3cf46e683d6a27db33805d994ca8476ec" dependencies = [ "derivative", "serde", @@ -1077,14 +1077,13 @@ dependencies = [ [[package]] name = "circuit_sequencer_api" version = "0.1.50" -source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#05502ec874bd1dbfd8a72cd7df340a2fe3f6d3a0" +source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#a9b1c3a3cf46e683d6a27db33805d994ca8476ec" dependencies = [ "bellman_ce", "circuit_encodings 0.1.50", "derivative", "rayon", "serde", - "zk_evm 1.5.0", ] [[package]] @@ -2596,9 +2595,9 @@ dependencies = [ [[package]] name = "hashbrown" -version = "0.13.2" +version = "0.13.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "43a3c133739dddd0d2990f9a4bdf8eb4b21ef50e4851ca85ab661199821d510e" +checksum = "33ff8ae62cd3a9102e5637afc8452c55acf3844001bd5374e0b0bd7b6616c038" dependencies = [ "ahash 0.8.7", ] @@ -3242,13 +3241,16 @@ dependencies = [ [[package]] name = "kzg" version = "0.1.50" -source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#394e1c7d1aec06d2f3abd63bdc2ddf0efef5ac49" +source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#a9b1c3a3cf46e683d6a27db33805d994ca8476ec" dependencies = [ "boojum", "derivative", + "hex", + "once_cell", "rayon", "serde", "serde_json", + "serde_with", "zkevm_circuits 1.5.0", ] @@ -3598,13 +3600,13 @@ dependencies = [ [[package]] name = "metrics-util" -version = "0.15.0" +version = "0.15.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "111cb375987443c3de8d503580b536f77dc8416d32db62d9456db5d93bd7ac47" +checksum = "4de2ed6e491ed114b40b732e4d1659a9d53992ebd87490c44a6ffe23739d973e" dependencies = [ "crossbeam-epoch 0.9.15", "crossbeam-utils 0.8.16", - "hashbrown 0.13.2", + "hashbrown 0.13.1", "metrics", "num_cpus", "quanta 0.11.1", @@ -7834,7 +7836,7 @@ dependencies = [ [[package]] name = "zk_evm" version = "1.5.0" -source = "git+https://github.com/matter-labs/era-zk_evm.git?branch=v1.5.0#c42da1512334c3d95869198e41ee4f0da68812b4" +source = "git+https://github.com/matter-labs/era-zk_evm.git?branch=v1.5.0#9bbf7ffd2c38ee8b9667e96eaf0c111037fe976f" dependencies = [ "anyhow", "lazy_static", @@ -7926,7 +7928,7 @@ dependencies = [ [[package]] name = "zkevm_circuits" version = "1.5.0" -source = "git+https://github.com/matter-labs/era-zkevm_circuits.git?branch=v1.5.0#a93a3a5c34ec1ec31d73191d11ab00b4d8215a3f" +source = "git+https://github.com/matter-labs/era-zkevm_circuits.git?branch=v1.5.0#28fe577bbb2b95c18d3959ba3dd37ca8ce5bd865" dependencies = [ "arrayvec 0.7.4", "boojum", @@ -7984,7 +7986,7 @@ dependencies = [ [[package]] name = "zkevm_opcode_defs" version = "1.5.0" -source = "git+https://github.com/matter-labs/era-zkevm_opcode_defs.git?branch=v1.5.0#109d9f734804a8b9dc0531c0b576e2a0f55a40de" +source = "git+https://github.com/matter-labs/era-zkevm_opcode_defs.git?branch=v1.5.0#28d2edabf902ea9b08f6a26a4506831fd89346b9" dependencies = [ "bitflags 2.4.1", "blake2 0.10.6 (registry+https://github.com/rust-lang/crates.io-index)", @@ -8060,9 +8062,11 @@ dependencies = [ "circuit_sequencer_api 0.1.40", "circuit_sequencer_api 0.1.41", "circuit_sequencer_api 0.1.50", + "futures 0.3.28", "itertools 0.10.5", - "jsonrpsee", "multivm", + "num_cpus", + "rand 0.8.5", "serde_json", "tokio", "tracing", @@ -8075,6 +8079,8 @@ dependencies = [ "zksync_eth_client", "zksync_health_check", "zksync_l1_contract_interface", + "zksync_node_genesis", + "zksync_node_test_utils", "zksync_types", "zksync_utils", "zksync_web3_decl", @@ -8083,7 +8089,7 @@ dependencies = [ [[package]] name = "zksync_concurrency" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "once_cell", @@ -8114,7 +8120,7 @@ dependencies = [ [[package]] name = "zksync_consensus_bft" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "async-trait", @@ -8135,7 +8141,7 @@ dependencies = [ [[package]] name = "zksync_consensus_crypto" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "blst", @@ -8156,7 +8162,7 @@ dependencies = [ [[package]] name = "zksync_consensus_executor" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "rand 0.8.5", @@ -8175,7 +8181,7 @@ dependencies = [ [[package]] name = "zksync_consensus_network" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "async-trait", @@ -8200,7 +8206,7 @@ dependencies = [ [[package]] name = "zksync_consensus_roles" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "bit-vec", @@ -8221,7 +8227,7 @@ dependencies = [ [[package]] name = "zksync_consensus_storage" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "async-trait", @@ -8239,7 +8245,7 @@ dependencies = [ [[package]] name = "zksync_consensus_utils" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "rand 0.8.5", "thiserror", @@ -8598,7 +8604,7 @@ dependencies = [ [[package]] name = "zksync_external_node" -version = "24.3.0" +version = "24.4.0" dependencies = [ "anyhow", "assert_matches", @@ -8924,7 +8930,6 @@ dependencies = [ "zksync_consistency_checker", "zksync_contract_verification_server", "zksync_contracts", - "zksync_core_leftovers", "zksync_dal", "zksync_db_connection", "zksync_env_config", @@ -8941,9 +8946,11 @@ dependencies = [ "zksync_object_store", "zksync_proof_data_handler", "zksync_protobuf_config", + "zksync_queued_job_processor", "zksync_state", "zksync_state_keeper", "zksync_storage", + "zksync_tee_verifier_input_producer", "zksync_types", "zksync_utils", "zksync_web3_decl", @@ -9056,7 +9063,7 @@ dependencies = [ [[package]] name = "zksync_protobuf" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "bit-vec", @@ -9076,7 +9083,7 @@ dependencies = [ [[package]] name = "zksync_protobuf_build" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "heck 0.5.0", @@ -9162,6 +9169,7 @@ dependencies = [ "anyhow", "clap 4.4.6", "futures 0.3.28", + "prometheus_exporter", "serde_json", "tikv-jemallocator", "tokio", @@ -9175,6 +9183,9 @@ dependencies = [ "zksync_core_leftovers", "zksync_env_config", "zksync_eth_client", + "zksync_metadata_calculator", + "zksync_node_api_server", + "zksync_node_framework", "zksync_node_genesis", "zksync_protobuf_config", "zksync_storage", @@ -9399,6 +9410,7 @@ dependencies = [ "hex", "itertools 0.10.5", "num", + "once_cell", "rand 0.8.5", "reqwest", "serde", @@ -9434,7 +9446,9 @@ dependencies = [ "zksync_state", "zksync_state_keeper", "zksync_storage", + "zksync_test_account", "zksync_types", + "zksync_utils", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index 816a1057c95b..77af41c63721 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -189,16 +189,16 @@ zk_evm_1_3_3 = { package = "zk_evm", git = "https://github.com/matter-labs/era-z zk_evm_1_4_0 = { package = "zk_evm", git = "https://github.com/matter-labs/era-zk_evm.git", branch = "v1.4.0" } zk_evm_1_4_1 = { package = "zk_evm", git = "https://github.com/matter-labs/era-zk_evm.git", branch = "v1.4.1" } zk_evm_1_5_0 = { package = "zk_evm", git = "https://github.com/matter-labs/era-zk_evm.git", branch = "v1.5.0" } -zksync_concurrency = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_bft = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_crypto = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_executor = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_network = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_roles = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_storage = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_consensus_utils = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_protobuf = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } -zksync_protobuf_build = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" } +zksync_concurrency = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_bft = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_crypto = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_executor = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_network = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_roles = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_storage = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_consensus_utils = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_protobuf = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } +zksync_protobuf_build = { version = "0.1.0", git = "https://github.com/matter-labs/era-consensus.git", rev = "3e6f101ee4124308c4c974caaa259d524549b0c6" } # "Local" dependencies multivm = { path = "core/lib/multivm" } diff --git a/ZkStack.yaml b/ZkStack.yaml new file mode 100644 index 000000000000..33af09572190 --- /dev/null +++ b/ZkStack.yaml @@ -0,0 +1,9 @@ +name: zk +l1_network: Localhost +link_to_code: . +chains: ./chains +config: ./configs/ +default_chain: era +era_chain_id: 270 +prover_version: NoProofs +wallet_creation: Localhost diff --git a/chains/era/ZkStack.yaml b/chains/era/ZkStack.yaml new file mode 100644 index 000000000000..17b307cac4f6 --- /dev/null +++ b/chains/era/ZkStack.yaml @@ -0,0 +1,12 @@ +id: 1 +name: era +chain_id: 271 +prover_version: NoProofs +configs: ./chains/era/configs/ +rocks_db_path: ./chains/era/db/ +l1_batch_commit_data_generator_mode: Rollup +base_token: + address: '0x0000000000000000000000000000000000000001' + nominator: 1 + denominator: 1 +wallet_creation: Localhost diff --git a/checks-config/era.dic b/checks-config/era.dic index 34610e8e8099..063c129b3e66 100644 --- a/checks-config/era.dic +++ b/checks-config/era.dic @@ -961,3 +961,11 @@ vec zksync_merkle_tree TreeMetadata delegator +decrement +Bbellman +Sbellman +DCMAKE +preloaded +e2e +upcasting +foundryup diff --git a/configs/.gitkeep b/configs/.gitkeep new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/contracts b/contracts index 452a54f67243..5312fd40c12c 160000 --- a/contracts +++ b/contracts @@ -1 +1 @@ -Subproject commit 452a54f6724347b7e517be1a3d948299ab827d8c +Subproject commit 5312fd40c12c622e15db9b5515cff0e5d6c5324d diff --git a/core/CHANGELOG.md b/core/CHANGELOG.md index dc453ff54f07..424ab8c3a3ba 100644 --- a/core/CHANGELOG.md +++ b/core/CHANGELOG.md @@ -1,5 +1,22 @@ # Changelog +## [24.4.0](https://github.com/matter-labs/zksync-era/compare/core-v24.3.0...core-v24.4.0) (2024-05-21) + + +### Features + +* **prover:** add GPU feature for compressor ([#1838](https://github.com/matter-labs/zksync-era/issues/1838)) ([e9a2213](https://github.com/matter-labs/zksync-era/commit/e9a2213985928cd3804a3855ccfde6a7d99da238)) +* **pruning:** remove manual vaccum; add migration configuring autovacuum ([#1983](https://github.com/matter-labs/zksync-era/issues/1983)) ([3d98072](https://github.com/matter-labs/zksync-era/commit/3d98072468b1f7dac653b4ff04bda66e2fc8185e)) +* **tests:** Move all env calls to one place in ts-tests ([#1968](https://github.com/matter-labs/zksync-era/issues/1968)) ([3300047](https://github.com/matter-labs/zksync-era/commit/33000475b47831fc3791dac338aae4d0e7db25b0)) + + +### Bug Fixes + +* Disallow non null updates for transactions ([#1951](https://github.com/matter-labs/zksync-era/issues/1951)) ([a603ac8](https://github.com/matter-labs/zksync-era/commit/a603ac8eaab112738e1c2336b0f537273ad58d85)) +* **en:** Minor node fixes ([#1978](https://github.com/matter-labs/zksync-era/issues/1978)) ([74144e8](https://github.com/matter-labs/zksync-era/commit/74144e8240f633a587f0cd68f4d136a7a68af7be)) +* **en:** run `MainNodeFeeParamsFetcher` in API component ([#1988](https://github.com/matter-labs/zksync-era/issues/1988)) ([b62677e](https://github.com/matter-labs/zksync-era/commit/b62677ea5f8f6bb57d6ad02139a938ccf943e06a)) +* **merkle-tree:** Fix tree API health check status ([#1973](https://github.com/matter-labs/zksync-era/issues/1973)) ([6235561](https://github.com/matter-labs/zksync-era/commit/623556112c40400244906e42c5f84a047dc6f26b)) + ## [24.3.0](https://github.com/matter-labs/zksync-era/compare/core-v24.2.0...core-v24.3.0) (2024-05-16) diff --git a/core/bin/contract-verifier/src/main.rs b/core/bin/contract-verifier/src/main.rs index 73b2f919c31a..98b4a859d14d 100644 --- a/core/bin/contract-verifier/src/main.rs +++ b/core/bin/contract-verifier/src/main.rs @@ -3,6 +3,7 @@ use std::{cell::RefCell, time::Duration}; use anyhow::Context as _; use futures::{channel::mpsc, executor::block_on, SinkExt, StreamExt}; use prometheus_exporter::PrometheusExporterConfig; +use structopt::StructOpt; use tokio::sync::watch; use zksync_config::{ configs::{ObservabilityConfig, PrometheusConfig}, @@ -11,7 +12,7 @@ use zksync_config::{ use zksync_dal::{ConnectionPool, Core, CoreDal}; use zksync_env_config::FromEnv; use zksync_queued_job_processor::JobProcessor; -use zksync_utils::wait_for_tasks::ManagedTasks; +use zksync_utils::{wait_for_tasks::ManagedTasks, workspace_dir_or_current_dir}; use crate::verifier::ContractVerifier; @@ -25,9 +26,9 @@ async fn update_compiler_versions(connection_pool: &ConnectionPool) { let mut storage = connection_pool.connection().await.unwrap(); let mut transaction = storage.start_transaction().await.unwrap(); - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); + let zksync_home = workspace_dir_or_current_dir(); - let zksolc_path = format!("{}/etc/zksolc-bin/", zksync_home); + let zksolc_path = zksync_home.join("etc/zksolc-bin/"); let zksolc_versions: Vec = std::fs::read_dir(zksolc_path) .unwrap() .filter_map(|file| { @@ -48,7 +49,7 @@ async fn update_compiler_versions(connection_pool: &ConnectionPool) { .await .unwrap(); - let solc_path = format!("{}/etc/solc-bin/", zksync_home); + let solc_path = zksync_home.join("etc/solc-bin/"); let solc_versions: Vec = std::fs::read_dir(solc_path) .unwrap() .filter_map(|file| { @@ -69,7 +70,7 @@ async fn update_compiler_versions(connection_pool: &ConnectionPool) { .await .unwrap(); - let zkvyper_path = format!("{}/etc/zkvyper-bin/", zksync_home); + let zkvyper_path = zksync_home.join("etc/zkvyper-bin/"); let zkvyper_versions: Vec = std::fs::read_dir(zkvyper_path) .unwrap() .filter_map(|file| { @@ -90,7 +91,7 @@ async fn update_compiler_versions(connection_pool: &ConnectionPool) { .await .unwrap(); - let vyper_path = format!("{}/etc/vyper-bin/", zksync_home); + let vyper_path = zksync_home.join("etc/vyper-bin/"); let vyper_versions: Vec = std::fs::read_dir(vyper_path) .unwrap() .filter_map(|file| { @@ -115,7 +116,6 @@ async fn update_compiler_versions(connection_pool: &ConnectionPool) { transaction.commit().await.unwrap(); } -use structopt::StructOpt; use zksync_config::configs::DatabaseSecrets; #[derive(StructOpt)] diff --git a/core/bin/contract-verifier/src/verifier.rs b/core/bin/contract-verifier/src/verifier.rs index 938ea2fd1ba6..8d5ba9fccfe2 100644 --- a/core/bin/contract-verifier/src/verifier.rs +++ b/core/bin/contract-verifier/src/verifier.rs @@ -1,6 +1,5 @@ use std::{ collections::HashMap, - env, path::Path, time::{Duration, Instant}, }; @@ -22,6 +21,7 @@ use zksync_types::{ }, Address, }; +use zksync_utils::workspace_dir_or_current_dir; use crate::{ error::ContractVerifierError, @@ -34,6 +34,10 @@ lazy_static! { static ref DEPLOYER_CONTRACT: Contract = zksync_contracts::deployer_contract(); } +fn home_path() -> &'static Path { + workspace_dir_or_current_dir() +} + #[derive(Debug)] enum ConstructorArgs { Check(Vec), @@ -120,8 +124,7 @@ impl ContractVerifier { }; let input = Self::build_zksolc_input(request.clone(), file_name.clone())?; - let zksync_home = env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); - let zksolc_path = Path::new(&zksync_home) + let zksolc_path = Path::new(&home_path()) .join("etc") .join("zksolc-bin") .join(request.req.compiler_versions.zk_compiler_version()) @@ -133,7 +136,7 @@ impl ContractVerifier { )); } - let solc_path = Path::new(&zksync_home) + let solc_path = Path::new(&home_path()) .join("etc") .join("solc-bin") .join(request.req.compiler_versions.compiler_version()) @@ -219,8 +222,7 @@ impl ContractVerifier { }; let input = Self::build_zkvyper_input(request.clone())?; - let zksync_home = env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); - let zkvyper_path = Path::new(&zksync_home) + let zkvyper_path = Path::new(&home_path()) .join("etc") .join("zkvyper-bin") .join(request.req.compiler_versions.zk_compiler_version()) @@ -232,7 +234,7 @@ impl ContractVerifier { )); } - let vyper_path = Path::new(&zksync_home) + let vyper_path = Path::new(&home_path()) .join("etc") .join("vyper-bin") .join(request.req.compiler_versions.compiler_version()) diff --git a/core/bin/external_node/Cargo.toml b/core/bin/external_node/Cargo.toml index 3743d82ac818..b5815a9a223c 100644 --- a/core/bin/external_node/Cargo.toml +++ b/core/bin/external_node/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "zksync_external_node" -version = "24.3.0" # x-release-please-version +version = "24.4.0" # x-release-please-version edition.workspace = true authors.workspace = true homepage.workspace = true diff --git a/core/bin/external_node/src/config/mod.rs b/core/bin/external_node/src/config/mod.rs index 63c0433eda0d..56d66a3a4253 100644 --- a/core/bin/external_node/src/config/mod.rs +++ b/core/bin/external_node/src/config/mod.rs @@ -17,6 +17,7 @@ use zksync_config::{ use zksync_core_leftovers::temp_config_store::decode_yaml_repr; #[cfg(test)] use zksync_dal::{ConnectionPool, Core}; +use zksync_metadata_calculator::MetadataCalculatorRecoveryConfig; use zksync_node_api_server::{ tx_sender::TxSenderConfig, web3::{state::InternalApiConfig, Namespace}, @@ -745,6 +746,20 @@ pub(crate) struct ExperimentalENConfig { /// Maximum number of files concurrently opened by state keeper cache RocksDB. Useful to fit into OS limits; can be used /// as a rudimentary way to control RAM usage of the cache. pub state_keeper_db_max_open_files: Option, + + // Snapshot recovery + /// Approximate chunk size (measured in the number of entries) to recover in a single iteration. + /// Reasonable values are order of 100,000 (meaning an iteration takes several seconds). + /// + /// **Important.** This value cannot be changed in the middle of tree recovery (i.e., if a node is stopped in the middle + /// of recovery and then restarted with a different config). + #[serde(default = "ExperimentalENConfig::default_snapshots_recovery_tree_chunk_size")] + pub snapshots_recovery_tree_chunk_size: u64, + + // Commitment generator + /// Maximum degree of parallelism during commitment generation, i.e., the maximum number of L1 batches being processed in parallel. + /// If not specified, commitment generator will use a value roughly equal to the number of CPU cores with some clamping applied. + pub commitment_generator_max_parallelism: Option, } impl ExperimentalENConfig { @@ -752,12 +767,18 @@ impl ExperimentalENConfig { 128 } + fn default_snapshots_recovery_tree_chunk_size() -> u64 { + MetadataCalculatorRecoveryConfig::default().desired_chunk_size + } + #[cfg(test)] fn mock() -> Self { Self { state_keeper_db_block_cache_capacity_mb: Self::default_state_keeper_db_block_cache_capacity_mb(), state_keeper_db_max_open_files: None, + snapshots_recovery_tree_chunk_size: Self::default_snapshots_recovery_tree_chunk_size(), + commitment_generator_max_parallelism: None, } } diff --git a/core/bin/external_node/src/main.rs b/core/bin/external_node/src/main.rs index 07ca5b9f0f28..503b0e03516e 100644 --- a/core/bin/external_node/src/main.rs +++ b/core/bin/external_node/src/main.rs @@ -22,7 +22,7 @@ use zksync_db_connection::{ use zksync_health_check::{AppHealthCheck, HealthStatus, ReactiveHealthCheck}; use zksync_metadata_calculator::{ api_server::{TreeApiClient, TreeApiHttpClient}, - MetadataCalculator, MetadataCalculatorConfig, + MetadataCalculator, MetadataCalculatorConfig, MetadataCalculatorRecoveryConfig, }; use zksync_node_api_server::{ execution_sandbox::VmConcurrencyLimiter, @@ -41,7 +41,7 @@ use zksync_reorg_detector::ReorgDetector; use zksync_state::{PostgresStorageCaches, RocksdbStorageOptions}; use zksync_state_keeper::{ seal_criteria::NoopSealer, AsyncRocksdbCache, BatchExecutor, MainBatchExecutor, OutputHandler, - StateKeeperPersistence, ZkSyncStateKeeper, + StateKeeperPersistence, TreeWritesPersistence, ZkSyncStateKeeper, }; use zksync_storage::RocksDB; use zksync_types::L2ChainId; @@ -96,11 +96,8 @@ async fn build_state_keeper( stop_receiver_clone.changed().await?; result })); - let batch_executor_base: Box = Box::new(MainBatchExecutor::new( - Arc::new(storage_factory), - save_call_traces, - true, - )); + let batch_executor_base: Box = + Box::new(MainBatchExecutor::new(save_call_traces, true)); let io = ExternalIO::new( connection_pool, @@ -117,6 +114,7 @@ async fn build_state_keeper( batch_executor_base, output_handler, Arc::new(NoopSealer), + Arc::new(storage_factory), )) } @@ -141,6 +139,9 @@ async fn run_tree( .merkle_tree_include_indices_and_filters_in_block_cache, memtable_capacity: config.optional.merkle_tree_memtable_capacity(), stalled_writes_timeout: config.optional.merkle_tree_stalled_writes_timeout(), + recovery: MetadataCalculatorRecoveryConfig { + desired_chunk_size: config.experimental.snapshots_recovery_tree_chunk_size, + }, }; let max_concurrency = config @@ -230,9 +231,11 @@ async fn run_core( tracing::warn!("Disabling persisting protective reads; this should be safe, but is considered an experimental option at the moment"); persistence = persistence.without_protective_reads(); } + let tree_writes_persistence = TreeWritesPersistence::new(connection_pool.clone()); - let output_handler = - OutputHandler::new(Box::new(persistence)).with_handler(Box::new(sync_state.clone())); + let output_handler = OutputHandler::new(Box::new(persistence)) + .with_handler(Box::new(tree_writes_persistence)) + .with_handler(Box::new(sync_state.clone())); let state_keeper = build_state_keeper( action_queue, config.required.state_cache_path.clone(), @@ -359,14 +362,13 @@ async fn run_core( ); app_health.insert_component(batch_status_updater.health_check())?; - let commitment_generator_pool = singleton_pool_builder - .build() - .await - .context("failed to build a commitment_generator_pool")?; - let commitment_generator = CommitmentGenerator::new( - commitment_generator_pool, + let mut commitment_generator = CommitmentGenerator::new( + connection_pool.clone(), config.optional.l1_batch_commit_data_generator_mode, ); + if let Some(parallelism) = config.experimental.commitment_generator_max_parallelism { + commitment_generator.set_max_parallelism(parallelism); + } app_health.insert_component(commitment_generator.health_check())?; let commitment_generator_handle = tokio::spawn(commitment_generator.run(stop_receiver.clone())); @@ -675,20 +677,6 @@ async fn init_tasks( .await?; } - if let Some(prometheus) = config.observability.prometheus() { - tracing::info!("Starting Prometheus exporter with configuration: {prometheus:?}"); - - let (prometheus_health_check, prometheus_health_updater) = - ReactiveHealthCheck::new("prometheus_exporter"); - app_health.insert_component(prometheus_health_check)?; - task_handles.push(tokio::spawn(async move { - prometheus_health_updater.update(HealthStatus::Ready.into()); - let result = prometheus.run(stop_receiver).await; - drop(prometheus_health_updater); - result - })); - } - Ok(()) } @@ -884,6 +872,24 @@ async fn run_node( ([0, 0, 0, 0], config.required.healthcheck_port).into(), app_health.clone(), ); + // Start exporting metrics at the very start so that e.g., snapshot recovery metrics are timely reported. + let prometheus_task = if let Some(prometheus) = config.observability.prometheus() { + tracing::info!("Starting Prometheus exporter with configuration: {prometheus:?}"); + + let (prometheus_health_check, prometheus_health_updater) = + ReactiveHealthCheck::new("prometheus_exporter"); + app_health.insert_component(prometheus_health_check)?; + let stop_receiver_for_exporter = stop_receiver.clone(); + Some(tokio::spawn(async move { + prometheus_health_updater.update(HealthStatus::Ready.into()); + let result = prometheus.run(stop_receiver_for_exporter).await; + drop(prometheus_health_updater); + result + })) + } else { + None + }; + // Start scraping Postgres metrics before store initialization as well. let pool_for_metrics = singleton_pool_builder.build().await?; let mut stop_receiver_for_metrics = stop_receiver.clone(); @@ -921,6 +927,7 @@ async fn run_node( Ok(()) }); let mut task_handles = vec![metrics_task, validate_chain_ids_task, version_sync_task]; + task_handles.extend(prometheus_task); // Make sure that the node storage is initialized either via genesis or snapshot recovery. ensure_storage_initialized( diff --git a/core/bin/snapshots_creator/README.md b/core/bin/snapshots_creator/README.md index 481e01551d5a..5d9b599599c1 100644 --- a/core/bin/snapshots_creator/README.md +++ b/core/bin/snapshots_creator/README.md @@ -44,7 +44,7 @@ filesystem, or Google Cloud Storage (GCS). Beware that for end-to-end testing of the main node configuration must be reflected in the external node configuration. Creating a snapshot is a part of the [snapshot recovery integration test]. You can run the test using -`yarn snapshot-recovery-test snapshot-recovery-test`. It requires the main node to be launched with a command like +`yarn recovery-test snapshot-recovery-test`. It requires the main node to be launched with a command like `zk server --components api,tree,eth,state_keeper,commitment_generator`. ## Snapshots format @@ -66,4 +66,4 @@ Each snapshot consists of three types of data (see [`snapshots.rs`] for exact de [`snapshots.rs`]: ../../lib/types/src/snapshots.rs [object store]: ../../lib/object_store -[snapshot recovery integration test]: ../../tests/snapshot-recovery-test/tests/snapshot-recovery.test.ts +[snapshot recovery integration test]: ../../tests/recovery-test/tests/snapshot-recovery.test.ts diff --git a/core/bin/system-constants-generator/src/main.rs b/core/bin/system-constants-generator/src/main.rs index 548d4c9a0ce1..b0276aeb7fa1 100644 --- a/core/bin/system-constants-generator/src/main.rs +++ b/core/bin/system-constants-generator/src/main.rs @@ -17,6 +17,7 @@ use zksync_types::{ IntrinsicSystemGasConstants, ProtocolVersionId, GUARANTEED_PUBDATA_IN_TX, L1_GAS_PER_PUBDATA_BYTE, MAX_NEW_FACTORY_DEPS, REQUIRED_L1_TO_L2_GAS_PER_PUBDATA_BYTE, }; +use zksync_utils::workspace_dir_or_current_dir; // For configs we will use the default value of `800_000` to represent the rough amount of L1 gas // needed to cover the batch expenses. @@ -209,8 +210,8 @@ fn generate_rust_fee_constants(intrinsic_gas_constants: &IntrinsicSystemGasConst } fn save_file(path_in_repo: &str, content: String) { - let zksync_home = std::env::var("ZKSYNC_HOME").expect("No ZKSYNC_HOME env var"); - let fee_constants_path = format!("{zksync_home}/{path_in_repo}"); + let zksync_home = workspace_dir_or_current_dir(); + let fee_constants_path = zksync_home.join(path_in_repo); fs::write(fee_constants_path, content) .unwrap_or_else(|_| panic!("Failed to write to {}", path_in_repo)); diff --git a/core/bin/zksync_server/Cargo.toml b/core/bin/zksync_server/Cargo.toml index 118288dfe671..a2f9067872e2 100644 --- a/core/bin/zksync_server/Cargo.toml +++ b/core/bin/zksync_server/Cargo.toml @@ -35,5 +35,10 @@ tokio = { workspace = true, features = ["full"] } tracing.workspace = true futures.workspace = true +zksync_node_framework.workspace = true +zksync_metadata_calculator.workspace = true +zksync_node_api_server.workspace = true +prometheus_exporter.workspace = true + [target.'cfg(not(target_env = "msvc"))'.dependencies] tikv-jemallocator.workspace = true diff --git a/core/bin/zksync_server/src/main.rs b/core/bin/zksync_server/src/main.rs index 8579ac04b696..955a0232ae3b 100644 --- a/core/bin/zksync_server/src/main.rs +++ b/core/bin/zksync_server/src/main.rs @@ -28,7 +28,10 @@ use zksync_eth_client::clients::Client; use zksync_storage::RocksDB; use zksync_utils::wait_for_tasks::ManagedTasks; +use crate::node_builder::MainNodeBuilder; + mod config; +mod node_builder; #[cfg(not(target_env = "msvc"))] #[global_allocator] @@ -63,6 +66,9 @@ struct Cli { /// Path to the yaml with genesis. If set, it will be used instead of env vars. #[arg(long)] genesis_path: Option, + /// Run the node using the node framework. + #[arg(long)] + use_node_framework: bool, } #[derive(Debug, Clone)] @@ -84,7 +90,6 @@ impl FromStr for ComponentsToRun { #[tokio::main] async fn main() -> anyhow::Result<()> { let opt = Cli::parse(); - let sigint_receiver = setup_sigint_handler(); // Load env config and use it if file config is not provided let tmp_config = load_env_config()?; @@ -209,7 +214,30 @@ async fn main() -> anyhow::Result<()> { opt.components.0 }; + // If the node framework is used, run the node. + if opt.use_node_framework { + // We run the node from a different thread, since the current thread is in tokio context. + std::thread::spawn(move || -> anyhow::Result<()> { + let node = MainNodeBuilder::new( + configs, + wallets, + genesis, + contracts_config, + secrets, + consensus, + ) + .build(components)?; + node.run()?; + Ok(()) + }) + .join() + .expect("Failed to run the node")?; + + return Ok(()); + } + // Run core actors. + let sigint_receiver = setup_sigint_handler(); let (core_task_handles, stop_sender, health_check_handle) = initialize_components( &configs, &wallets, @@ -263,9 +291,7 @@ fn load_env_config() -> anyhow::Result { state_keeper_config: StateKeeperConfig::from_env().ok(), house_keeper_config: HouseKeeperConfig::from_env().ok(), fri_proof_compressor_config: FriProofCompressorConfig::from_env().ok(), - fri_prover_config: FriProverConfig::from_env() - .context("fri_prover_config") - .ok(), + fri_prover_config: FriProverConfig::from_env().ok(), fri_prover_group_config: FriProverGroupConfig::from_env().ok(), fri_prover_gateway_config: FriProverGatewayConfig::from_env().ok(), fri_witness_vector_generator: FriWitnessVectorGeneratorConfig::from_env().ok(), diff --git a/core/bin/zksync_server/src/node_builder.rs b/core/bin/zksync_server/src/node_builder.rs new file mode 100644 index 000000000000..163835044cac --- /dev/null +++ b/core/bin/zksync_server/src/node_builder.rs @@ -0,0 +1,487 @@ +//! This module provides a "builder" for the main node, +//! as well as an interface to run the node with the specified components. + +use anyhow::Context; +use prometheus_exporter::PrometheusExporterConfig; +use zksync_config::{ + configs::{consensus::ConsensusConfig, wallets::Wallets, GeneralConfig, Secrets}, + ContractsConfig, GenesisConfig, +}; +use zksync_core_leftovers::Component; +use zksync_metadata_calculator::MetadataCalculatorConfig; +use zksync_node_api_server::{ + tx_sender::{ApiContracts, TxSenderConfig}, + web3::{state::InternalApiConfig, Namespace}, +}; +use zksync_node_framework::{ + implementations::layers::{ + circuit_breaker_checker::CircuitBreakerCheckerLayer, + commitment_generator::CommitmentGeneratorLayer, + consensus::{ConsensusLayer, Mode as ConsensusMode}, + contract_verification_api::ContractVerificationApiLayer, + eth_sender::{EthTxAggregatorLayer, EthTxManagerLayer}, + eth_watch::EthWatchLayer, + healtcheck_server::HealthCheckLayer, + house_keeper::HouseKeeperLayer, + l1_gas::SequencerL1GasLayer, + metadata_calculator::MetadataCalculatorLayer, + object_store::ObjectStoreLayer, + pk_signing_eth_client::PKSigningEthClientLayer, + pools_layer::PoolsLayerBuilder, + prometheus_exporter::PrometheusExporterLayer, + proof_data_handler::ProofDataHandlerLayer, + query_eth_client::QueryEthClientLayer, + sigint::SigintHandlerLayer, + state_keeper::{ + main_batch_executor::MainBatchExecutorLayer, mempool_io::MempoolIOLayer, + StateKeeperLayer, + }, + tee_verifier_input_producer::TeeVerifierInputProducerLayer, + web3_api::{ + caches::MempoolCacheLayer, + server::{Web3ServerLayer, Web3ServerOptionalConfig}, + tree_api_client::TreeApiClientLayer, + tx_sender::{PostgresStorageCachesConfig, TxSenderLayer}, + tx_sink::TxSinkLayer, + }, + }, + service::{ZkStackService, ZkStackServiceBuilder}, +}; + +/// Macro that looks into a path to fetch an optional config, +/// and clones it into a variable. +macro_rules! try_load_config { + ($path:expr) => { + $path.as_ref().context(stringify!($path))?.clone() + }; +} + +pub struct MainNodeBuilder { + node: ZkStackServiceBuilder, + configs: GeneralConfig, + wallets: Wallets, + genesis_config: GenesisConfig, + contracts_config: ContractsConfig, + secrets: Secrets, + consensus_config: Option, +} + +impl MainNodeBuilder { + pub fn new( + configs: GeneralConfig, + wallets: Wallets, + genesis_config: GenesisConfig, + contracts_config: ContractsConfig, + secrets: Secrets, + consensus_config: Option, + ) -> Self { + Self { + node: ZkStackServiceBuilder::new(), + configs, + wallets, + genesis_config, + contracts_config, + secrets, + consensus_config, + } + } + + fn add_sigint_handler_layer(mut self) -> anyhow::Result { + self.node.add_layer(SigintHandlerLayer); + Ok(self) + } + + fn add_pools_layer(mut self) -> anyhow::Result { + let config = try_load_config!(self.configs.postgres_config); + let secrets = try_load_config!(self.secrets.database); + let pools_layer = PoolsLayerBuilder::empty(config, secrets) + .with_master(true) + .with_replica(true) + .with_prover(true) // Used by house keeper. + .build(); + self.node.add_layer(pools_layer); + Ok(self) + } + + fn add_prometheus_exporter_layer(mut self) -> anyhow::Result { + let prom_config = try_load_config!(self.configs.prometheus_config); + let prom_config = PrometheusExporterConfig::pull(prom_config.listener_port); + self.node.add_layer(PrometheusExporterLayer(prom_config)); + Ok(self) + } + + fn add_pk_signing_client_layer(mut self) -> anyhow::Result { + let eth_config = try_load_config!(self.configs.eth); + let wallets = try_load_config!(self.wallets.eth_sender); + self.node.add_layer(PKSigningEthClientLayer::new( + eth_config, + self.contracts_config.clone(), + self.genesis_config.l1_chain_id, + wallets, + )); + Ok(self) + } + + fn add_query_eth_client_layer(mut self) -> anyhow::Result { + let genesis = self.genesis_config.clone(); + let eth_config = try_load_config!(self.secrets.l1); + let query_eth_client_layer = + QueryEthClientLayer::new(genesis.l1_chain_id, eth_config.l1_rpc_url); + self.node.add_layer(query_eth_client_layer); + Ok(self) + } + + fn add_sequencer_l1_gas_layer(mut self) -> anyhow::Result { + let gas_adjuster_config = try_load_config!(self.configs.eth) + .gas_adjuster + .context("Gas adjuster")?; + let state_keeper_config = try_load_config!(self.configs.state_keeper_config); + let eth_sender_config = try_load_config!(self.configs.eth); + let sequencer_l1_gas_layer = SequencerL1GasLayer::new( + gas_adjuster_config, + self.genesis_config.clone(), + state_keeper_config, + try_load_config!(eth_sender_config.sender).pubdata_sending_mode, + ); + self.node.add_layer(sequencer_l1_gas_layer); + Ok(self) + } + + fn add_object_store_layer(mut self) -> anyhow::Result { + let object_store_config = try_load_config!(self.configs.prover_config) + .object_store + .context("object_store_config")?; + self.node + .add_layer(ObjectStoreLayer::new(object_store_config)); + Ok(self) + } + + fn add_metadata_calculator_layer(mut self, with_tree_api: bool) -> anyhow::Result { + let merkle_tree_env_config = try_load_config!(self.configs.db_config).merkle_tree; + let operations_manager_env_config = + try_load_config!(self.configs.operations_manager_config); + let metadata_calculator_config = MetadataCalculatorConfig::for_main_node( + &merkle_tree_env_config, + &operations_manager_env_config, + ); + let mut layer = MetadataCalculatorLayer::new(metadata_calculator_config); + if with_tree_api { + let merkle_tree_api_config = try_load_config!(self.configs.api_config).merkle_tree; + layer = layer.with_tree_api_config(merkle_tree_api_config); + } + self.node.add_layer(layer); + Ok(self) + } + + fn add_state_keeper_layer(mut self) -> anyhow::Result { + let wallets = self.wallets.clone(); + let sk_config = try_load_config!(self.configs.state_keeper_config); + let mempool_io_layer = MempoolIOLayer::new( + self.genesis_config.l2_chain_id, + self.contracts_config.clone(), + sk_config.clone(), + try_load_config!(self.configs.mempool_config), + try_load_config!(wallets.state_keeper), + ); + let db_config = try_load_config!(self.configs.db_config); + let main_node_batch_executor_builder_layer = MainBatchExecutorLayer::new(sk_config); + let state_keeper_layer = StateKeeperLayer::new(db_config); + self.node + .add_layer(mempool_io_layer) + .add_layer(main_node_batch_executor_builder_layer) + .add_layer(state_keeper_layer); + Ok(self) + } + + fn add_eth_watch_layer(mut self) -> anyhow::Result { + let eth_config = try_load_config!(self.configs.eth); + self.node.add_layer(EthWatchLayer::new( + try_load_config!(eth_config.watcher), + self.contracts_config.clone(), + )); + Ok(self) + } + + fn add_proof_data_handler_layer(mut self) -> anyhow::Result { + self.node.add_layer(ProofDataHandlerLayer::new( + try_load_config!(self.configs.proof_data_handler_config), + self.genesis_config.l1_batch_commit_data_generator_mode, + )); + Ok(self) + } + + fn add_healthcheck_layer(mut self) -> anyhow::Result { + let healthcheck_config = try_load_config!(self.configs.api_config).healthcheck; + self.node.add_layer(HealthCheckLayer(healthcheck_config)); + Ok(self) + } + + fn add_tx_sender_layer(mut self) -> anyhow::Result { + let sk_config = try_load_config!(self.configs.state_keeper_config); + let rpc_config = try_load_config!(self.configs.api_config).web3_json_rpc; + let postgres_storage_caches_config = PostgresStorageCachesConfig { + factory_deps_cache_size: rpc_config.factory_deps_cache_size() as u64, + initial_writes_cache_size: rpc_config.initial_writes_cache_size() as u64, + latest_values_cache_size: rpc_config.latest_values_cache_size() as u64, + }; + + // On main node we always use master pool sink. + self.node.add_layer(TxSinkLayer::MasterPoolSink); + self.node.add_layer(TxSenderLayer::new( + TxSenderConfig::new( + &sk_config, + &rpc_config, + try_load_config!(self.wallets.state_keeper) + .fee_account + .address(), + self.genesis_config.l2_chain_id, + ), + postgres_storage_caches_config, + rpc_config.vm_concurrency_limit(), + ApiContracts::load_from_disk(), // TODO (BFT-138): Allow to dynamically reload API contracts + )); + Ok(self) + } + + fn add_api_caches_layer(mut self) -> anyhow::Result { + let rpc_config = try_load_config!(self.configs.api_config).web3_json_rpc; + self.node.add_layer(MempoolCacheLayer::new( + rpc_config.mempool_cache_size(), + rpc_config.mempool_cache_update_interval(), + )); + Ok(self) + } + + fn add_tree_api_client_layer(mut self) -> anyhow::Result { + let rpc_config = try_load_config!(self.configs.api_config).web3_json_rpc; + self.node + .add_layer(TreeApiClientLayer::http(rpc_config.tree_api_url)); + Ok(self) + } + + fn add_http_web3_api_layer(mut self) -> anyhow::Result { + let rpc_config = try_load_config!(self.configs.api_config).web3_json_rpc; + let state_keeper_config = try_load_config!(self.configs.state_keeper_config); + let with_debug_namespace = state_keeper_config.save_call_traces; + + let mut namespaces = Namespace::DEFAULT.to_vec(); + if with_debug_namespace { + namespaces.push(Namespace::Debug) + } + namespaces.push(Namespace::Snapshots); + + let optional_config = Web3ServerOptionalConfig { + namespaces: Some(namespaces), + filters_limit: Some(rpc_config.filters_limit()), + subscriptions_limit: Some(rpc_config.subscriptions_limit()), + batch_request_size_limit: Some(rpc_config.max_batch_request_size()), + response_body_size_limit: Some(rpc_config.max_response_body_size()), + ..Default::default() + }; + self.node.add_layer(Web3ServerLayer::http( + rpc_config.http_port, + InternalApiConfig::new(&rpc_config, &self.contracts_config, &self.genesis_config), + optional_config, + )); + + Ok(self) + } + + fn add_ws_web3_api_layer(mut self) -> anyhow::Result { + let rpc_config = try_load_config!(self.configs.api_config).web3_json_rpc; + let state_keeper_config = try_load_config!(self.configs.state_keeper_config); + let circuit_breaker_config = try_load_config!(self.configs.circuit_breaker_config); + let with_debug_namespace = state_keeper_config.save_call_traces; + + let mut namespaces = Namespace::DEFAULT.to_vec(); + if with_debug_namespace { + namespaces.push(Namespace::Debug) + } + namespaces.push(Namespace::Snapshots); + + let optional_config = Web3ServerOptionalConfig { + namespaces: Some(namespaces), + filters_limit: Some(rpc_config.filters_limit()), + subscriptions_limit: Some(rpc_config.subscriptions_limit()), + batch_request_size_limit: Some(rpc_config.max_batch_request_size()), + response_body_size_limit: Some(rpc_config.max_response_body_size()), + websocket_requests_per_minute_limit: Some( + rpc_config.websocket_requests_per_minute_limit(), + ), + replication_lag_limit: circuit_breaker_config.replication_lag_limit(), + }; + self.node.add_layer(Web3ServerLayer::ws( + rpc_config.ws_port, + InternalApiConfig::new(&rpc_config, &self.contracts_config, &self.genesis_config), + optional_config, + )); + + Ok(self) + } + + fn add_eth_tx_manager_layer(mut self) -> anyhow::Result { + let eth_sender_config = try_load_config!(self.configs.eth); + + self.node + .add_layer(EthTxManagerLayer::new(eth_sender_config)); + + Ok(self) + } + + fn add_eth_tx_aggregator_layer(mut self) -> anyhow::Result { + let eth_sender_config = try_load_config!(self.configs.eth); + + self.node.add_layer(EthTxAggregatorLayer::new( + eth_sender_config, + self.contracts_config.clone(), + self.genesis_config.l2_chain_id, + self.genesis_config.l1_batch_commit_data_generator_mode, + )); + + Ok(self) + } + + fn add_house_keeper_layer(mut self) -> anyhow::Result { + let house_keeper_config = try_load_config!(self.configs.house_keeper_config); + let fri_prover_config = try_load_config!(self.configs.prover_config); + let fri_witness_generator_config = try_load_config!(self.configs.witness_generator); + let fri_prover_group_config = try_load_config!(self.configs.prover_group_config); + let fri_proof_compressor_config = try_load_config!(self.configs.proof_compressor_config); + + self.node.add_layer(HouseKeeperLayer::new( + house_keeper_config, + fri_prover_config, + fri_witness_generator_config, + fri_prover_group_config, + fri_proof_compressor_config, + )); + + Ok(self) + } + + fn add_commitment_generator_layer(mut self) -> anyhow::Result { + self.node.add_layer(CommitmentGeneratorLayer::new( + self.genesis_config.l1_batch_commit_data_generator_mode, + )); + + Ok(self) + } + + fn add_circuit_breaker_checker_layer(mut self) -> anyhow::Result { + let circuit_breaker_config = try_load_config!(self.configs.circuit_breaker_config); + self.node + .add_layer(CircuitBreakerCheckerLayer(circuit_breaker_config)); + + Ok(self) + } + + fn add_contract_verification_api_layer(mut self) -> anyhow::Result { + let config = try_load_config!(self.configs.contract_verifier); + self.node.add_layer(ContractVerificationApiLayer(config)); + Ok(self) + } + + fn add_consensus_layer(mut self) -> anyhow::Result { + self.node.add_layer(ConsensusLayer { + mode: ConsensusMode::Main, + config: self.consensus_config.clone(), + secrets: self.secrets.consensus.clone(), + }); + + Ok(self) + } + + fn add_tee_verifier_input_producer_layer(mut self) -> anyhow::Result { + self.node.add_layer(TeeVerifierInputProducerLayer::new( + self.genesis_config.l2_chain_id, + )); + + Ok(self) + } + + pub fn build(mut self, mut components: Vec) -> anyhow::Result { + // Add "base" layers (resources and helper tasks). + self = self + .add_sigint_handler_layer()? + .add_pools_layer()? + .add_object_store_layer()? + .add_circuit_breaker_checker_layer()? + .add_healthcheck_layer()? + .add_prometheus_exporter_layer()? + .add_query_eth_client_layer()? + .add_sequencer_l1_gas_layer()?; + + // Sort the components, so that the components they may depend on each other are added in the correct order. + components.sort_unstable_by_key(|component| match component { + // API consumes the resources provided by other layers (multiple ones), so it has to come the last. + Component::HttpApi | Component::WsApi => 1, + // Default priority. + _ => 0, + }); + + // Add "component-specific" layers. + // Note that the layers are added only once, so it's fine to add the same layer multiple times. + for component in &components { + match component { + Component::HttpApi => { + self = self + .add_tx_sender_layer()? + .add_tree_api_client_layer()? + .add_api_caches_layer()? + .add_http_web3_api_layer()?; + } + Component::WsApi => { + self = self + .add_tx_sender_layer()? + .add_tree_api_client_layer()? + .add_api_caches_layer()? + .add_ws_web3_api_layer()?; + } + Component::ContractVerificationApi => { + self = self.add_contract_verification_api_layer()?; + } + Component::Tree => { + let with_tree_api = components.contains(&Component::TreeApi); + self = self.add_metadata_calculator_layer(with_tree_api)?; + } + Component::TreeApi => { + anyhow::ensure!( + components.contains(&Component::Tree), + "Merkle tree API cannot be started without a tree component" + ); + // Do nothing, will be handled by the `Tree` component. + } + Component::EthWatcher => { + self = self.add_eth_watch_layer()?; + } + Component::EthTxAggregator => { + self = self + .add_pk_signing_client_layer()? + .add_eth_tx_aggregator_layer()?; + } + Component::EthTxManager => { + self = self.add_eth_tx_manager_layer()?; + } + Component::StateKeeper => { + self = self.add_state_keeper_layer()?; + } + Component::TeeVerifierInputProducer => { + self = self.add_tee_verifier_input_producer_layer()?; + } + Component::Housekeeper => { + self = self.add_house_keeper_layer()?; + } + Component::ProofDataHandler => { + self = self.add_proof_data_handler_layer()?; + } + Component::Consensus => { + self = self.add_consensus_layer()?; + } + Component::CommitmentGenerator => { + self = self.add_commitment_generator_layer()?; + } + } + } + Ok(self.node.build()?) + } +} diff --git a/core/lib/basic_types/src/protocol_version.rs b/core/lib/basic_types/src/protocol_version.rs index b5d15e6cbc70..1ba41c47aee7 100644 --- a/core/lib/basic_types/src/protocol_version.rs +++ b/core/lib/basic_types/src/protocol_version.rs @@ -1,4 +1,7 @@ -use std::convert::{TryFrom, TryInto}; +use std::{ + convert::{TryFrom, TryInto}, + fmt, +}; use num_enum::TryFromPrimitive; use serde::{Deserialize, Serialize}; @@ -158,6 +161,12 @@ impl Default for ProtocolVersionId { } } +impl fmt::Display for ProtocolVersionId { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!(f, "{}", *self as u16) + } +} + impl TryFrom for ProtocolVersionId { type Error = String; diff --git a/core/lib/config/src/configs/eth_sender.rs b/core/lib/config/src/configs/eth_sender.rs index 8768bd62160d..58b81fa0a145 100644 --- a/core/lib/config/src/configs/eth_sender.rs +++ b/core/lib/config/src/configs/eth_sender.rs @@ -39,7 +39,6 @@ impl EthConfig { timestamp_criteria_max_allowed_lag: 30, l1_batch_min_age_before_execute_seconds: None, max_acceptable_priority_fee_in_gwei: 100000000000, - proof_loading_mode: ProofLoadingMode::OldProofFromDb, pubdata_sending_mode: PubdataSendingMode::Calldata, }), gas_adjuster: Some(GasAdjusterConfig { @@ -115,9 +114,6 @@ pub struct SenderConfig { // Max acceptable fee for sending tx it acts as a safeguard to prevent sending tx with very high fees. pub max_acceptable_priority_fee_in_gwei: u64, - /// The mode in which proofs are loaded, either from DB/GCS for FRI/Old proof. - pub proof_loading_mode: ProofLoadingMode, - /// The mode in which we send pubdata, either Calldata or Blobs pub pubdata_sending_mode: PubdataSendingMode, } diff --git a/core/lib/config/src/configs/fri_proof_compressor.rs b/core/lib/config/src/configs/fri_proof_compressor.rs index 4b4e062dee28..0fceac509aca 100644 --- a/core/lib/config/src/configs/fri_proof_compressor.rs +++ b/core/lib/config/src/configs/fri_proof_compressor.rs @@ -20,7 +20,7 @@ pub struct FriProofCompressorConfig { /// Path to universal setup key file pub universal_setup_path: String, - /// https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2\^26.key + /// https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2\^24.key pub universal_setup_download_url: String, // Whether to verify wrapper proof or not. diff --git a/core/lib/config/src/testonly.rs b/core/lib/config/src/testonly.rs index f914a0390d43..0e99c57b9fac 100644 --- a/core/lib/config/src/testonly.rs +++ b/core/lib/config/src/testonly.rs @@ -366,7 +366,6 @@ impl Distribution for EncodeDist { timestamp_criteria_max_allowed_lag: self.sample(rng), l1_batch_min_age_before_execute_seconds: self.sample(rng), max_acceptable_priority_fee_in_gwei: self.sample(rng), - proof_loading_mode: self.sample(rng), pubdata_sending_mode: PubdataSendingMode::Calldata, } } diff --git a/core/lib/contracts/src/lib.rs b/core/lib/contracts/src/lib.rs index 285f9f0430e7..6ab80e18e943 100644 --- a/core/lib/contracts/src/lib.rs +++ b/core/lib/contracts/src/lib.rs @@ -1,6 +1,6 @@ //! Set of utility functions to read contracts both in Yul and Sol format. //! -//! Careful: some of the methods are reading the contracts based on the ZKSYNC_HOME environment variable. +//! Careful: some of the methods are reading the contracts based on the workspace environment variable. #![allow(clippy::derive_partial_eq_without_eq)] @@ -15,7 +15,7 @@ use ethabi::{ }; use once_cell::sync::Lazy; use serde::{Deserialize, Serialize}; -use zksync_utils::{bytecode::hash_bytecode, bytes_to_be_words}; +use zksync_utils::{bytecode::hash_bytecode, bytes_to_be_words, workspace_dir_or_current_dir}; pub mod test_contracts; @@ -25,31 +25,44 @@ pub enum ContractLanguage { Yul, } -const BRIDGEHUB_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/bridgehub/IBridgehub.sol/IBridgehub.json"; -const STATE_TRANSITION_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/state-transition/IStateTransitionManager.sol/IStateTransitionManager.json"; -const ZKSYNC_HYPERCHAIN_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/state-transition/chain-interfaces/IZkSyncHyperchain.sol/IZkSyncHyperchain.json"; -const DIAMOND_INIT_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/state-transition/chain-interfaces/IDiamondInit.sol/IDiamondInit.json"; -const GOVERNANCE_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/governance/IGovernance.sol/IGovernance.json"; -const MULTICALL3_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/dev-contracts/Multicall3.sol/Multicall3.json"; -const VERIFIER_CONTRACT_FILE: &str = - "contracts/l1-contracts/artifacts/contracts/state-transition/Verifier.sol/Verifier.json"; +/// During the transition period we have to support both paths for contracts artifacts +/// One for forge and another for hardhat. +/// Meanwhile, hardhat has one more intermediate folder. That's why, we have to represent each contract +/// by two constants, intermediate folder and actual contract name. For Forge we use only second part +const HARDHAT_PATH_PREFIX: &str = "contracts/l1-contracts/artifacts/contracts"; +const FORGE_PATH_PREFIX: &str = "contracts/l1-contracts-foundry/out"; + +const BRIDGEHUB_CONTRACT_FILE: (&str, &str) = ("bridgehub", "IBridgehub.sol/IBridgehub.json"); +const STATE_TRANSITION_CONTRACT_FILE: (&str, &str) = ( + "state-transition", + "IStateTransitionManager.sol/IStateTransitionManager.json", +); +const ZKSYNC_HYPERCHAIN_CONTRACT_FILE: (&str, &str) = ( + "state-transition/", + "chain-interfaces/IZkSyncHyperchain.sol/IZkSyncHyperchain.json", +); +const DIAMOND_INIT_CONTRACT_FILE: (&str, &str) = ( + "state-transition", + "chain-interfaces/IDiamondInit.sol/IDiamondInit.json", +); +const GOVERNANCE_CONTRACT_FILE: (&str, &str) = ("governance", "IGovernance.sol/IGovernance.json"); +const MULTICALL3_CONTRACT_FILE: (&str, &str) = ("dev-contracts", "Multicall3.sol/Multicall3.json"); +const VERIFIER_CONTRACT_FILE: (&str, &str) = ("state-transition", "Verifier.sol/Verifier.json"); const _IERC20_CONTRACT_FILE: &str = "contracts/l1-contracts/artifacts/contracts/common/interfaces/IERC20.sol/IERC20.json"; -const _FAIL_ON_RECEIVE_CONTRACT_FILE: &str = +const _FAIL_ON_RECEIVE_CONTRACT_FILE: &str = "contracts/l1-contracts/artifacts/contracts/zksync/dev-contracts/FailOnReceive.sol/FailOnReceive.json"; const LOADNEXT_CONTRACT_FILE: &str = "etc/contracts-test-data/artifacts-zk/contracts/loadnext/loadnext_contract.sol/LoadnextContract.json"; const LOADNEXT_SIMPLE_CONTRACT_FILE: &str = "etc/contracts-test-data/artifacts-zk/contracts/loadnext/loadnext_contract.sol/Foo.json"; -fn read_file_to_json_value(path: impl AsRef) -> serde_json::Value { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); +fn home_path() -> &'static Path { + workspace_dir_or_current_dir() +} + +fn read_file_to_json_value(path: impl AsRef + std::fmt::Debug) -> serde_json::Value { + let zksync_home = home_path(); let path = Path::new(&zksync_home).join(path); serde_json::from_reader( File::open(&path).unwrap_or_else(|e| panic!("Failed to open file {:?}: {}", path, e)), @@ -58,7 +71,7 @@ fn read_file_to_json_value(path: impl AsRef) -> serde_json::Value { } fn load_contract_if_present + std::fmt::Debug>(path: P) -> Option { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); + let zksync_home = home_path(); let path = Path::new(&zksync_home).join(path); path.exists().then(|| { serde_json::from_value(read_file_to_json_value(&path)["abi"].take()) @@ -66,6 +79,26 @@ fn load_contract_if_present + std::fmt::Debug>(path: P) -> Option }) } +fn load_contract_for_hardhat(path: (&str, &str)) -> Option { + let path = Path::new(HARDHAT_PATH_PREFIX).join(path.0).join(path.1); + load_contract_if_present(path) +} + +fn load_contract_for_forge(file_path: &str) -> Option { + let path = Path::new(FORGE_PATH_PREFIX).join(file_path); + load_contract_if_present(path) +} + +fn load_contract_for_both_compilers(path: (&str, &str)) -> Contract { + if let Some(contract) = load_contract_for_forge(path.1) { + return contract; + }; + + load_contract_for_hardhat(path).unwrap_or_else(|| { + panic!("Failed to load contract from {:?}", path); + }) +} + pub fn load_contract + std::fmt::Debug>(path: P) -> Contract { load_contract_if_present(&path).unwrap_or_else(|| { panic!("Failed to load contract from {:?}", path); @@ -79,7 +112,7 @@ pub fn load_sys_contract(contract_name: &str) -> Contract { )) } -pub fn read_contract_abi(path: impl AsRef) -> String { +pub fn read_contract_abi(path: impl AsRef + std::fmt::Debug) -> String { read_file_to_json_value(path)["abi"] .as_str() .expect("Failed to parse abi") @@ -87,31 +120,31 @@ pub fn read_contract_abi(path: impl AsRef) -> String { } pub fn bridgehub_contract() -> Contract { - load_contract(BRIDGEHUB_CONTRACT_FILE) + load_contract_for_both_compilers(BRIDGEHUB_CONTRACT_FILE) } pub fn governance_contract() -> Contract { - load_contract_if_present(GOVERNANCE_CONTRACT_FILE).expect("Governance contract not found") + load_contract_for_both_compilers(GOVERNANCE_CONTRACT_FILE) } pub fn state_transition_manager_contract() -> Contract { - load_contract(STATE_TRANSITION_CONTRACT_FILE) + load_contract_for_both_compilers(STATE_TRANSITION_CONTRACT_FILE) } pub fn hyperchain_contract() -> Contract { - load_contract(ZKSYNC_HYPERCHAIN_CONTRACT_FILE) + load_contract_for_both_compilers(ZKSYNC_HYPERCHAIN_CONTRACT_FILE) } pub fn diamond_init_contract() -> Contract { - load_contract(DIAMOND_INIT_CONTRACT_FILE) + load_contract_for_both_compilers(DIAMOND_INIT_CONTRACT_FILE) } pub fn multicall_contract() -> Contract { - load_contract(MULTICALL3_CONTRACT_FILE) + load_contract_for_both_compilers(MULTICALL3_CONTRACT_FILE) } pub fn verifier_contract() -> Contract { - load_contract(VERIFIER_CONTRACT_FILE) + load_contract_for_both_compilers(VERIFIER_CONTRACT_FILE) } #[derive(Debug, Clone)] @@ -149,6 +182,11 @@ pub fn l1_messenger_contract() -> Contract { load_sys_contract("L1Messenger") } +/// Reads bytecode from the path RELATIVE to the Cargo workspace location. +pub fn read_bytecode(relative_path: impl AsRef + std::fmt::Debug) -> Vec { + read_bytecode_from_path(relative_path) +} + pub fn eth_contract() -> Contract { load_sys_contract("L2BaseToken") } @@ -157,16 +195,9 @@ pub fn known_codes_contract() -> Contract { load_sys_contract("KnownCodesStorage") } -/// Reads bytecode from the path RELATIVE to the ZKSYNC_HOME environment variable. -pub fn read_bytecode(relative_path: impl AsRef) -> Vec { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); - let artifact_path = Path::new(&zksync_home).join(relative_path); - read_bytecode_from_path(artifact_path) -} - /// Reads bytecode from a given path. -fn read_bytecode_from_path(artifact_path: PathBuf) -> Vec { - let artifact = read_file_to_json_value(artifact_path.clone()); +fn read_bytecode_from_path(artifact_path: impl AsRef + std::fmt::Debug) -> Vec { + let artifact = read_file_to_json_value(&artifact_path); let bytecode = artifact["bytecode"] .as_str() @@ -187,19 +218,17 @@ static DEFAULT_SYSTEM_CONTRACTS_REPO: Lazy = /// Structure representing a system contract repository - that allows /// fetching contracts that are located there. -/// As most of the static methods in this file, is loading data based on ZKSYNC_HOME environment variable. +/// As most of the static methods in this file, is loading data based on the Cargo workspace location. pub struct SystemContractsRepo { // Path to the root of the system contracts repository. pub root: PathBuf, } impl SystemContractsRepo { - /// Returns the default system contracts repository with directory based on the ZKSYNC_HOME environment variable. + /// Returns the default system contracts repository with directory based on the Cargo workspace location. pub fn from_env() -> Self { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); - let zksync_home = PathBuf::from(zksync_home); SystemContractsRepo { - root: zksync_home.join("contracts/system-contracts"), + root: home_path().join("contracts/system-contracts"), } } @@ -237,10 +266,9 @@ fn read_playground_batch_bootloader_bytecode() -> Vec { read_bootloader_code("playground_batch") } -/// Reads zbin bytecode from a given path, relative to ZKSYNC_HOME. +/// Reads zbin bytecode from a given path, relative to workspace location. pub fn read_zbin_bytecode(relative_zbin_path: impl AsRef) -> Vec { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| ".".into()); - let bytecode_path = Path::new(&zksync_home).join(relative_zbin_path); + let bytecode_path = Path::new(&home_path()).join(relative_zbin_path); read_zbin_bytecode_from_path(bytecode_path) } diff --git a/core/lib/dal/.sqlx/query-120970162104e0560784ee4b8fa44a0202265d741912125f7865e570411997d7.json b/core/lib/dal/.sqlx/query-120970162104e0560784ee4b8fa44a0202265d741912125f7865e570411997d7.json new file mode 100644 index 000000000000..00e558b13628 --- /dev/null +++ b/core/lib/dal/.sqlx/query-120970162104e0560784ee4b8fa44a0202265d741912125f7865e570411997d7.json @@ -0,0 +1,15 @@ +{ + "db_name": "PostgreSQL", + "query": "\n UPDATE l1_batches\n SET\n tree_writes = $1\n WHERE\n number = $2\n ", + "describe": { + "columns": [], + "parameters": { + "Left": [ + "Bytea", + "Int8" + ] + }, + "nullable": [] + }, + "hash": "120970162104e0560784ee4b8fa44a0202265d741912125f7865e570411997d7" +} diff --git a/core/lib/dal/.sqlx/query-148dd243ab476724a430e74406119a148b59a79b03dacf3b1c32223c5ebf8d4b.json b/core/lib/dal/.sqlx/query-148dd243ab476724a430e74406119a148b59a79b03dacf3b1c32223c5ebf8d4b.json new file mode 100644 index 000000000000..4f14d753fd6f --- /dev/null +++ b/core/lib/dal/.sqlx/query-148dd243ab476724a430e74406119a148b59a79b03dacf3b1c32223c5ebf8d4b.json @@ -0,0 +1,20 @@ +{ + "db_name": "PostgreSQL", + "query": "\n SELECT\n number\n FROM\n l1_batches\n WHERE\n hash IS NOT NULL\n AND commitment IS NULL\n ORDER BY\n number DESC\n LIMIT\n 1\n ", + "describe": { + "columns": [ + { + "ordinal": 0, + "name": "number", + "type_info": "Int8" + } + ], + "parameters": { + "Left": [] + }, + "nullable": [ + false + ] + }, + "hash": "148dd243ab476724a430e74406119a148b59a79b03dacf3b1c32223c5ebf8d4b" +} diff --git a/core/lib/dal/.sqlx/query-16e1a17bfc426bb32489595bd8cccb1ef34292fcf694deddc06b6dd5b72a02f3.json b/core/lib/dal/.sqlx/query-294005d0b9445cc8b9c8e4ce7453f71664dcb5ebbc35005a18c5251c3d902f62.json similarity index 50% rename from core/lib/dal/.sqlx/query-16e1a17bfc426bb32489595bd8cccb1ef34292fcf694deddc06b6dd5b72a02f3.json rename to core/lib/dal/.sqlx/query-294005d0b9445cc8b9c8e4ce7453f71664dcb5ebbc35005a18c5251c3d902f62.json index 479bc818b9bb..e2aeb15b19af 100644 --- a/core/lib/dal/.sqlx/query-16e1a17bfc426bb32489595bd8cccb1ef34292fcf694deddc06b6dd5b72a02f3.json +++ b/core/lib/dal/.sqlx/query-294005d0b9445cc8b9c8e4ce7453f71664dcb5ebbc35005a18c5251c3d902f62.json @@ -1,6 +1,6 @@ { "db_name": "PostgreSQL", - "query": "\n SELECT\n MAX(INDEX) AS \"max?\"\n FROM\n initial_writes\n WHERE\n l1_batch_number = $1\n ", + "query": "\n SELECT\n MAX(INDEX) AS \"max?\"\n FROM\n initial_writes\n WHERE\n l1_batch_number = (\n SELECT\n MAX(l1_batch_number) AS \"max?\"\n FROM\n initial_writes\n WHERE\n l1_batch_number <= $1\n )\n ", "describe": { "columns": [ { @@ -18,5 +18,5 @@ null ] }, - "hash": "16e1a17bfc426bb32489595bd8cccb1ef34292fcf694deddc06b6dd5b72a02f3" + "hash": "294005d0b9445cc8b9c8e4ce7453f71664dcb5ebbc35005a18c5251c3d902f62" } diff --git a/core/lib/dal/.sqlx/query-730095f41fd5e2ea376fd869887be82028e2865e646677ff67d0720ad17b1eac.json b/core/lib/dal/.sqlx/query-730095f41fd5e2ea376fd869887be82028e2865e646677ff67d0720ad17b1eac.json new file mode 100644 index 000000000000..e6d2f72575e3 --- /dev/null +++ b/core/lib/dal/.sqlx/query-730095f41fd5e2ea376fd869887be82028e2865e646677ff67d0720ad17b1eac.json @@ -0,0 +1,22 @@ +{ + "db_name": "PostgreSQL", + "query": "\n SELECT\n tree_writes\n FROM\n l1_batches\n WHERE\n number = $1\n ", + "describe": { + "columns": [ + { + "ordinal": 0, + "name": "tree_writes", + "type_info": "Bytea" + } + ], + "parameters": { + "Left": [ + "Int8" + ] + }, + "nullable": [ + true + ] + }, + "hash": "730095f41fd5e2ea376fd869887be82028e2865e646677ff67d0720ad17b1eac" +} diff --git a/core/lib/dal/.sqlx/query-77de28ce78e1e5827f03d7e7550aec881fb5d3cde2f3aad9a5db5629070d6b7c.json b/core/lib/dal/.sqlx/query-77de28ce78e1e5827f03d7e7550aec881fb5d3cde2f3aad9a5db5629070d6b7c.json new file mode 100644 index 000000000000..6fc747ea1237 --- /dev/null +++ b/core/lib/dal/.sqlx/query-77de28ce78e1e5827f03d7e7550aec881fb5d3cde2f3aad9a5db5629070d6b7c.json @@ -0,0 +1,22 @@ +{ + "db_name": "PostgreSQL", + "query": "\n SELECT\n (tree_writes IS NOT NULL) AS \"tree_writes_are_present!\"\n FROM\n l1_batches\n WHERE\n number = $1\n ", + "describe": { + "columns": [ + { + "ordinal": 0, + "name": "tree_writes_are_present!", + "type_info": "Bool" + } + ], + "parameters": { + "Left": [ + "Int8" + ] + }, + "nullable": [ + null + ] + }, + "hash": "77de28ce78e1e5827f03d7e7550aec881fb5d3cde2f3aad9a5db5629070d6b7c" +} diff --git a/core/lib/dal/.sqlx/query-a79a53e2510c5dabe08b6341cff304af1c40bad69b8646b6db5f8c33f10f6fb5.json b/core/lib/dal/.sqlx/query-a79a53e2510c5dabe08b6341cff304af1c40bad69b8646b6db5f8c33f10f6fb5.json deleted file mode 100644 index afea22f62e8c..000000000000 --- a/core/lib/dal/.sqlx/query-a79a53e2510c5dabe08b6341cff304af1c40bad69b8646b6db5f8c33f10f6fb5.json +++ /dev/null @@ -1,40 +0,0 @@ -{ - "db_name": "PostgreSQL", - "query": "\n SELECT\n miniblocks.hash,\n miniblocks.number,\n prev_miniblock.hash AS \"parent_hash?\",\n miniblocks.timestamp\n FROM\n miniblocks\n LEFT JOIN miniblocks prev_miniblock ON prev_miniblock.number = miniblocks.number - 1\n WHERE\n miniblocks.number > $1\n ORDER BY\n miniblocks.number ASC\n ", - "describe": { - "columns": [ - { - "ordinal": 0, - "name": "hash", - "type_info": "Bytea" - }, - { - "ordinal": 1, - "name": "number", - "type_info": "Int8" - }, - { - "ordinal": 2, - "name": "parent_hash?", - "type_info": "Bytea" - }, - { - "ordinal": 3, - "name": "timestamp", - "type_info": "Int8" - } - ], - "parameters": { - "Left": [ - "Int8" - ] - }, - "nullable": [ - false, - false, - false, - false - ] - }, - "hash": "a79a53e2510c5dabe08b6341cff304af1c40bad69b8646b6db5f8c33f10f6fb5" -} diff --git a/core/lib/dal/.sqlx/query-e28e052dbd306ba408dd26e01c38176f2f9dbba45761a2117c185912d1e07bdf.json b/core/lib/dal/.sqlx/query-e28e052dbd306ba408dd26e01c38176f2f9dbba45761a2117c185912d1e07bdf.json new file mode 100644 index 000000000000..580a5370c89d --- /dev/null +++ b/core/lib/dal/.sqlx/query-e28e052dbd306ba408dd26e01c38176f2f9dbba45761a2117c185912d1e07bdf.json @@ -0,0 +1,64 @@ +{ + "db_name": "PostgreSQL", + "query": "\n SELECT\n miniblocks.hash AS \"block_hash\",\n miniblocks.number AS \"block_number\",\n prev_miniblock.hash AS \"parent_hash?\",\n miniblocks.timestamp AS \"block_timestamp\",\n miniblocks.base_fee_per_gas AS \"base_fee_per_gas\",\n miniblocks.gas_limit AS \"block_gas_limit?\",\n transactions.gas_limit AS \"transaction_gas_limit?\",\n transactions.refunded_gas AS \"transaction_refunded_gas?\"\n FROM\n miniblocks\n LEFT JOIN miniblocks prev_miniblock ON prev_miniblock.number = miniblocks.number - 1\n LEFT JOIN transactions ON transactions.miniblock_number = miniblocks.number\n WHERE\n miniblocks.number > $1\n ORDER BY\n miniblocks.number ASC,\n transactions.index_in_block ASC\n ", + "describe": { + "columns": [ + { + "ordinal": 0, + "name": "block_hash", + "type_info": "Bytea" + }, + { + "ordinal": 1, + "name": "block_number", + "type_info": "Int8" + }, + { + "ordinal": 2, + "name": "parent_hash?", + "type_info": "Bytea" + }, + { + "ordinal": 3, + "name": "block_timestamp", + "type_info": "Int8" + }, + { + "ordinal": 4, + "name": "base_fee_per_gas", + "type_info": "Numeric" + }, + { + "ordinal": 5, + "name": "block_gas_limit?", + "type_info": "Int8" + }, + { + "ordinal": 6, + "name": "transaction_gas_limit?", + "type_info": "Numeric" + }, + { + "ordinal": 7, + "name": "transaction_refunded_gas?", + "type_info": "Int8" + } + ], + "parameters": { + "Left": [ + "Int8" + ] + }, + "nullable": [ + false, + false, + false, + false, + false, + true, + true, + false + ] + }, + "hash": "e28e052dbd306ba408dd26e01c38176f2f9dbba45761a2117c185912d1e07bdf" +} diff --git a/core/lib/dal/migrations/20240513155728_l1_batches_add_tree_input.down.sql b/core/lib/dal/migrations/20240513155728_l1_batches_add_tree_input.down.sql new file mode 100644 index 000000000000..2e8ace3fadd8 --- /dev/null +++ b/core/lib/dal/migrations/20240513155728_l1_batches_add_tree_input.down.sql @@ -0,0 +1 @@ +ALTER TABLE l1_batches DROP COLUMN IF EXISTS tree_writes; diff --git a/core/lib/dal/migrations/20240513155728_l1_batches_add_tree_input.up.sql b/core/lib/dal/migrations/20240513155728_l1_batches_add_tree_input.up.sql new file mode 100644 index 000000000000..580f539acaa6 --- /dev/null +++ b/core/lib/dal/migrations/20240513155728_l1_batches_add_tree_input.up.sql @@ -0,0 +1 @@ +ALTER TABLE l1_batches ADD COLUMN IF NOT EXISTS tree_writes BYTEA; diff --git a/core/lib/dal/src/blocks_dal.rs b/core/lib/dal/src/blocks_dal.rs index 3e805e92f5f1..28d57ee51dc4 100644 --- a/core/lib/dal/src/blocks_dal.rs +++ b/core/lib/dal/src/blocks_dal.rs @@ -9,7 +9,7 @@ use anyhow::Context as _; use bigdecimal::{BigDecimal, FromPrimitive, ToPrimitive}; use zksync_db_connection::{ connection::Connection, - error::DalResult, + error::{DalResult, SqlxContext}, instrument::{InstrumentExt, Instrumented}, interpolate_query, match_query_as, }; @@ -18,6 +18,7 @@ use zksync_types::{ block::{BlockGasCount, L1BatchHeader, L1BatchTreeData, L2BlockHeader, StorageOracleInfo}, circuit::CircuitStatistic, commitment::{L1BatchCommitmentArtifacts, L1BatchWithMetadata}, + writes::TreeWrite, Address, L1BatchNumber, L2BlockNumber, ProtocolVersionId, H256, U256, }; @@ -164,6 +165,8 @@ impl BlocksDal<'_, '_> { Ok(row.number.map(|num| L1BatchNumber(num as u32))) } + /// Gets a number of the earliest L1 batch that is ready for commitment generation (i.e., doesn't have commitment + /// yet, and has tree data). pub async fn get_next_l1_batch_ready_for_commitment_generation( &mut self, ) -> DalResult> { @@ -190,6 +193,34 @@ impl BlocksDal<'_, '_> { Ok(row.map(|row| L1BatchNumber(row.number as u32))) } + /// Gets a number of the last L1 batch that is ready for commitment generation (i.e., doesn't have commitment + /// yet, and has tree data). + pub async fn get_last_l1_batch_ready_for_commitment_generation( + &mut self, + ) -> DalResult> { + let row = sqlx::query!( + r#" + SELECT + number + FROM + l1_batches + WHERE + hash IS NOT NULL + AND commitment IS NULL + ORDER BY + number DESC + LIMIT + 1 + "# + ) + .instrument("get_last_l1_batch_ready_for_commitment_generation") + .report_latency() + .fetch_optional(self.storage) + .await?; + + Ok(row.map(|row| L1BatchNumber(row.number as u32))) + } + /// Returns the number of the earliest L1 batch with metadata (= state hash) present in the DB, /// or `None` if there are no such L1 batches. pub async fn get_earliest_l1_batch_number_with_metadata( @@ -2175,6 +2206,83 @@ impl BlocksDal<'_, '_> { .await?; Ok(()) } + + pub async fn set_tree_writes( + &mut self, + l1_batch_number: L1BatchNumber, + tree_writes: Vec, + ) -> DalResult<()> { + let instrumentation = + Instrumented::new("set_tree_writes").with_arg("l1_batch_number", &l1_batch_number); + let tree_writes = bincode::serialize(&tree_writes) + .map_err(|err| instrumentation.arg_error("tree_writes", err))?; + + let query = sqlx::query!( + r#" + UPDATE l1_batches + SET + tree_writes = $1 + WHERE + number = $2 + "#, + &tree_writes, + i64::from(l1_batch_number.0), + ); + + instrumentation.with(query).execute(self.storage).await?; + + Ok(()) + } + + pub async fn get_tree_writes( + &mut self, + l1_batch_number: L1BatchNumber, + ) -> DalResult>> { + Ok(sqlx::query!( + r#" + SELECT + tree_writes + FROM + l1_batches + WHERE + number = $1 + "#, + i64::from(l1_batch_number.0), + ) + .try_map(|row| { + row.tree_writes + .map(|data| bincode::deserialize(&data).decode_column("tree_writes")) + .transpose() + }) + .instrument("get_tree_writes") + .with_arg("l1_batch_number", &l1_batch_number) + .fetch_optional(self.storage) + .await? + .flatten()) + } + + pub async fn check_tree_writes_presence( + &mut self, + l1_batch_number: L1BatchNumber, + ) -> DalResult { + Ok(sqlx::query!( + r#" + SELECT + (tree_writes IS NOT NULL) AS "tree_writes_are_present!" + FROM + l1_batches + WHERE + number = $1 + "#, + i64::from(l1_batch_number.0), + ) + .instrument("check_tree_writes_presence") + .with_arg("l1_batch_number", &l1_batch_number) + .fetch_optional(self.storage) + .await? + .map(|row| row.tree_writes_are_present) + .unwrap_or(false)) + } } /// These methods should only be used for tests. diff --git a/core/lib/dal/src/blocks_web3_dal.rs b/core/lib/dal/src/blocks_web3_dal.rs index 3536b40e4102..f7b88f94a673 100644 --- a/core/lib/dal/src/blocks_web3_dal.rs +++ b/core/lib/dal/src/blocks_web3_dal.rs @@ -165,20 +165,26 @@ impl BlocksWeb3Dal<'_, '_> { &mut self, from_block: L2BlockNumber, ) -> DalResult> { - let rows = sqlx::query!( + let blocks_rows: Vec<_> = sqlx::query!( r#" SELECT - miniblocks.hash, - miniblocks.number, + miniblocks.hash AS "block_hash", + miniblocks.number AS "block_number", prev_miniblock.hash AS "parent_hash?", - miniblocks.timestamp + miniblocks.timestamp AS "block_timestamp", + miniblocks.base_fee_per_gas AS "base_fee_per_gas", + miniblocks.gas_limit AS "block_gas_limit?", + transactions.gas_limit AS "transaction_gas_limit?", + transactions.refunded_gas AS "transaction_refunded_gas?" FROM miniblocks LEFT JOIN miniblocks prev_miniblock ON prev_miniblock.number = miniblocks.number - 1 + LEFT JOIN transactions ON transactions.miniblock_number = miniblocks.number WHERE miniblocks.number > $1 ORDER BY - miniblocks.number ASC + miniblocks.number ASC, + transactions.index_in_block ASC "#, i64::from(from_block.0), ) @@ -187,30 +193,50 @@ impl BlocksWeb3Dal<'_, '_> { .fetch_all(self.storage) .await?; - let blocks = rows.into_iter().map(|row| BlockHeader { - hash: Some(H256::from_slice(&row.hash)), - parent_hash: row - .parent_hash - .as_deref() - .map_or_else(H256::zero, H256::from_slice), - uncles_hash: EMPTY_UNCLES_HASH, - author: H160::zero(), - state_root: H256::zero(), - transactions_root: H256::zero(), - receipts_root: H256::zero(), - number: Some(U64::from(row.number)), - gas_used: U256::zero(), - gas_limit: U256::zero(), - base_fee_per_gas: None, - extra_data: Bytes::default(), - // TODO: include logs - logs_bloom: H2048::default(), - timestamp: U256::from(row.timestamp), - difficulty: U256::zero(), - mix_hash: None, - nonce: None, - }); - Ok(blocks.collect()) + let mut headers_map = std::collections::HashMap::new(); + + for row in blocks_rows.iter() { + let entry = headers_map + .entry(row.block_number) + .or_insert_with(|| BlockHeader { + hash: Some(H256::from_slice(&row.block_hash)), + parent_hash: row + .parent_hash + .as_deref() + .map_or_else(H256::zero, H256::from_slice), + uncles_hash: EMPTY_UNCLES_HASH, + author: H160::zero(), + state_root: H256::zero(), + transactions_root: H256::zero(), + receipts_root: H256::zero(), + number: Some(U64::from(row.block_number)), + gas_used: U256::zero(), + gas_limit: (row + .block_gas_limit + .unwrap_or(i64::from(LEGACY_BLOCK_GAS_LIMIT)) + as u64) + .into(), + base_fee_per_gas: Some(bigdecimal_to_u256(row.base_fee_per_gas.clone())), + extra_data: Bytes::default(), + logs_bloom: H2048::default(), + timestamp: U256::from(row.block_timestamp), + difficulty: U256::zero(), + mix_hash: None, + nonce: None, + }); + + if let (Some(gas_limit), Some(refunded_gas)) = ( + row.transaction_gas_limit.clone(), + row.transaction_refunded_gas, + ) { + entry.gas_used += bigdecimal_to_u256(gas_limit) - U256::from(refunded_gas as u64); + } + } + + let mut headers: Vec = headers_map.into_values().collect(); + headers.sort_by_key(|header| header.number); + + Ok(headers) } pub async fn resolve_block_id( diff --git a/core/lib/dal/src/storage_logs_dedup_dal.rs b/core/lib/dal/src/storage_logs_dedup_dal.rs index 2ad4f2a3c71a..159f331a4753 100644 --- a/core/lib/dal/src/storage_logs_dedup_dal.rs +++ b/core/lib/dal/src/storage_logs_dedup_dal.rs @@ -172,28 +172,34 @@ impl StorageLogsDedupDal<'_, '_> { .map(|max| max as u64)) } - /// Returns the maximum enumeration index assigned in a specific L1 batch. - pub async fn max_enumeration_index_for_l1_batch( + /// Returns the max enumeration index by the provided L1 batch number. + pub async fn max_enumeration_index_by_l1_batch( &mut self, l1_batch_number: L1BatchNumber, ) -> DalResult> { - let row = sqlx::query!( + Ok(sqlx::query!( r#" SELECT MAX(INDEX) AS "max?" FROM initial_writes WHERE - l1_batch_number = $1 + l1_batch_number = ( + SELECT + MAX(l1_batch_number) AS "max?" + FROM + initial_writes + WHERE + l1_batch_number <= $1 + ) "#, i64::from(l1_batch_number.0) ) - .instrument("max_enumeration_index_for_l1_batch") - .with_arg("l1_batch_number", &l1_batch_number) + .instrument("max_enumeration_index_by_l1_batch") .fetch_one(self.storage) - .await?; - - Ok(row.max.map(|max| max as u64)) + .await? + .max + .map(|max| max as u64)) } pub async fn initial_writes_for_batch( @@ -326,12 +332,12 @@ mod tests { use crate::{ConnectionPool, CoreDal}; #[tokio::test] - async fn getting_max_enumeration_index_for_batch() { + async fn getting_max_enumeration_index_in_batch() { let pool = ConnectionPool::::test_pool().await; let mut conn = pool.connection().await.unwrap(); let max_index = conn .storage_logs_dedup_dal() - .max_enumeration_index_for_l1_batch(L1BatchNumber(0)) + .max_enumeration_index_by_l1_batch(L1BatchNumber(0)) .await .unwrap(); assert_eq!(max_index, None); @@ -348,7 +354,7 @@ mod tests { let max_index = conn .storage_logs_dedup_dal() - .max_enumeration_index_for_l1_batch(L1BatchNumber(0)) + .max_enumeration_index_by_l1_batch(L1BatchNumber(0)) .await .unwrap(); assert_eq!(max_index, Some(2)); @@ -364,14 +370,14 @@ mod tests { let max_index = conn .storage_logs_dedup_dal() - .max_enumeration_index_for_l1_batch(L1BatchNumber(0)) + .max_enumeration_index_by_l1_batch(L1BatchNumber(0)) .await .unwrap(); assert_eq!(max_index, Some(2)); let max_index = conn .storage_logs_dedup_dal() - .max_enumeration_index_for_l1_batch(L1BatchNumber(1)) + .max_enumeration_index_by_l1_batch(L1BatchNumber(1)) .await .unwrap(); assert_eq!(max_index, Some(4)); diff --git a/core/lib/env_config/src/eth_sender.rs b/core/lib/env_config/src/eth_sender.rs index 397d1ad0f87a..bd48f80609e8 100644 --- a/core/lib/env_config/src/eth_sender.rs +++ b/core/lib/env_config/src/eth_sender.rs @@ -41,9 +41,7 @@ impl FromEnv for GasAdjusterConfig { #[cfg(test)] mod tests { - use zksync_config::configs::eth_sender::{ - ProofLoadingMode, ProofSendingMode, PubdataSendingMode, - }; + use zksync_config::configs::eth_sender::{ProofSendingMode, PubdataSendingMode}; use super::*; use crate::test_utils::{hash, EnvMutex}; @@ -71,7 +69,6 @@ mod tests { proof_sending_mode: ProofSendingMode::SkipEveryProof, l1_batch_min_age_before_execute_seconds: Some(1000), max_acceptable_priority_fee_in_gwei: 100_000_000_000, - proof_loading_mode: ProofLoadingMode::OldProofFromDb, pubdata_sending_mode: PubdataSendingMode::Calldata, }), gas_adjuster: Some(GasAdjusterConfig { @@ -133,7 +130,6 @@ mod tests { ETH_SENDER_SENDER_MAX_ETH_TX_DATA_SIZE="120000" ETH_SENDER_SENDER_L1_BATCH_MIN_AGE_BEFORE_EXECUTE_SECONDS="1000" ETH_SENDER_SENDER_MAX_ACCEPTABLE_PRIORITY_FEE_IN_GWEI="100000000000" - ETH_SENDER_SENDER_PROOF_LOADING_MODE="OldProofFromDb" ETH_SENDER_SENDER_PUBDATA_SENDING_MODE="Calldata" ETH_CLIENT_WEB3_URL="http://127.0.0.1:8545" diff --git a/core/lib/eth_client/src/clients/http/query.rs b/core/lib/eth_client/src/clients/http/query.rs index c395bdd87263..33d9838dc735 100644 --- a/core/lib/eth_client/src/clients/http/query.rs +++ b/core/lib/eth_client/src/clients/http/query.rs @@ -134,6 +134,7 @@ where }; latency.observe(); + // base_fee_per_gas always exists after London fork Ok(block.base_fee_per_gas.unwrap()) } diff --git a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/mod.rs b/core/lib/l1_contract_interface/src/i_executor/commit/kzg/mod.rs index f48f4b361f84..49ab7b93be70 100644 --- a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/mod.rs +++ b/core/lib/l1_contract_interface/src/i_executor/commit/kzg/mod.rs @@ -1,288 +1 @@ -use std::convert::TryInto; - -pub use kzg::KzgSettings; -use kzg::{ - compute_commitment, compute_proof, compute_proof_poly, - zkevm_circuits::{ - boojum::pairing::{ - bls12_381::{Fr, FrRepr, G1Affine}, - ff::{PrimeField, PrimeFieldRepr}, - CurveAffine, - }, - eip_4844::{ - bitreverse, fft, - input::{BLOB_CHUNK_SIZE, ELEMENTS_PER_4844_BLOCK}, - zksync_pubdata_into_ethereum_4844_data, zksync_pubdata_into_monomial_form_poly, - }, - }, -}; -use sha2::Sha256; -use sha3::{Digest, Keccak256}; -use zksync_types::H256; - -use self::trusted_setup::KZG_SETTINGS; - -#[cfg(test)] -mod tests; -mod trusted_setup; - -pub const ZK_SYNC_BYTES_PER_BLOB: usize = BLOB_CHUNK_SIZE * ELEMENTS_PER_4844_BLOCK; -const EIP_4844_BYTES_PER_BLOB: usize = 32 * ELEMENTS_PER_4844_BLOCK; - -/// Packed pubdata commitments. -/// Format: opening point (16 bytes) || claimed value (32 bytes) || commitment (48 bytes) -/// || opening proof (48 bytes)) = 144 bytes -const BYTES_PER_PUBDATA_COMMITMENT: usize = 144; - -const VERSIONED_HASH_VERSION_KZG: u8 = 0x01; - -/// All the info needed for both the network transaction and by our L1 contracts. As part of the network transaction we -/// need to encode the sidecar which contains the: blob, `kzg` commitment, and the blob proof. The transaction payload -/// will utilize the versioned hash. The info needed for `commitBatches` is the `kzg` commitment, opening point, -/// opening value, and opening proof. -#[derive(Debug, PartialEq, Eq, Clone)] -pub struct KzgInfo { - /// 4844 Compatible blob containing pubdata - pub blob: [u8; EIP_4844_BYTES_PER_BLOB], - /// KZG commitment to the blob - pub kzg_commitment: [u8; 48], - /// Point used by the point evaluation precompile - pub opening_point: [u8; 32], - /// Value retrieved by evaluation the `kzg` commitment at the `opening_point` - pub opening_value: [u8; 32], - /// Proof that opening the `kzg` commitment at the opening point yields the opening value - pub opening_proof: [u8; 48], - /// Hash of the `kzg` commitment where the first byte has been substituted for `VERSIONED_HASH_VERSION_KZG` - pub versioned_hash: [u8; 32], - /// Proof that the blob and `kzg` commitment represent the same data. - pub blob_proof: [u8; 48], -} - -/// Given a KZG commitment, calculate the versioned hash. -fn commitment_to_versioned_hash(kzg_commitment: G1Affine) -> [u8; 32] { - let mut versioned_hash = [0u8; 32]; - - let mut versioned_hash_bytes = Sha256::digest(kzg_commitment.into_compressed()); - versioned_hash_bytes[0] = VERSIONED_HASH_VERSION_KZG; - - versioned_hash.copy_from_slice(&versioned_hash_bytes); - versioned_hash -} - -/// Calculate the opening point for a given `linear_hash` and `versioned_hash`. We calculate -/// this point by hashing together the linear hash and versioned hash and only taking the last 16 bytes -fn compute_opening_point(linear_hash: [u8; 32], versioned_hash: [u8; 32]) -> u128 { - let evaluation_point = &Keccak256::digest([linear_hash, versioned_hash].concat())[16..]; - - u128::from_be_bytes(evaluation_point.try_into().expect("should have 16 bytes")) -} - -/// Copies the specified number of bytes from the input into out returning the rest of the data -fn copy_n_bytes_return_rest<'a>(out: &'a mut [u8], input: &'a [u8], n: usize) -> &'a [u8] { - let (bytes, data) = input.split_at(n); - out.copy_from_slice(bytes); - data -} - -impl KzgInfo { - /// Size of `KzgInfo` is equal to size(blob) + size(`kzg_commitment`) + size(bytes32) + size(bytes32) - /// + size(`kzg_proof`) + size(bytes32) + size(`kzg_proof`) - /// Here we use the size of the blob expected for 4844 (4096 elements * 32 bytes per element) and not - /// `BYTES_PER_BLOB_ZK_SYNC` which is (4096 elements * 31 bytes per element) - /// The zksync interpretation of the blob uses 31 byte fields so we can ensure they fit into a field element. - const SERIALIZED_SIZE: usize = EIP_4844_BYTES_PER_BLOB + 48 + 32 + 32 + 48 + 32 + 48; - - /// Returns the bytes necessary for pubdata commitment part of batch commitments when blobs are used. - /// Return format: opening point (16 bytes) || claimed value (32 bytes) || commitment (48 bytes) - /// || opening proof (48 bytes) - pub fn to_pubdata_commitment(&self) -> [u8; BYTES_PER_PUBDATA_COMMITMENT] { - let mut res = [0u8; BYTES_PER_PUBDATA_COMMITMENT]; - // The crypto team/batch commitment expects the opening point to be 16 bytes - let mut truncated_opening_point = [0u8; 16]; - truncated_opening_point.copy_from_slice(&self.opening_point.as_slice()[16..]); - res[0..16].copy_from_slice(&truncated_opening_point); - res[16..48].copy_from_slice(self.opening_value.as_slice()); - res[48..96].copy_from_slice(self.kzg_commitment.as_slice()); - res[96..144].copy_from_slice(self.opening_proof.as_slice()); - res - } - - /// Computes the commitment to the blob needed as part of the batch commitment through the aux output - /// Format is: Keccak(versioned hash || opening point || opening value) - pub fn to_blob_commitment(&self) -> [u8; 32] { - let mut commitment = [0u8; 32]; - let hash = &Keccak256::digest( - [ - &self.versioned_hash, - &self.opening_point[16..], - &self.opening_value, - ] - .concat(), - ); - commitment.copy_from_slice(hash); - commitment - } - - /// Deserializes `Self::SERIALIZED_SIZE` bytes into `KzgInfo` struct - pub fn from_slice(data: &[u8]) -> Self { - assert_eq!(data.len(), Self::SERIALIZED_SIZE); - - let mut blob = [0u8; EIP_4844_BYTES_PER_BLOB]; - let data = copy_n_bytes_return_rest(&mut blob, data, EIP_4844_BYTES_PER_BLOB); - - let mut kzg_commitment = [0u8; 48]; - let data = copy_n_bytes_return_rest(&mut kzg_commitment, data, 48); - - let mut opening_point = [0u8; 32]; - let data = copy_n_bytes_return_rest(&mut opening_point, data, 32); - - let mut opening_value = [0u8; 32]; - let data = copy_n_bytes_return_rest(&mut opening_value, data, 32); - - let mut opening_proof = [0u8; 48]; - let data = copy_n_bytes_return_rest(&mut opening_proof, data, 48); - - let mut versioned_hash = [0u8; 32]; - let data = copy_n_bytes_return_rest(&mut versioned_hash, data, 32); - - let mut blob_proof = [0u8; 48]; - let data = copy_n_bytes_return_rest(&mut blob_proof, data, 48); - - assert_eq!(data.len(), 0); - - Self { - blob, - kzg_commitment, - opening_point, - opening_value, - opening_proof, - versioned_hash, - blob_proof, - } - } - - /// Converts `KzgInfo` struct into a byte array - pub fn to_bytes(&self) -> [u8; Self::SERIALIZED_SIZE] { - let mut res = [0u8; Self::SERIALIZED_SIZE]; - - let mut ptr = 0; - - res[ptr..ptr + EIP_4844_BYTES_PER_BLOB].copy_from_slice(self.blob.as_slice()); - ptr += EIP_4844_BYTES_PER_BLOB; - - res[ptr..ptr + 48].copy_from_slice(self.kzg_commitment.as_slice()); - ptr += 48; - - res[ptr..ptr + 32].copy_from_slice(self.opening_point.as_slice()); - ptr += 32; - - res[ptr..ptr + 32].copy_from_slice(self.opening_value.as_slice()); - ptr += 32; - - res[ptr..ptr + 48].copy_from_slice(self.opening_proof.as_slice()); - ptr += 48; - - res[ptr..ptr + 32].copy_from_slice(self.versioned_hash.as_slice()); - ptr += 32; - - res[ptr..ptr + 48].copy_from_slice(self.blob_proof.as_slice()); - ptr += 48; - - assert_eq!(ptr, Self::SERIALIZED_SIZE); - - res - } - - /// Construct all the KZG info we need for turning a piece of zksync pubdata into a 4844 blob. - /// The information we need is: - /// 1. zksync blob <- `pad_right`(pubdata) - /// 2. linear hash <- hash(zksync blob) - /// 3. 4844 blob <- `zksync_pubdata_into_ethereum_4844_data`(zksync blob) - /// 4. `kzg` polynomial <- `zksync_pubdata_into_monomial_form_poly`(zksync blob) - /// 5. 4844 `kzg` commitment <- `compute_commitment`(4844 blob) - /// 6. versioned hash <- hash(4844 `kzg` commitment) - /// 7. opening point <- keccak(linear hash || versioned hash)`[16..]` - /// 8. opening value, opening proof <- `compute_kzg_proof`(4844) - /// 9. blob proof <- `compute_proof_poly`(blob, 4844 `kzg` commitment) - pub fn new(pubdata: &[u8]) -> Self { - assert!(pubdata.len() <= ZK_SYNC_BYTES_PER_BLOB); - - let mut zksync_blob = [0u8; ZK_SYNC_BYTES_PER_BLOB]; - zksync_blob[0..pubdata.len()].copy_from_slice(pubdata); - - let linear_hash: [u8; 32] = Keccak256::digest(zksync_blob).into(); - - // We need to convert pubdata into poly form and apply `fft/bitreverse` transformations - let mut poly = zksync_pubdata_into_monomial_form_poly(&zksync_blob); - fft(&mut poly); - bitreverse(&mut poly); - - let kzg_commitment = compute_commitment(&KZG_SETTINGS, &poly); - let versioned_hash = commitment_to_versioned_hash(kzg_commitment); - let opening_point = compute_opening_point(linear_hash, versioned_hash); - let opening_point_repr = Fr::from_repr(FrRepr([ - opening_point as u64, - (opening_point >> 64) as u64, - 0u64, - 0u64, - ])) - .expect("should have a valid field element from 16 bytes"); - - let (opening_proof, opening_value) = - compute_proof(&KZG_SETTINGS, &poly, &opening_point_repr); - - let blob_proof = compute_proof_poly(&KZG_SETTINGS, &poly, &kzg_commitment); - - let blob_bytes = zksync_pubdata_into_ethereum_4844_data(&zksync_blob); - let mut blob = [0u8; EIP_4844_BYTES_PER_BLOB]; - blob.copy_from_slice(&blob_bytes); - - let mut commitment = [0u8; 48]; - commitment.copy_from_slice(kzg_commitment.into_compressed().as_ref()); - - let mut challenge_point = [0u8; 32]; - challenge_point[16..].copy_from_slice(&opening_point.to_be_bytes()); - - let mut challenge_value = [0u8; 32]; - opening_value - .into_repr() - .write_be(&mut challenge_value[..]) - .unwrap(); - - let mut challenge_proof = [0u8; 48]; - challenge_proof.copy_from_slice(opening_proof.into_compressed().as_ref()); - - let mut commitment_proof = [0u8; 48]; - commitment_proof.copy_from_slice(blob_proof.into_compressed().as_ref()); - - Self { - blob, - kzg_commitment: commitment, - opening_point: challenge_point, - opening_value: challenge_value, - opening_proof: challenge_proof, - versioned_hash, - blob_proof: commitment_proof, - } - } -} - -pub fn pubdata_to_blob_commitments(num_blobs: usize, pubdata_input: &[u8]) -> Vec { - assert!( - pubdata_input.len() <= num_blobs * ZK_SYNC_BYTES_PER_BLOB, - "Pubdata length exceeds size of blobs" - ); - - let mut blob_commitments = pubdata_input - .chunks(ZK_SYNC_BYTES_PER_BLOB) - .map(|blob| { - let kzg_info = KzgInfo::new(blob); - H256(kzg_info.to_blob_commitment()) - }) - .collect::>(); - - // Depending on the length of `pubdata_input`, we will sending the ceiling of - // `pubdata_input / ZK_SYNC_BYTES_PER_BLOB (126976)` blobs. The rest of the blob commitments will be 32 zero bytes. - blob_commitments.resize(num_blobs, H256::zero()); - blob_commitments -} +pub use kzg::{pubdata_to_blob_commitments, KzgInfo, ZK_SYNC_BYTES_PER_BLOB}; diff --git a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/tests/kzg_test_0.json b/core/lib/l1_contract_interface/src/i_executor/commit/kzg/tests/kzg_test_0.json deleted file mode 100644 index e348c5c1391a..000000000000 --- a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/tests/kzg_test_0.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "pubdata": "00000005000100550000000000000000000000000000000000008008000000000000000000000000000000000000000000000000000000000000800a42efd743e554bf21985b562e6b1f3ee3a2ed72f4dd98d76a68a1146cc9a21a86000102290000000000000000000000000000000000008008000000000000000000000000000000000000000000000000000000000000800acda5158c3dbcac23211672d2cc9c3359fc98a11a515a328ff84730ce388e0b29000103170000000000000000000000000000000000008008000000000000000000000000000000000000000000000000000000000000800a034d6369f58fa6f6577da618b1402ae4035b815c468a5e5e120833f35064d7d60001035b0000000000000000000000000000000000008008000000000000000000000000000000000000000000000000000000000000800aab90d8e21f8f6be792eaceaeaf2a7e262b8cd8b1a950cef7856d3c00d3e0fe240001036c000000000000000000000000000000000000800800000000000000000000000011f943b2c77b743ab90f4a0ae7d5a4e7fca3e102e0c3af29edc7fd00e8d0a61d41ed085b5999bd76d0ec57c8b794117edb48006600000005000000386c0960f9e67050e6b7ae575837febf738d9dd205bdd96129000000000000000000000000000000000000000000000000001da2dea1775200000000386c0960f9a255d58fcc01e78bd7f30db1c5d736a19d6de5d5000000000000000000000000000000000000000000000000018df33ae339d000000000386c0960f94f1b62090fc5498593cf8803898feecd5109b2d1000000000000000000000000000000000000000000000000001d9dedfce0a600000000386c0960f9988bf802a450b1a23b01a16361041b8cc4c0096c000000000000000000000000000000000000000000000000001bdd15e065b0000000004c11a2ccc170716ffaffcb4579f8226130d4a8625904162263a0b86991c6218b36c1d19d4a2e9eb0ce3606eb4800000000000000000000000000000000000000000000000000000000138ce2000000000001008d34040102f2e6ed1059ff0ad70d4e11336e3e5ad30e827e008cc2173fcbd003105e10deef0001000117285a242b48c751232df502f6448f1dbf4a1d9132f1cccfce399608c8b90b3a0a1f96e79f02ef0c7d92a6ba1968507b07a357809bf9eaf6ccf0fda94f00010013eff990900684eac2c81359f38c1b0a834d011273ae30310f9c7807464a1db77812daebf491cddd7c2c4af8b712a218f35b8f3ceeba86aad8ee402526c200010000939c0f641c18abe55b3a876a26f1720e4903c3f12e25b8b6214942db9069af66d7b664990e3f84e4633afb9c21a8da0ef39901d9bc286fad7e19d3e6690901c02aa7bf71858e06cbaedda3277e925e055d21c73be1309a36203fe4727785b409018a07091e19392f970a79e422657f4765f3c6498b0e174b862d050c05dcaf5bc50901ffa6d75a531a1505992e4c1ff4ae286f3c5336d1464bc9b25d9178749c5a0fc70901a9729e5ec3b68a9346d3dffbbb2027b1b0f40417a4e59e60fad413467c1e90a40901b4e5bfca3d03904333c9db34f701ee88c866ecab38b47bb219ecac0861ec961b0901adc5ed37ef2f96fcf6b4c8ae77d5dc254da413dcce18fbabba941fbbaeeb46c809024dfdf044fd553fd9ba2ef4251f008ee4cf873fe70182d1b2f75f28c91d282c06a1996a719fbc67f35a4344f73890c1172eb194a88cfb6b291ade6a03a87fceb351b6ccd7a8ba72fd46817f800389c924d23232fcc6a118cd1fc2a93f3a384b0535afd162716154269523f7345e0bc56795149c3dbd12ad80b8d16c271a08ce2873c63cd21f8b28275563390881110a3ecc00415f285ab6f8e3d9ba138242a46510c44a0c83a350067a6ee880f6fdcfb01ee4391e32b4789740000b73f38daa02eca7cdd85642f68cd8cf51fa015d796fc2dbcd1aebabf654bea441016b288c64f0c000bab081a7b6f92a4ec1f02eb005520b7673af201c4ebe79e761538585717fafbb39b59ec022d606f068ebf6717faab388c9c53c18fabe7838c5608820c5d2430989d90b92dd26451e4102c68af0bb1400006f4753c45e67f5311ae7255bf8889f7fe7e58b6af06d9806e5194a1c5c66d05139054ec8fb4b6000bd906f224650fc26515436b3c25a87d68addb31c6a5373f7e00f754d39119486391e3e8744f910003c67eee680524913ae37ae69f0baa481d0e4e6d3484a307479eb229bc30f479e391be3888d7ca00075aa7f75697d3feb547969a0d55f8731fead1e14041729ebd1d392848e7cb2af3914ae1318ecd9020f2f4ddce402fdc9ee7b3c86ecd111dbb325d01422bf470db82108a3abd1e3b34101192d1069581b941d50c402518ad6fdbed21958ed6c5724923454fdd2495cde5abc4bf4516ef98c3923b46bf7fe2000f6d7022ebe58dc19e751106f2cb98fd24bd6e14d8073a53d3f8e11b45c5df48b00a99e289160656884525d83a63124c30e63f37032d635435d92d49995a8f8b408dd140b448cfff621010bc3dd28788ddd82f2bfc323f8b837afc1ac6c5c1580b2091462157e0771f69f064d0d7bf52f7ebb7e91c66e35d4914a9dba6c7e3cd75ce6200914183241104a8b011521fc34fb04dc72026ec296cd803af9bb39e7dd5bf873ba5d191a524bc07d6ca5418464758476f726a9d98e7a81bdf17ac4ec7badc550129a1151b03c194c4b4034b276b9d891a6f4775fe20e1e5cc98babd8744c81e7daa3000b49c7e5c28abda1a47fd53ccec8fe0ec67794aea9e3cd392a49b88e01e96050ca4793360ad636d62a6d89ed1dd50d100393a25b769f51148d33e9c80901df92d744c1917c79c8c38fa95bb485b984c521a4253c1ac6c12035ee61226f5c417bf8114d29f36913853abff5b4c974f781edf65caff6e8293912904b86c909f9559cbfd685c0ba6e491dfafba0f1bd4300000231b5c9c8cae149bd749c383d5bebd9ee775377beb035eb84377eff29f5edac291b6103f419ce028a771a813617dc177b50fd2f2d4ea489a7f9c2b6dfbd7cb0608ff6bb949911d83cc543063aef4cc63829a471bd77f68576941d08cfd681e84f80d3e1645920171f09019ad2dc29fffa2ee724cd419095cff0605b7df10d65420c3f20d7522ab9fb81be2165a6de4ee706aedeba038f6e38ad43723329856ac986583787b60bc1f54a7f570a91b33269294c912982def982794434499fa4ce83d8a32f8629d2c1abf127af5e85f34909aaeb23a7a5450a60ce1493c3bc690643f16943df5367f77aa54a54bc8a8200c68674c041892fd6f77453341b524740e2f87d0a3df7f0ed575478a6b9041b04f0d563c8544007d64a89d51131d5adef571aa36d21bb0f9917c463b5c9906d07f248619a83b48509c798407bc4da757ab7035dcc0901a0703adc7ccba7cd1054340b8b59b73e6b8b455833b8cf427b7beee4a0f02c09a12a9096df9ef41ce9cac50cfaacf1151071f975ad45cc7125fba34a20bffab916ff36625d666f049583117e53406ad65493808667a1d3d91634cf4c04ad1b76ce2c06f7385a897f54d3bb11626357265a0e408c00504dcf29352cc80c19ea774e4a6f1cbe01933d6dbe0901567e3d3f853b2bf658b34a086e56c374fb85eb4c313829401d431924d7465d4aa180115c708e12edd42e504c1cd52aea96c547c05cbe9a44851e30b39b56297fdf090f6bf257d6c5bab3479854a280ba419fb6d5daa180115c708e12edd42e504c1cd52aea96c547c05c76a06ae72e8a8f120cf6b43317b34dba4b119b37e8d01ca2b28c514b97d2887e090171503ac05f8ca52d7cdb786cae95ae5062c8a97dac126ed9272dc7e30eb806c80901fd2e18833a48b67bbaaa3c77c79cbccee2f71b4ce0a92b5e2423c707cce4533aa180115c708e12edd42e504c1cd52aea96c547c05c8bc265e1dc0dcb7c9ecc7cbf5f7c1843c5da43bf586c40fa9560774c5eb1ff080901e8bb18c16a016ed06541ccb16b38473c9357af9322a2ac573175b7b1f9ec07cf090107ca4196285a78fd294c7947fbd5febe8b6ae006c9d658e31b60017ff3f97ae9090166df2a30dddf70b6f44a42de4833a3680b9c5ca958accae9dbf7f0b1600947f40901281c95c77398a8b558c513bea568f8be9be22fa8f1a6d5645573de661f2ee6a0a1c2fd44a8497e2fac310f4fc8418908125546f32efa28383624da04595bf930920bb58bc4b15b5de94ac42056027c1daba69332e0313f2a01718652997cd9436f8f814343cb15afa1919b850b03c4a0d28149ec068207f3c551c6c2090136d8384f8f88b18de18e3ae63b1057cf0772abe2067b98b68446b10ff458fa2e89010000000000000000000000000000000041e13027c1b52df1bd9803cf1545bf4ab94a16d168d9104f09d8766103d550bb0a01bf35d33c89f5e664ae2b92a66a939eb013ac1de6515a57ffea70d56fcf6b706619d057eab78b3a25d00ef7f39c4967eb220327371363098ab55ae8ac283be6cace684d200a01966f1152c6a8a2f9a663c24ed9440e728138744090cde391c63e8e1786df53102106db12ccc6511c9c36ac5967ccae5da01cd21ee0445c43211dd1198302ae036d5e3b2cc60a013abc0472eb8a8849953715f91ecd85ec47ef839e466dc143cd2d32c6b61349410a01a0dc9b0d01fe770f0cbed8572f05d38c58e5551d15c16d5608fc302b7ad49e0a2102a5692aad0f28864b4e3691a85974872a6ce40e6eab98b56b6d3b8236786a0da0292531210f3753091014a4ce05520dbaf7a2654d893c76a471863e356ea366e825fd296753bad41b1962c6ceeb0388076118d3a1e6e25ee9084ef2e74cfaecf782cf4003319c0f335f96911d19b0978b507111c55737118180a4607b2e9f115ff1c10ceb656a32a6f4cfbb21c8c6e0472101726199c58545f9a610de3b4061e87c21f85dd19449ca4aaca2dd868172c540728497bf311d9e0171863e7f0ea5f165f28a75168a2e392f1c22732864565c491a136cfa70189eba027acea12fbef53b81aad68c4727f3329869ee704609ee749f7a7407fb1ba20d656da3c5ff74892871546e61dda8da470c0b99caf78854120901688a021fdf7d14582bf5bd2eb5c93cf86f21171c52b86ca334776c7be06633e9a14aa5cc5f01876266b9a0abca2a77fb4af9425cc3c0f84c80c45a8236856be149391bf75bc23441e7e25070407a08efd97811e33da165d8e9461f9018596fe0ab94fa050ffb3bca242fef613fb876ab1c59ee2b84cb39b85d0df611e9380de6df4f5da1f8f02a925c8d490c944fd8e64880228050aac29f405aaa380863f232d74b43ef4408dd111745f3a9d46bda7ec362ed97a144ef12d4b87c5646869b28e6b62e8aca975afdfa753b08a8b2b9aaf55d8853d07c452777fc4c529029633816d926fdd31689a2ae29680171861aef4adc1342aa75eca6cd65fbec28e32540eeb9cc63c12a4e7fd74288094cc1e10901e9aa2c34ba3c83030e0d8f90d17ac623eb912fbf8cfbee93413bbf7329d959b00a013fec76ab6733aeb76b088570a0023b4fdcb8d0c828e9dcc8f81bb0e75645e6d419f2662152b7f4b7dfc66f12b9afe0f70362b0a6c5b42e041c3e4e9bed9df52f3dacadd60a012f93b53ae541da4ffad8c2e72ea9d77c1860404254cd8cdf8a5bff8708bf99de0a01e63866cab2b4c30eecb868b0dc9f50372b1f79ad7c93e37fb5a0ff411f5cf0c50a0123783ae730d29ee74f13dadbe6201b3063acccbadb0b768d5c80e8ce06a50d7409017282df9bdedfab9ea9fae6c127d63bb89e4c5e2bc6243b231ccdd2a835b1a8ac19a7e215f14e64c035ed9a1f45aa7f9decad368ac1292a3835f448e952b348ed2324b84d21027e577b5763f3ac147822e73df0eb878313e7e55df3db4405eb1db3746083c12f19b9d10a015f06be9c9642a6321a48bd6eb51fb69e85767218915fb4ef3e1ad12115ef907419121c1b961e6a90ac18d287abbe6645e314566d6068483ada0ecc53c1bad6918d130c5009018beba255936bf7debacdf5ae874cf4b74789861b8386733eff99d087b394c6a20a01df4b1208f951cd68e9169a4820678631882ce39999e2b20d20bc94b0f6273ae60a01dd845bf2c9bc9886ac2286d0dd8c08b0e7b227570809e3229895e49f65a298064902bbc099d5fb6d81289a8e412e9c64e1d0237b95f8899e9314631b7abf0fad93881ba7e447abfaadfc0920cc1d27e9ee7f5678cdad1e1140c4d852ca0c64f04346089c0809b48771cc695f090876af4949a5dda017af2fde7eb64989734c48bd449a895b7d22c7e20f4e85b27b0901d1b4ad97ce6833314f1374ed5de8210ccec6ac500552c49b35c486056d5c52580a016a975864c94755cc754e06fc62e04462f62979fbc2df237884199fdef76b8cf7096e352fcea326988173e6d09589571f5522c74e25834abe3e70b4ddddef84c6a603096e5ac400e164e56f8210320bc96798e6e0b5fec0470bca0041c760d46057277ed7210134d6f4e47a623cc0243610a628b008fe34865b98e100fe9c64dbb6aab3df82edfae169210134d6f442dc4493dd70032707de9f746062785a10d98a1fe54541e04a2c3fba6a25770f3904325b0881a86b033ed80ea5f81caf6ea810c94a038363a914e2bfd4bf89397f140896043438f549344692099866068fc7efbc7ca398ad874cfc7be2cfbcf493af177a976e278b076487fa4c23906040af191a524bd08b413cba0ad1f7ae3cdf10fa1a5a78fb0f763e8b958af889d1cc65742a81956906f17c2a1f17621892641e82adf64a6735ab0d469b6222445a27a0509b7334974617b086da6556919cb32bc86d310150969bfade2922350e35eea1b624bc0860fc6d1a88ef8949f6d193932a74f1cdadc3b17ded0901b91f047425ec49fcd708d173677a6a5c4294172167d6b9d3082d475119697f680901de1770af232be79daffa32d59cb27e047f3230375cbb6c6f359115ad2113d99c090101a485d8779c9a044e5957d15aa43aa8206a29204e97e578f6235eb52e23afc23103600171864a9d2c5b32b59cec41409eaccb473a7c1b6816ed36a719b35fa372f988da2d20d4a138f784e67869bbf5aaad5d664dcd05d7a47a9885d224de36f00d582d2a3defab0029281b54c6b681af3517a007ce32b98b6b01a7517f6a4e06b86e198796198f82561983b0e61ebc8b86a629f04ad9d09e9a785c984231c98d0eb041c971d9591415c9488a1690396681007c669c526d4b341486bedb9c3b8635c26b27602f7f0df7a6a0d653a140caac8c39571a923ca04ee1965836020f5cdc7d95a8ec1546ff2b8c7bec15af2af26384f5c96cfe387c870d310c40f9ef7267cee79a8cd93df1aa7d9f40c4e4ae2b7d1d4740f1e5e1ba1e977201cc6af3b0810a01b7b4e11ddfeeef52ca0c336858792be12d31db6c53b00d24a6776f4f08d08b7d392386f26fc100007ef3af95e82367c7180371ab903da968240e8720165f2d6d113d4e10f8d00279413ae5b783aac200009712281cec3fe27aec63dccb979a34e7d86e37e3b3da429e23697846a2ca958e0a011d3984ca7ed5c262e1067856cb7191d87bb7d1e8acf0230d4a600d07ca22bad2418ac7230489e80000c454190d333fcbedadfa7051049955993572cbd63cf42e68aa77b717d073c8a40901fb77b81a76104598d9ffda1152005f97e9820700085c91c332280307da83eae900810972423f7d915b94ae6a76e50f4327fb1e8b9d0e80c318ab05fd0819e639b4fd05e860183b1015058ae32bcb7719855ec7aa4d7200910ee810344e3cf034f9315af3107a4000f006789c75571d41a9aacb4465f684309aa3d0597b2c75366cde06e60bbbedf969012c000002ee00000000000000b9ef0eb0e2d93d6d27b065c20e022506f10a469215692dc4fab8347ccf84682b0061706500000000000000000000000000000000000000000000000000000000063e47078faf6c734bfe7e188b586ade665b4e478038a57f99fccd918751aa2b69006170650000000000000000000000000000000000000000000000000000000006dcbe7fcd0d4a4b51538f54650f1686c598455da5d3d4caf3857d4c6c8c8cce4609011235fc129aa7c102d35b04dc78a7d91183ec8dcebe12c0e5364660bff015328519033ba9f6c2a13297570dce587d6c1fc75835bf680950683a2c355761d8a473af1f2a9f090101acc07e9cf7f84857070a7a1f6320eb97b3e5415aa19cb3dcb262893651b0971903c55efe2aa6546051344415d9136c36d7224b5de88d5f50081448ac9b71fdb50ac10da1149173b9c112466d7bba1f647e2cdcdd6ff184e4275d59e3ff3f1f2321277c45d8ff9326dfa34100857af52b1e5c91a8768b04538139d4793ad35846718f9892a36e8b925b2b8e01b97a36abb0deeeb6503f11c3be5753c9d069dd7fd681798028313bdf09a144e8d30c05e5a2219d5f7e17e7b87e60f6c1353e9a3cd0481a52950a63256e2e403fd76b17c7229e295ecdfc3db33f44b869b3ada16dd28c2c5b91dd63b4d4e78ecac7139878371768686724262c779b02e44e6909c79dda2440adfd1f4e3da630f1b18562762c34594175b37816b749b66b0ca5180a81cdded691fa9475dd73710a0f98169e0077e37dd5bae3a8316c9c220a01fd9ffe9dfd6d0978af66e263e8dc7277c8e21b6a802c63a315f9922d8d43e3a6a10d5328bdfba313310c9d224c25772b6fa751b9d40213977d8228758a54d817457dad5de4b806b939a352c83bbde7f0a36e32e5243107b9017186586ff4762b92786831ba36ed9120db1a885d01d0d763980956b200644c7d26e1230901bce25afb1bddc81d393a7d30c196e5cf3753a0e876d4184457a9806266d3e436290a9d9896c78e9c6203892a9d5862e8511d457c70c7f63f5ba7637eedbb0f4ab76ed8ce64bf290f98114c0f326b8c018ec3ba88d28b7e146a1e3631b33af636c11a656bbe24dc43cbfe2cc429e84fb60e3bff1427493f2900dd9f0aed1df7c1510fa629ad473d2e32bddcbbfafa717a889311ef465c121784579443da935b76be97d7d06fd561934765db9344fea3ec4409d0b31b390d22f86191843d6e37d777cc8e8a5edec34e69d8cd7512f38c4cd1e90d4aab6a46c3433fb68d4b392c8850ad006bf0b94d78ea6081d51ce7b39f9aa65e254f9adb05243550fc4c377edc24663b6d78e10342a9706aaac244d001368b000000000000000000006a89e8b51815f5e8fd9e659eec56c8366856d20501bf14e46707331fc77ff92516e12cb25b0c3a0bf18a9d2882caa41fdfe4eb2119a2459da249b6bc6e3f23d4e12d1ba35abbfb0e961961328a4b2101c9c38047d89ddae6658dfc2fa241515f7b12b8ca79c357bdf4ec4aa0c2c2d6721ffcaf41107d4ec3af4257850cb66c12d0f834afc9b5f4ac808e7d80773c7d420dcdaee1fdd0f2d62e55a40209011b30b332b3d0c14d4041d58ea84b247009dcea5973fc8a3cacc21f7aa99dbcbaa1037119632437119bcd9a6b18ed9ea9ebccfbab7287cbb29a78f07e708e996654f2294c3da7277f0fadd03990d835b3838becdf9f4901f9e8e7ffc272bce3672a4b4cb989f3df5fde6f1604adf28cce47f315c72cf18f697884170e5fd57ae1034712a12495b627ab792d400000000000000000000000000012926a5b6731e66599d907c4913385770f91b103e1b072e98d12483a688f4b19309140090185ba802574057d16e01335eedc59ede451c146aa72c0ae54e419d23a671e5225a15039543c3b1e342da8ba16ea9c7dba464f74edf637dd4224f9cd1eedf6c115dfae7363382984a8772162a8a210da403f22c009203136d80171863947f55b48c65591d2d2fb62ee6858c2ada93941ed7895f9841e9851201a321993090a369ddb839ca0979e7c8c6d7cac3ebda8de4bd7055bbb218b511620d4d17bd55a09090eb9872b456aeeadd7a743ad65a50fe854d7b558990d88caaa2a8ab09b07049d09012afb31d0ff976e745a1ba27500820d92d6fca5b13db5ff96b13f51905046a4d6a1a47fd53ccec8fe0ec67794aea9e3cd392a49b88e7d4ebc6b7d65c7d59842ad93ee1a54e63c5a0df0e01a692bb464e57f6bdbffb30901ee19685744894fdc7f1f2f607e43fa01753c2204690dee07620a2900bffa7fc3312ff8017186165abc764a218ac3636196d9cf8d989974df3f0aa8b4fc13a2252810e20caa0d35312ff9017186226275857c0de7cc3bf2c0006472738b883e0829a6e5637f90a80ba4ddcfa0973f09016248f1313c0bcd995cb8c908eca722a4cdcd3c5f48a6c6db02fe2cb03a3a81e9a165477af971c552412773320c4c567228176872b3328334eab1e5acaedcc39e3f8e5dc1d57e1d9822c549b057678f25cce7fa5fefa17163d25153fe1b2f6f994e0f5fb2d47a84df264f26e63210489eeb2f2a274f2a82f8c61aaac5dd3ba8244d1b90fd32f45fee5310090151810e421c5111fda876cece5c9056080dd972a726a1e4428ab391a8618337020a0181399ee467cf03c8cd376963f8607f6e276a92dd89b19201f2137b4179f5bfaf210171861360ba191ef16d883d699184a191dbf55c7408c1b2f7bcfa70bb1a0a9d4c2de2094101b9eafba6c4c718db591fe422c98b3d9eaab1212c4217d10b7bfa96619c2a30c94f02c3931346ad190736b0fab0139776905c15dbdeeba04817991ad158e3cb31861b82a709ff11224f2415190738f5088b1bc394fb62a4a22daf0dfc14900023e60ec6d4cd59dc14959dd0a57f0d9e1907383578bbe26d685cdd03317abea7e600424a52b0593c10dea4270f7bc57f80bb8825097be8fbcb15cacd4bad0b7e6ab7091480ff8a8e566b9f33f13a09377d12e50a69c3190739b50ff653f4135296911a4f49f84aca85a6c002906ef30b7ee66193a515d32bbf51190765a2fa57d4d6e2f5ae8d1802033116982745fd5e1c51a08b0fe48b1277fac114a4d81909b5bcfe5ac6a61e70b02e52f8be996f9e3e7909aaa27a95efdfa5b528c8c433cf5dba1909130cb5d61b3701314843fc1f3cf26e69dfe568683bf5300908bdaa41da1deba8a5e9111f2dc5150b02810c8f02faceef1a14eaeb104c0864a565534f17a5152ef9eb82dd72116c06808f9ca701e7b88decec88ea71710cde44a2b0496fb2ca008a17da43138b1a3911054f269485e4113f0bc54e4ed3962928dea4ed30dc6e6acb1d28f318a696286a31610a01e98bcf7c91021dec06def30d4da8391f52e36ee64adfabfaebb0f4e31a7d01f7111f24432c1c9a293bb476887074caee2e22a5bc98b2858d6b9412dc667d6d0df38acea1d948da7ba45ba8d119a64e5498b247feaebd8b739b8f0070e47f6cb2d361a6ae2e89f0aaf5de3a5b29947bc280675ceb680c6ec2311af0017186401cd861d51220a7f456584ea2fb9f6001d2ea4c0aff83f825deb6c6de173afeed3904325b0881a86bac268ec7a39f04bd59bfeebad1668b7be705b8e278da30a00a393b6eb780edda79c097ce7bc90715b34b9f10000000001e61955552a75fd44eb3bc8a520617e7626f867c9784b0934d33df13fb0248fb79c097ce7bc90715b34b9f10000000009175577d243744580e4d3eba6d001ae4a52ecf41915633aaefaec812c0b255bc0a016da4f25e21ea9a9da707266a06a38385e4f45a7521d930bf7ac684d28dfc860e090131a49c135824a04a04360f4ecbc6867c746f5ccc530ccb0102622dd71f56c56109013e4006101e0fde5109283e76e0abb0a6496f75a63c979ef7bfe90c55a20c8c8831047cae0f1089b2c416fc546ac82f94a49374c48a6ed2011ad43b0c71cecd92f8990edb9a2db4090175505f6b9b08556a1e16c1c50dbb8c161c3abac8f3f50613e08d71037e0ec6d74170f5f9d3e35fef0d2fba6acb46454534d4fcf1d2027bd50dc1ff77ea016718c670d3def0da26a2a44170f5f9d3e35fef0d0ebfaff93008adcb69843ec9881d6b2e2bcccc3aa4b30bee9ade9a0221fcddc60a011cacced5b6f672ebeeb7e99dd69158424e69b24fa9075218dc525b51fa4c99d029d1adada9f99f1e7b9405b6770849049e59e363dedfa1a73f467405a64edd954f4f0709e6880901d6bc4f049abf9c4b0c663c1f137c302ded9772d4f923840246a2b5c0db47177e0901257acae0efe704629e1552412a67ef32106bf2191fb3dfef4bfeb844d40cc503090117391cd68366e6f2eefeec472aea267d5ed97727327927739f800b924f5d0f5d09016669faf6484b99ccf6b8de37b06ba36ced8277d91d45ddb18926ceb91bad8c700901ad8ec34ec3708e06a44070e9f8f0697fd925fc70d8115fd03b2767c5b02562260901940fea0027879592e27dbd46f7ce4c6c799e2f04415645566c340c0d318ffdbf09013b1d910ca21f4561e426cd233ad98c9a2925154a77cebf3bc5d6847a3d91075b090167b6c29087dac56105580e5f677e349bb179da137ece1d2430fd75ebdc1b22d509010b32db0729a5182cb3ad4f86f2328502c7db024a3656a057b33153f096e6b74f09014835b724a8d30ad18af48320980d1ff38ddc9b3fbdee2a6ac7f9211d9b05631f0901b10ef22029d3990a5edc1775a038412559a0b8194d37476656809d5c84f387eb09016278d477fe67124d3d3aeb8c2f5d726e6cd285bd543a5f827d33c759335b55100901a70ebe30bcfec080f540e96cd54c2b49c7aa18bf7b771b6168582113804bae300901dee3974538ddc7a0866fe26c52a389cc4c96767e5c09d3917ea04bb3395f7c2109014a4d16024ab61163e4442862311bd0fe61b720678d02e03e095437404b4b03e75979861e6a2dad8a248b8c0ea6150af75e4e48b97c1882bcea43aca1b01662ba932e546fe7d5ade1b949871e090190869960b5bd7dcfcf9447b302cd99aee59d882710ef2a0a1299920f7f7c685d09015d6c16992385cb5d833df31f4090ee6825b2c8701f92ce0477b63ce12b75b862005261657374726f0000000000000000000000000000000000000000000000000e46f3883d5c3a2477bda1bbfb8924fc5ecd14869c2d2fc38500a2faba696c14c000a8f6616edf999f5ce01f1fe1b894a726336cb13115a59043494944a072f579a4d1e7217d66d3d43fab8faba493feb42965e80cad570bb29988aab46c3b6153cb00c89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc6f47586930205713a6a363a9d06c15d0db51436ec49fd51b8de8900ae32c2b96c096f0d5c261ad9aff66480c7b22a49e9c185589b67131e8e1099d86a3536086637cfa17816a972b63d0679c935879cfaeb6453c7c1c8c9e853ece842d12de2a0d1b7dfde9c9617fba97871e6a3cdc0a9f60f66e0c6c0f1a17816a972b63d0679c935879cfaeb6453c7c1c8c902e961595aeb5c304f2849d4c1a759413a26d8c473fe769319a18e319ace9d3a09026e708a3dc48ed3b38f705bae99f8fef0f0c4881cc5e96537f32f6d703106b50c09016fe4a5488c3854e373bdba93eb4552bfb427e6951995bd65b2c3a55b156d23b6090292e278a79d3343959e9f5c390277e32d556e615e476cf0a108f9b4524688ac47a156abb6a3f25dccdada106191053b1cc54c196deec8507d19a898613091d6415824a6244a7ba6212ee76a9d0a7c6c904a82f3e01b09015aaf9b799076012479363653434fa93b89f2822dd6150e4b56377b207f26fbbc0901f826346b5d717de99d1774a11e6b897315c60845807efda369ac20905b3f64e009012ed7a51071d0844752958bd1d67971666340bab5f8fd15c2d471a0fde4506147a17816a972b63d0679c935879cfaeb6453c7c1c8c9677f8ff1a98b7c11868d0408a7f9ba3d523562b858d34affa57ace4e74caf269a17816a972b63d0679c935879cfaeb6453c7c1c8c9d7219df5808c4bc0854d28b38fb26bbded52e00a9ba02dd85ef8285e6d0c6971a17816a972b63d0679c935879cfaeb6453c7c1c8c91749e7427159f18bb6e298b10b3679b956f41187a25bbcdc4b9b0daee8e3444c00697066733a2f2f516d556a6554533668747756766b51766d6463556372786165f0682fef72a02e4f48fea058e5bd83e1c82630d0db2348003a7514d343b80a61006f665178793452326e636d566f37476f436a666d482f30000000000000000000b589cd787e4d0d1d47e6b6dc38a21581d7a2d99c0a2a61a4db2b0b836149ff04090123f933b221f2747e4f696debdecc366a1de91271d188a260a8215ef3092186cd0901460aedc9249b63a4a2971178bafff352020838fa1fb066b099ddef723a32d3e509012cfec83d80457e553b6c58f546dd5015923eb928e9e020bc002703a6c47129b50901ae2f0a3b41bb84ac766d32c6615da315c2323727e715c6ca7eaf356d3da269c109013295ce215f546349bf345f5f7f8406a76b4a37c24167ba851c5cf449de84b6a03904a4a455de4348ba4bd574f6eaa9f4908c97ddd69ae4939bbbcc78db3a6d2589821800363d06ba21042aa1e8dac60805d1eb0ec310bbb4aa02e9c61feaf4a8879215462a9719f06e271c8d04210182219e1d25a204efca6d46244cce7ad7d2ef46033c6ba0583cf38098dbec0b21040fdd2165a6de4dc6ebcd94d5b43f27e0ec3962ac378a92c79cab9df11395fa083c1d3277c3be9c5922c2c22b62ce831b53bc81395387a23aea94411ae8d026836e206a43364996812a9d53eba32a5b22729ca729227610346e5dbad9b528bf307acda1e21df661b94eef1a5405710aadaf463ecfc73ee1496e0901f10404afee7818eb073f32117a11d500a20c73c046a0e27b52f453bf11beaa46a1a47fd53ccec8fe0ec67794aea9e3cd392a49b88e2109690fd4953c3057fb56444aeba5a904b21a5215cf37c4af87636f187b2b38a178bfec5e4693aee12dedff44f40e5ccce66ddb1c7b9fca3d919f48037bd36a73eee0c506d169bc80a4875fac3ffa2e9d38c491db0901cbac5c7603ae0a7ff05beac2bc7946663fcf627d46749c9a61436fae633ac0520901df0ee468c7b9b9c10d8b28697256da3a2a13433ca96e9d30f22ad3495553b1d2a144ef12d4b87c5646869b28e6b62e8aca975afdfaa29734b9b4680e31e370be9fc36c9b54e6523c48935f7740c3108bc03319ecad296c01718640d645e474f86fe66f2c63f0857e0aff27fb4ce12f94d4fcf320ef56a21720d163090190b19c1f41dddb2f54d53ad0d8f961ec9a866c37b50141da2932ef7f3eedf69f59524d10dbc0a39e7cea39c8626163e80cc225e64719f6c5b9404107acd41f117f495f1157516b47e17a4b8b2165a6de2b483b68e50ca45c31cb92e1d43b79be414335d8e3ae2d72889153d9bd4fdf0b6a31048c2739500077c1fab877047b94ce5b6df69d1a164a0f1164ef29985f833c0b27a6a64747ca2165a6de2b8c838f415041162cdf3cbc028978669046f32675466b835a337dd17b26c742400901313603fbaafd9b9dc2ee76867de15c515ba8a3ddac50b606c0dfc9ef3e9bdb860a014be94257e76c4c0957240e07b32bf0a494b4e72b28ff32e6c51b5660e705ff99313b890171863f8296538c0df71d959117b23991ab81cbf8793b09a2818127cfcfa4826adb86e609015b05d98d465ff9fed32b201bcff0fbf6a2406f6c98dade1505a29b0be6d96630a17163d25153fe1b2f6f994e0f5fb2d47a84df264f0c66ab9e09010b2107ae09010019933f090108afd9f609010806212009010133ae7409010a1ecea20901068018cd09010c64307509010669606409010b2a8fa809010601f0e5090107b21d038901000000000000000000000000000000000677b0eb09010c41cf7909010b8e7b8009020c2c70740901095a7281090109fde82c0901056783de090100579c9a090106bfc71209010c377173090104f1c34c09010077ee580901007444c9090200e76a4a090100564914090106b5d97909010019a3de0901036d2629090208086c30090100041aab09020c67e2ee09020766c88a09030c6805f7093704faaceb090200462b3609010ba8f6710901058faec509010b533a5a09010b6f0909090109f8fa190901056d73660901073f475b0901072886a6090101b8cf1009010264c0440901000290b409020c6709578901000000000000000000000000000000010c09638309010bfeba32090101309dd909010a0b6b9d09010c6808ea09320638faec090106106b0c09010a3b7c1d0901056d5fe9090105ddeaff09010071973209010a17304c0901019ca8c809020b61f03b090100676af40928001478ee09010bd9a6fe09020005e4d3090100327de70901008c6dcb0901004dcf1609010b07c96009020669c2f5090106706ac809010bdbbe9809010561870b0902003bae2609020c68079e0901034021c009010660d93d0901066f19390901032648dd09010aed18af09020a400f7e09030115cc6f090105d6ead3090100e5558109010660ef780901031ded4909010c67fddd09010520bbea09010c67de0c0902086ab54c09010589a7860901070e5651090106eb4b2f090109f9d2270903006593160901001109cd090109efaa8709030b209e2a090203189bd109010c67df880901000a2b32090105fd971c090202009826090100445cd0090109cd4b3a090701c197c909010b4076e1090100053ca909010c6807a009350bd9a3d4090109f07b6e090104d18eb709020491c84209010ba9972c09010b2b2f0509010ad6dcd2090100414751090202ea1dd1090107582da309010c67e4750901074faf5409010b4085a509010bda074b0901029b6ff909010b4bc7a209010b0033660901002a5be50903015e042409010b55111609010775e475090105b091920902002bf77a09010b8f00c709010289e0f509010b39f2b60901060b2d8c0901001d996c090205fcd07f09010046488189010000000000000000000000000000000006aa50d5090107cbf6970901000d9d7209020bd9be60090104e9bb72090106103a1e09010c67db19090302a8b08f0901032bfecc090109cd026a09010c54a109090109c2026209010c22df910902001831a1090205d2c8d50902002689ca0901091fb83c09020567809609010bdde3c709010c677e4609020017b82009010b2ab89909020bcff27209010173ca720901000fa0e2090101ad82dd09010bb3da6a090205a70bb509010b85934209010272cdba0901000e553909010ba8ff2e09010c680481090103fdd1b809020c526e49090107a89cdf09010b096577090109f9c5a409010829ef7b0901002556e6090100079f640901035440a4090107b21d06090205e37a1509010a16ca5c090205f310ae09010ae53901090101af593509010978121d090104eb6451090104f7d30a09060bee7558090100001fa30906060fe9e3090109daec820901023cd82409010010500f090100dc86c10901000f844b09020aff4bb109010afc8d38090100107676090106c0b79e0901089c8bd20901057da16109030a1625f009020233cff309010c614251090103e73810090109d4f6ab0901051778c409010b409e5909010610117c09010389a099090201d1637509010329bc3c090105554ea009010055bdda09020035758e09020c59cfd9090109765dcf09010b77065509010304597e09010693d5ff0902060f661209010c096e9e0901074cc6920901041362c609010c5efeec09010684f20309010bc4c98d09010bc4ebb80901056789be0901031fe67009030b5b499309010a0f25fb090106c0bdf809010ad6158c0901001d5ebe09010b258a7c0902003b5382090103266c4609010b8566d60901099c928e090109f4640f09010ae5292c09010005acd009010860ce950901070e64fc09010033df33090200c29fd20905002745410901065e5d7e09010008644509010bf29a71090300d25ae8090201a1100b09010ae0313d09010a6bd8f409010019f998090109e092b6090106b441fd09010326a3130901005b30060902000003f00901064f91be090101e5a8b2090200001fa4090309a108fa09020c6143f90902049bdf030901002f54ff09010be5f3cf09010643c35a0901009894240902063fcff80902087713c309020b7e71680901001bbbd709010babfa4109010bd0d3a5090109a178db09010017e472090208f489a9090105ce03f8090106eba5b5090100782698090100657ec7090109d42ca6090105e80d1609010b8d46f109010b185fb9090100014ed609010401a1ea090102bf6e1109010848de0e090102f7ea4c09020000f821090105d9f3040901018116d109020ab7244f090109eaed9a09010554bf0b09010c680484091a02a1a2970903012b60e609010a293b5c09010c67d77709020c67f3920903009d1c46090a031392fe0901000429de090107f0739509020c6808ed0931025b4ca309020babde7109010bcff83f090105677ed109010009588509010c67ed1009020a69dc670901036517b009010c67de1209020bd007bb0901088ae94b09060b197ff009010577ab5d0901006b80db090205677ed209010bc7af2509010b500dfa0901018b7176090a0b0f7e6f09010401387209030961737509010c677e490901003ade5f0901000bfe3b0902000b78130901065d17b70901003e69fc09010644dfe30901079dfad70902086ac190090106339b8309010c22f90d090104ec10ea090101c9dfd309010b8d5da709010a2a8e080901003bcb22090400111b3a09010201bd1509010445f9f909040bdb22f6090109310235090103f4df31090105e81baa09010a22e6aa090101af525b09020c1522d4090201f129b30901054756db09010b6f9eaf090100fb09e609020c644564090105678be709010a0502ef09010182b94f09010a6cb63909010c51c6bb0901021f4d3109020c2258a50901008abfcf09010124ab40090109eb2a2f090204ebff3609020b106d04090103a75f4c090109ff727f09010b2b256109010b61acce09010339be860901001fb231090105073d650901090adb6f090109a4f89009010b55e934090105e73d080901005ea21b090109f5f78b0901008b4e4509010567820b0901070e69cc09010c6809ff092f0b8f6b4609020b32ace109010207a1f8090104b4d5ac0902001eb0a2090100146b92090600f0a9c709020c6807a709360c15164e09020000b20f0901003575a5092503fcee1e09010a4127e4090100c6a402090205b0a97f09010064f1a4090205678dc909010069334a09020b955573090109fe7a6909010b2a989c09010687a8e7090107a09c9409010a81c1d509010a5a33bf09010c66bb1209010a32db290901003515870901055f03770901019cd93c090101571730090105678f7d090109cdb2160901000923c8090100e7171f09010030f41609010b17c7d90901054dd2df0905048d7abc090101538ff7090103ff66e709010938ea8309010000002f4201e5111d61f778000c515cc13278fc6fa250000b20d5853902c5a0525826eb00198ea73a01f04119e726ab07dbf5d1323eaf68e46f0008061fa53927c1250c1e32fe0133961d32287da4bef1000a1ecb9b3a06b5ec39cf3965067fea23323b13b95877000c617784322a4cf9fd2d0006625924322a9a3acf63000770e63e3ab017421402cff90b2648ac322811cbdfd90005bb81993a16345d84fd7ba006771350322811cbdfd9000c41cf8c322f58aca323000b8e6e9132d4e04b3c360007649f72391ef1b2e7fe60000c2c69cf3a0743da7c0b1c4708181dd232285e5fd7920009fde3c73914fcc59bb696ff056747893226fde45ecf00005797e4322a18d8824e0006bfae4732356b3cedf3000c376c6d322811cbdfd90004f0967732db65bc758c000073e7b6322f67162a2e0000743752329f7f3d0eb5cb00e75fc533216ac8e375c000561d0932f640e036610006b469933255aa0fed21000019a12d322f67162a2e00036d11ce3a0ec9c052e1baca0800f85c3278a1f544e2000003f9f9324df4f65319000c67df913251da5cb3af000b24ff1939354a6ba7a180020766b1ce3271823e01ea000c68011e3a343b838e24f88903a766c73a01435982a96300052e733531e35fa931a0000046273a322830402975000ba8d5df322f57dc0560000117fee83a0ec0fd33aa32000b44e366322a6c38eeab000b69876a32270b2fcbaa0009a5ad6b323c2d72f73b000556cf6a3a3830032768a0000bb23f913112309ce54000073a596532357a7712c10007287b3c328dd089866400085e6193312d79883d200001b8cbef3229770747e8000a419af53942af372e6f20010264acf732bc926e31bb0000028a8d324bf72208b0000c67077b3a03761b910ee9000c08b8b83917642f7a93e4bd0bfeb0de32c4e6e72ce00000b2c027316d23ad5f800001308cce39124b95997c0aea0957158f32285e5fd7920000903f65390793f3c34e50030c6806043a2f7a8b284b0d220638f2e0323b12d6d9110005d03aba3b37b810daa23fdc0a3b71f539019fdbc389019c056d601332bd2efc5b7f0005dc81003226e74eb8450000af040e390728a1bf38f000007195003b257c71d9d67eae0a16e2143a14f68cc320a078019c89ca3b9a6e7f18365f3902d458a031062f3f95a0000b1d051f32277d0a778400006765e13a26080a264e369000043a133a1b1e27620155000bd9a58732a7db68ec68000005d782322a4c295f6a000032762a322933aba83b00008c61a2322bc5115f26000c67d1c93a1eb22be53345000014300c3a07448a4b8298000b06031e39065bc1c2d7315e05d2adbc32270b2fcbaa0006604aae32b1fee9a95c000bd4cb403234cf082c5e000b951344315af3107a40000561873c33e7dc91cd9228003b993c324dc4ca1fa4000c6800363299e1cdd54f00033f6a0e3a056594fec7584a0b951582312d79883d20000646b60d3a06bb6eb3f15c15066f0be832356e4fb5f70002daf38c3a07231c46551a4a0aed136e3a0643f444b194a50a3fb741326aaf505b940000cdf3843a07cc564ee727180594fcb13a0139dcf5e45bdb0b2850a632b1912dfe130000e522fb326b804190e200065937073a0820accf6a500002b4820c3a2bd23a842363160c67fde23aa03a8b22428f5505209e92322e72af2b4d000c0b3a2e391572c8c51ff82e086a9b313a523ef0b8932d00058884be3a1772f6cb55239806c64f943226e74eb8450006d5dc133a063eae9072467b0c60b8d93903e28a86bcb55901665829390e3b276187c40009f9d0de3941b4e2f9cd009e00658ca03251b4632f30000c5268c73247fc441bc0000010fa7b322a6c38eeab0009d9f5e832715f4b59ad000246170c393822042b7380000ad0d95032f745047b660002da86213a052f09f7852cee0c06e26f3a1710a4c929f0940000214d412b85e64d385c77b30009694f32297636aa250005fd944a32b1d618f7a300020083183a08adfb6136a4b2003c28ac322a2f8bf63d0009cd3d6d3a0101033d331c0001bfacb0323744c4db25000b3f3b9c323d37e08f8e00000532c6322f67162a2e000c6800383a32558ca5c52da10bd9983332b2aa765d240009d628a0322d910c4ed200041209b03249a81d9a880000001cf609020491c5e1323efe56dded000b94ee213902f5d84a4647480ad6a75232fa28e81f6b2f003f05e13a02164b8d282d0002ea1de13236a0e12586000678bd583a04b4106596d6ce0c0e701b3a3cd5042bf7fb20074748003a0d0393910a8a450b9bb088312385416c274a0b40549b3901a305729a68dc0bd9f8c63a02fb5d0e94eb76029b6d39323df73d9e49000b2daa043a13d3fc726a7b3f0af9e1273235e3ae6988000029fa563a02e02b74baa996015dd46f32fa29b8bd2e2f0b5506b63a2421dfc3d1f5eb075444923a037a2968c32800056d1af032e702fbc5e0000055ca393917968e9bddd0080001fea83228203566e4000b8ef68e322f57dc0560000288da51322810fb4216000b39f2c1322701a401500003b7a7853a0d39d8902fa80000066915324dad8d947e0005ba36213a21fb6c3f63a66905c643581102d106aa20eb32407fe232bc0007cb5f9b32c46d7b5a6100000d9aff324da9c2063b000bd9a3e232278e0f90ff000405f1f139037803638fd62f04e946b03a01df501fc3960005d038d13b30f58adea393940c08153e32e5bf87528b7402a7fb26322770127cf70002dc122c3154536ec440f409ccf4583908d0533492945d0c526ac03a01b09b3908ad0009c1f0063934320e8087ff610c1b5e6732f8cf62d450000017e9004101633f21bfaab53405d2c72d32993ddafd1740002678f932271d64e0f800091ba0e4324753be443a000567478e3226fde45ecf000bdddebe3235ada42c9700034a6a64410122129a8a42fe000c1ad0a93946b350891e1386000b1a62410dca9095f543bb890700171f316d23ad5f80000b2ab8a532d9bae8b7d8fe0bcfaee7322f57dc0560000173c42c322f57dc056000000f87383a4763879fd9270001ad5872322a6b6850e8000b31a6333261c4c82ee20005a709c532d114f29e1f00062bb81f32323f7f4bc3000272cc643a04142a767bc011000e503a32356a6c5030000ba8faf6322f58aca323000c68031332ce2977ddb70003e179023a01ea112ef75f5e0c5245b231651e854f700007a7e510327e36e2e51e000badd1c5312d79883d20000b0948c6390e232b7f90b18009f9c0cd3901c981369935a8081d25853a02db4480c5136700250cda328b033b890c000007995632cb246690640002695eb43903ca5465a9e85207ac1aa83a02748344f70b0005e31241322811cbdfd9000a16ba0a324db2e87ca40005f24566322fa33a0b65000a3f3374322add966f10000b8f8651312d79883d200001add144322a5c3421fb000918ccfc3952be7de57809690420d23532270e12e4a60004f7d31d3a01e3662d9197400bee5edb32942c4198ba0000001fa5421fb4f598aec9fbc305d034883b365663bb27bda109dadcc339023b57fab9abc8023b72b7322810fb42160000104d7f322a5b6384380000dc837b32356a6c50300006316c8839237dda214e6008000f702d411c85284fd99258aa0c680a013a1ca575c01115000aff423f3913d173da3cb9ee0afc54c5390a1dd6054868b0001076af322a5b6384380006c0a59e322add966f100007f36510322811cbdfd9000559058d3a0105044d7b72be09e8400c3148c2739500000a15e993390adb2d23d4f1c90053bb7a3229606bab7d000c5dd2f5390496452197357c03e732933231bf70bdc70009d4d8633911a601d552c7be0516fe1e3b2a4015f26990c60b4081df3a47fd952fef7c2f05d036aa3b34c1a6c9bb52fd037490e132b4faaac71f000134ba45322c199bd1750002dbe764328cd77d63f5f105552739322f58aca323000024b54e3a12c21704b811000000f5d1324db205fd3e000c59c9583a23f10a92d0d7000c6803153a2223f0aad3b31f06648fde4102fac121da0b06510b76f9c23a1bd716e4db0900030430af322a5b6384380005d030ba3b2e3fba4a4bd9e30c096cf132356a6c50300006b079ee32270498fbef00041358f5322830402975000c5efd2639dd8b4acbc77a1805be472632270b715c550000002a3b39c060f02190c8000bc4c5d232942b70faf7000bc49f86322810fb421600056749353226fde45ecf00003921e44102ac9ba10837c000031fb92f32718ce7e674000b5b327c3a01a5b28f1525000a0f1880326c95ac800d0006c09a98322add966f10000ad48e973a0d0f9cbaa2203a095834d5411f2178e1a8ab7001001d574f32356a6c5030000b257bc13912ed745f2db6c6003b539b31940307c2800002dbc10f3a03f7f77b2d9cff0b85453e32fa29b8bd2e3b0992c0663b1d0f4231a8990009f462983297ff1385f2000ae4da1a390468b4d98d051c0000745f3229460a76b00008115b7b3a47d8712a5b7f0007014da33226e74eb8450002d9a53e310a1443614f4005413e8f31246139ca80000025e081324df425b5560000c2964632b33c07301000002728623274f1bc97b80006293bac32270b2fcbaa00000861c0322a4cf9fd2d000bf24cd1327160dab3900000c2f2c63a12b83a3fdf3b0001a10d4d39d88c554f4c22000adfd56839042a69f4f3fba00a64d62a322a5c3421fb0000196c88322e90a053930009dc4dea3a0f35b653d0b87406b43e2532357a7712c10002db8b1d3a052baa2d561de20a4c1bcb312d79883d2000005b2e2539333fab2c9c7c6c0000028e328401ab13430005b3148f3a0ce50f476c3e260c355294390a9f639fe0e00101e5a2bf324daf8242520000001e0b41019cef1dba180e34028ed6a039018a0c4092045c099996f53a037afa87bdd0ce0c5dd4fb390333d8242113d502b6623b3232e663e7c300002f4d7f32a857c9b986000be5f07e322be432833f00051223573235b6418bc900009894523a01191a09bdfa00063fbd9f3a012e2a4d79260008771082391179237adc68440b7e6ca93269d57a4088000c515cd039153b43d630197b001ad4053228210604a7000b94459d3229561b2f22000bd0c4be32b911de96b000096b03c632277d0a7784000017e2893273362bc8ad000a4b3fbb312d79883d200004461c2231f5904616e000070017223901ce692d85d0ef08f47ffe322a08981ad40005cc9a453a054440fd31100006eb2c8b3a23d6a64b3bd30000781b5b3a0bdb2f3e28709b00657a0d322967cd231a0009d4263c322f67162a2e0005e7fbfe3a45aeba54a0a7000b85f152322b176c3a44000b1856dd3a21c930528009ac00014c3d4107f67b64c9c589c50401050032abab044e560002bf5a793a0b2f4987b940000820a29032270498fbef0002f7e7c439226a8e88a3cc460000f5e03a2402e3036eac0005d9ce71322ced9fcec700017fe73f411a60757d070cf2000ab2c07232356b3cedf30009eaec2c3902a8c3eef4bdf10536bab93b0bb897ffdea5290c6802073a18b53e98fae366011911c73269bcd814a600011eabad322810fb4216000a291784390398664f09f3a70bfefcb63a02ba5971f628d600b1e4593910bb4b0ad579260c67ee413a023175f84aaf00009d197a3a0985cb050e8a7602c0dea33242a4c825f3000003ff353229770747e8000b1fffae3a04ccd605d8bb0007f05a5b3a0149cead855e000c6803193a2e8727771b48ff0a6de00d312d79883d200001f8aefb326ac4fd8cd7000babb53239bea2a1819f49e20bcfaeef322f57dc056000056745ac3226fde45ecf000009413d322a5b638438000c0e77ca3a3e9c85ae3d7ea606f90fb43118a5981233000a5caff431766e2fc0f63402ce87dc3a14ae70a14d97000c01a96d39122116f028465a0bcfaef2322f57dc056000088ae0f032dc59a05317000b19584b3a172396df85cd780544f49932270b2fcbaa00006b60b33a012db205b6b800056745ad3227b12bf263000b5d01403a075ec9e69360a30b4fface32372b4625be0000f97b5f4207212ca7732656000b0f10aa3905907cb88de7b40400b0ad3271e1f57a1900095fdb52323513db4e6a000c1afa8f393f9808a3ac5731003ada48322f57dc05600000004065422a566631615bb5b800004440324c2a9c9ef300000b6e9c322c91ef7fa50004f62d423944d0e9914ca00406589ac2324a872ada7400003bec053a044295fed86f030644d56432356a6c503000079d46e43b09e6cb6ea4eff2086aa23c3a2af736c9560500063336033a08ba03e4cd19000c22a54d322811cbdfd90003b30853322d858bd6a40001c9c26a322a5b638438000b8d5454322a4cf9fd2d000a2a713532b5f4a848f00000204dbe32912d43f9cc0009e8ae443148c27395000009e8e55c3148c273950000001113e9322b8da1817900018197123a04d45bf4dc3ccc06da451e31020a1f6a7900034a9b4d3a098577d4ae7c000bdb20ed39a5c06a979d684101572437410140500ad528f01f02a835dc322966fc85570003f3970d3241127f3e350005e804de323b2b84f0b5000a22632a3a070aeb2b01585705e8cc8a391930514375200001add14c324df5c6f0dc000c15185432495f5531120001f11b9342018e78b1397ed9000a60083439012d908be563250545776439277c14d8cc4f7c0b358554390843669f6bb4e100fb054b324bfe9b57d10006e346b04104dca0ca6af1c8c50567493a3226fde45ecf000891af2732285f3075550001828d41322fbe7dbf9800092dea034103810f6985c940000a66ea4f32b4e297a855000c517d543259f4be52f500020717fc3a0cc93f03899a000c672eca3358e0ba35b3000c2258b5322811cbdfd90000131b163a06bdd7efdb2a0008127dca313691d6afc0000124a56e322ac972a1cf0009eb1fb43a0159195a41640003b356533266b5a26a77000b103ee939064fe25927e76f039acc16322701a401500006107c5b310345b9830c0009fed9d7323a40f7389b000b61a0ae322beadb349d0002be170b39083b2ffef84aeb0ba7419d3a0e35fa931a0000001f7926329d5411b6b40004a3dbe83b1e7053e20c7ba508feed23324ba179dcbe15099d6cf53a01e664745bc5020b55d8d0323c28f4989a00059174363b072f2bcf9e07bb005e825832285f3075550009f5e22432c17732f229000057c1e2323ba81b62bc000bf6e3fc312d79883d2000056747933226fde45ecf0007016e073226e74eb845000c6804913a2ca0e536ab5d910b8f6b563a039bbf383c98dd0b32ab86322beadb349d000207a211322f7720ecbf0004b4a830390149c21822556400011e7732357a7712c1000013e14732dccdbd29320000eda10c392ea24f4ea033780c68020b3a3348659e826a4a0c14ddc033ed1b17a818340000721f3229596991f000003558e23a04c43bf509890003da1243391069d0b428cac30a410b2139027cae0e8d70d5000dbfc53a05b7b60044e80005aefea23226e74eb84500006145fa32bb77a26489000567493c3226fde45ecf0000233c193904bb4ee9020ecf0c51976739012f0b2ba3a0000b90564932dc877e70890009fe54073902114e1a1519300b2a91993a0104ae509d09000687a60e32d41d81fd4800078f11223251b4632f30000c67d1d73a1e66a2ba70a8000c5268cd3a13a79dca1b67000a78f227390b3c83fe966f7c0a59c3d0322685433432000c66bb1d3ab9fb739325a5000a326b7132730570882000003499fc32ac97e6125400055d1385322f67e6c7f100019ccdb439d88c554f4c2200015704ba322a6b6850e80005674b133226fde45ecf0009cd90283902f25cac4379e4000913ab327db8ecc83c0000e6ba96322a6c38eeab00002ff8bd3ac38f22a3e60a0c0b17a27e3901af152105307305488c9c32f1e2a3064500048d38093a025f94052c290000c52e4a32b433aab01c0003b031e93b0ba92c6de0f24c093898c9323b9b11868c000597bbcf42397f022f767c7a7c0000002489010000000000000000000000000000004d070a5197895200000000000000000000000000000054070a5bd9006d6ddd63ddd4a2157b38227c662e6d7952854772e2f3579dfb6bbe6a81973055070a5bda000709fab9fd95a8a0c200e329921b1b8d6800f788144e4daa7e50acde6057fc94070a5bdb008ffe7c315ac533335a182ac2b299acffe92810b4d5a6f3392b44c18bbac8e384070a5bdc0000f66c0a665811bd0cc14d319339151f8f59ef9ce4205ce2da326a8fbcde8512070a5bdd009fed2925bae3fdcc8e876587ae7d84a49bb7b05a241c01ca8c641ccb28e27fa1070a5bde0056a07be9babf358e6939609d16bdaaf25d0d7d5ece9e955b7e61e0fcbbc512a3070a5bdf0009900c19f7270a23399e2a1cd7a18623ba071b7d024853be75ac912159ebf289070a5be000bd852a8b80acd256fd209287595c4a4b487c82d16bac6a306b9ade6148f8ffcc070a5be1009982bf903474e958933dada56128d7003252a71340d03bd2a93baa1f47f7f4cb070a5be200377be92ff2e22a80607b9682548b57db6a1ff7b4e853f81d79b1e06a8464fc97070a5be30061bd1f1b93743d45db854d7f211309f3eff323ee728152b1b97d956f2c4d3823070a5be4009290725faece0b8804c81b21f62a92b06acf54e28ffe8bdf870aa2ddb7e09130070a5be500697b1c38d6b9582d41b8adf4a41a0ff6c20e15bc81c3899db7a61cc594f448f6070a5be6002efa65cb2ac822400156a605ab9a435e1d3d278e825efa3f2435cfdf652bedf5070a5be700c162bcb089f06c0cfc7f00e2e4863e2badd3e086db45acf08b852b4e5916208f070a5be800d8839e92cd3f6b32e5db6dc273e928e88c3fa42e6aaf46d0798640745bda4712070a5be90074e96903c0088ba0a33ba5f1f3b0df345dcbf84d5a1ab344b56961964430218f070a5bea000f4ffcffbdcf3ede64895966ff0004b664165bbe0c5ac5877fa9a2004cf4e991070a5beb00f9226300947636f58fbbfcc0b2ea48fe6b8926a42fad01ad8bf17222532377b3070a5bec00144e5d780f5e8e63ab026594b2b646aaf4331c29c2aa3a98a8b2a2232586bbf9070a5bed0042c638e91fae55af2f7162c90be63d81e7d2a8d0c9c9b0e188566b1ef0f6d801070a5bee00f856242dda26db04d3bde02025e471aad0ddd5e990d3e0d56f0e963503c03ac4070a5bef005b5ecbc123ea027d3fd8fe859b92ea983a6bb09d02926cbced73163d353188b6070a5bf000c3574ca2d0b213f6580d8f2a4db53f9cf00cf62bc1cbf37bb5ee4aee314f7fb5070a5bf100baa903bbb707977a5dd902a116c217f00a5f5a3ebb4054c9b5912274d5d1e3b4070a5bf20013c90febf653fb0348d9699ddd7d8947d59ddb1bbc32035985a47cbb24a2be60070a5d68007ac4b937f9565187f243c181707b57a59fd1f5f39987b15b33419a63ca72da01070a5d690072d1ade4dba274d2e1ad9455a658036f58bb8e6949f61b8e20420ec2e9d61ec2070a5d6a007188ebb23df5cdde059cc951110a7f73d528063c1605776c43b0eb5770d32928070a5d6b007c0fdbe358890e7bbabd96af73caed7804a97cba21783a050b88674cbb567a65070a5d6c006b4c52b358482aecc9d32e881bc54015fdf330b2c8e9dc2eec54074d684dd39d070a5d6d00b94b9d12fd4bcb411a316ffe858dd56ac6911d118663cc99fb84eafdf7095937070a5d6e00f1c1ccb97fc759e38fa46bed43b4f3be2f317cc73ede6e3143f23eb853e06b7e070a5d6f00e13d13242407f1de4f35d773c34319ff16b3689e44ceed365f6cd63680c3d030070a5d70008c29eb8041bc65d83ab58c2da3211baf91181436a3b63841ad747622faad981f070a5d710090303b47416508494b02ef5bc09d8c5ebd82aeb714ab5dbdb5d3ec2daa8ee5a9070a5d720031cc2923a8c7151887f9ba4ecf0200e239f34e1069e9f965997ddf0cb53b5d22070a5d7300f91229b9061141167361f7434fd4b72bf0d3e96e780c07a772f229e8b92ebf3a070a5d74006b4df17ac3c6cf19bd6c9629fbf2da722dc1170911a1773e6e7893135064f1c9070a5d750097ca97d150a2073e39dc94f8b0d9a9a6f88420fb532854aca77603f1675f37eb070a5d76008582c2e971c1b4a19f2f3fd9c3f4a89e36894a60abae8c4c8b2522027cb15338070a5d7700dcb00fc263f4cc921eee1c92673d6eea3d0b2ef62973104e5e8578ff34684204070a5d78009246e2bfeaa6057a6094f47c3fb865dae8b3c37a14580fcd3b1077175be4052d070a5d79003365ee5d0809baa3b40b430e00295269ac70b3d8a7cf9a7e1495ab27bf27b3c4070a5d7a00c458c29d92e8a8f8794f6feb4cec8fa54472c8b3c3a3a3a3a25ff76f657a08c1070a5d7b00ee18964a3e32fa71b872c951d43e130de1401309ce478cc0b5de678fbaac113e070a5d7c00c90a52fbf38019fafbac330dc28cea28f37ff93d139af7db00f9437ad1c0b042070a5d7d00fafa57de19eba9f28dd376ca2a6536fd93e85c7d833340801e060b5d68959def070a5d7e007069c6eb8fb74dc61c32f9a7a4ee845ea4d2ffd3e64d411e3bd60b72cead5e37070a5d7f00022beaa918accdae2feec5f9237a2ee83d4b82f265ce10a24d772774a1a42400070a5d800062a78e575d55449a5c44085e8d183504e8423d19c22de694a52e50bc5bb1b082070a5d81008cbd7b1bf67f3d8bec0a0414099c3121ec834fc71571065f15eae86df2650ffa070a5f4600d4850de787c7f43b65848bd2704a33411ddaac19bbe343c76f36fbb226c09f7c070a5f47008bd4f9a36128af6473be2cc7f3e6185df45d207b9eb375c4b3896769bb7d9395070a5f4800f8f2044fea300a28e9baf449e9908308b7cf41cc3c94207c7c2cddbd80694a21070a5f490095956db16a70150ea5cc2dce87bcb0bd142a8de3ab43b76e1be7d05b6be6ea09070a5f4a00023c43e4072d5232341c2f8f7105118e040f75b94e8d8514d1897c0d4cf8f374070a5f4b00cb2355cd0e3da87b2fca7f1e10bf45ea7ff04b6823728cfbf3ef28ed50b1b9ea070a5f4c00d0766acdca630024fd9e93dc0faff11e4088f53382b9ab594a3e0216fcf8dcf6070a5f4d00d4af8767239c07c84c2d913447ffbbdd2eb208a6442e2b73b1479ef8052f1c10070a5f4e001031fc14a65bffde9d99bd4217249e0702e3c1292a363dfef4d0df3c87f646c9070a519800040dfcda364264a10250a8351d85760e5f1c792e1b9db6bc5bd1de2942992163070a51990010a917a1b1b18c06b4185c56a71d7324185a40157e4946c2a220c1b8cc6b8125070a519a009143965d07019d59398f22ebc20bd7ec5ec20ce33528047b5f786329a4d37a16070a519b006c01d762f30c93b4c3013d4ecf8a9949439a474ccfe9fa2ff1355ca2abf0a303070a519c00d96f7c034e15beac5793b085f44ef0651af9a93a000a7a4b76b41b951836d264070a519d002b08d63b52cfb758f53ec40ed6b8336fec91385f9314120a481870e9ecb29ea4070a519e0082c62bde8e27cb11b1830685f112856273ecee17840f761b3c1f8d999d49d6ca070a519f00e18edf2dbd13501903870e36b56b5a567aab1cd0cef198c4fba92c2c834c4641070a51a000ec127887571a1cbed8589954e486bafced260a12cc4d64270eedbebc8bb3ad5a070a51a100007808b24cfac2044ffcedf329826092efb7dad9071a3386e34667a8c6e207a9070a51a200ecc041d8572ec21390650cd1cac935a8de9b47e24097071cd8d867f1cbe5db6f070a51a300584739c60fe19e4be6127ae4241430e63758e1b8dd181be3ec1127561e5deeaf070a51a400561f4aaa956190fe717f30152dea97e9a79ccd8f8c5b2641e36bc5abf578303a070a51a50083dd8255250064311b6cdc7447bb2fe2af5734fd9a740e2f7bd7c8ece9a85ad3070a51a600b6a3cc5e298d2d526ba7d457f291f82e1bc23374e50dcd30e89039febcddf767070a51a70008508bc8161c992ddf482fdb73912e60937b8b8276e7befbdd4be54eb506398c070a51a800bbeabe7a3667da4b51a8b94b8ed9a3fcbdbd5b26bde1d58060fe49dfabc0fafa070a5bf7004a280ec902ab384c80401403f27f293f9fb9b64f3bf0312ffb380ba16b32fb25070a5bf8006d20d1f212370912efa1d590db2e560c18886f5c2d74444b3782ff5b0ad86960070a5bf900e35b3c32ee3db94bb7360466f310a021147325bef615b7ebfd4b244046dddee4070a5bfa001e0dbc9f581223db8c0bc92de7823a7c60f3f4c87062364a4f359461f49578e4070a51b489520000000000000000000000000000005404cd9b8b194faed20b243cd9d10b00a0dba8c0fbbba183fffffffffffffffffffffffff95aa3ba0c094696d112972607071247ed9ddf00000000000000000000000006f50b180a44533df10180ffffffffffffffffffffffff8346000000000000c05cdff5d0a500000a44a69449fd6824ad9b95702b410a44a695b10248edd51289e8a7cca89bb648ee14afe9383b6137a30055ca673903bbd58314d3350055ca683913dab918c8fcd300ae069b095b00ae069c29214b5a810600b1fc8b1902388200b33c1c11aa8b00b1e78621014b1f0d0a3f8c060b130c6808fa2102556bc106e2e8ac71060eba4d771d5f2580c4f55af0d806dfe728aa240000000000000023899629915b43fae0e5eb9edb06dfeb88a29854d091fffffffffffffffffffffffff8446d0306dfe72941049f5e4e0000018806e2e8ad1109990c6786d379435fd717081259938d77160532a9ed0c6786d4990682c5a4e34ebeb12fb14d5983e8634a22c20a0c36b9b4792cca63e4bacf5b5cd10324d2a8ac2b0c36b9b5990477339d209eab7ccbf87b6165a05c39e4150602831e4f4202fb6929df376a5102831e504102fbe64019dd65350b8e7d874904e6167b4922d798500bd2b8504918650127cc3dc800000c6807c14a1b1ae4d6e2ef5000000be4a3da4902b5e3af16b18800000b8220b7030afcb38b4a04e6167b4922d7985009e8ae590901022d52a01af3d745022d52a119f419330775cf5e4155fbafdc33c988000a21c72842d1f3c1295de116ad0000015639711dee449a58b305c673c01924259a000f0f3d490193f24bd6d41100000ba74de54901040ac1d852a1a41f0000215041e9251466e89f46320c1608bd03000021c64101c477b9126962cf077630254140ea3a16c0959bba009dc7e9030c6803331303050bc7f393427672bb4157efc3a90a38ff6d291b6103f4190a38ff72f106e1000000000000000855b5b3b66a0000000000000000000000005b8d7f0a3900fd99a05183cfcfaddf218bfb3cdd50c9259c7c08100a3900fe614b8f82b2fea145776931c3580c5ea8a2030c02aebb81c86d6ea56a1115b76e77ff60a3bd7b2f0b98862d893c0000000000000000000000000000000004e468a05103830e756d6e90c1f6840a60470f4921aa81897338b5df6b04c7e4bf5203830e756d6e90c1f68401b401364a21aa81897338b5df6b000f7641490193f24bd6d4110000000f76423a16148d3144922e041f12ecf101810000000000000000000000007cb9ffffffffffffffecf0504b95f11604258eaf5909cdc071e8a2b2a49315ac04258eb0a13b0f43acedbe1a5c683f1d1926ba5391134fa7e00153fa441a2ceb660153fa4539041533775289600153fa46110100015408bc3138682b16a000015408bd5105207a0234ee0d634c000155a137390846e94ae9209502d98b76096b02d98b772914e31dae59039809cc298c39e3218a03980c01292a115dbd4202d9a56e11d83c0542e1f3494efbcb440149940000084b25ca0305c64e50310156768521d40c23b1e742023b4af5ee4800000052ea914a4ef98ffa61d1d121d404cc8e927102a44d95485a4cad0f075164febf04cc81968903feaed71e34638bc44dfb75189dec1a4604c272d4aa0f000000000000000ecec162a8f52c310118b096c304c272d5aa4639526cd70000000000000000000000008c1859f204c272d6410109a3b00000005804cc8e93190186a004cc819731024c57b536340c2cf5c281017c1d7b9875cea9b016f67a025c06d20c2cf5c3992dd01d70de3fd1996b3392218d84b5c6b5506f0c2cf5c481017471f7d38aaa339b93033f17148a8e0c2cf5c5992d26fc22f757aa27d0f6ecadb03b20fff250bd05918767095205918768290fa9a8ac5b05a4deda31058160c9fc6305ae0b3a3102da016d8e4f0597bbf12a04cbcd780d0c680a272a0aa6d5da510c680a28290fa9a8ac5b0b3dc6c70b01072b4fa5211e3a627f0803977d13074a00d1e36022030176dd00d1e3614902b247fa7757ee000009a4f8df090103f40694090103f40695419f325f4a4e0a91d903f40696420b9163762a3984a703f406971117d703f5ca7e5918babe0c3f8264355d065f03f5ca7f5901cd18a7d5250e029941030b81673a09010c680909a2974dfe99c61e93bc863fc8091f8c6948f693b58e03cfea0509020c54a13f030000928a1913ec630000928b3a01d02222f7ee1f0000928c09620000939829900127da480000939949045f1622cf1cb4925c02cef47819a71b5b02cef72d39086727102ac37109c0754f090106534d6a8902cb15f25e1bd992a6be8194dc6fc75d310652ee338937000000000000000000000000000000000c6807dc425aa86fd5c7f81c5c0b2910b641b432ebf0af45b1420c4b625842b432ebf0af45b1420ac1147841eb586625c40c51cc07782a0142eb586625c40c51cc005502ae19766e41005502af3a0ad21d9f971e8d005502b0096c005506dc299139616b28005506dd49046e64158b4d62942c085da0520901085e61ef090100001f3e22138ce20007a8a0941902bb740c43d98f03003b544a21047eaa720c5480a1030b1216cf0b010b2b03d421016d63950775e4de191f5c36074fafaf198a956c06b5d9d91a21a32409e1e459110fc201af5d8e03029b178f194349820c5ec3d8030bac5edf0304cd9c331950755d03d46f17210f91b3590544110e0309a8606522015d537b029ae042210d9d2f72000f851e0305f312ea81ffffffffffffffffffffffffff9da130086ab5a221037bfcd300f1d782194c4b400046083422037bfcd30b0b04e9030b46da6c1a1afa8a0a6742bc22017b46330935bbc6030000219f2137b6a3a401b7ada42202686ee80c6022162101c9c38004c2732f211bdeacda0acb35850305942c2b0300274627194c3af209cd02c81a65f3be001994a919124c0703ff7866211a4fba0a00aceddf1a4c4b4004c7e4ea210f4327ed0a293ba51a2c88020b0ae8c01a75512b07cbf31503036e1acb220129bdd20153fa591a2ceb6602f7eb110300f0aa8f220214a5730102ee0d03025b4ddc0307cebd8d03008ff8922239a47ca60092d30f221dcd6500084bb4c6030687a92f030c67de5919976e6d000092a91913ec6309fde86f030a854f3729010becb92609f83ffd123cb109e1e2c0210301632b000021d11939565500fa05281a74a37b086b56641997a63c00199762220177c8d70a32db7e1a4349820ae031ae1a32442d0c643349030a3bc80019e287c5044b21fa0304b4d64b03088b1ed203034a043a1a53e83a09c179bd21018722430017918721030c4d520b258ace030b7e946d19404db30ac22e420306378d32030c61447913016902bf6eac1974a37b0b2076f01927192409eaf0c22120daf6fd0c680a361a1529de09e0930719a1d339086ac1f02101d42d5d076870660306706b42030c51c6f4194c4b400300f28d1917110a0c67e4af2102985e8609dafc861a1cf36b0ba6b530030018327f0309f9d264030a3bcdc41a170b070548bba821381fe6750728854a2106b5eb9908570d4f03017d0e2a030548b65519121b9c0597c0602286b4dbd405150aac03059ffcc6030339bf551a64a225058a22ed19fdd40100008330190e5aa80c2b4d42030c61447d0bf2007fc4fc030c67dfe219f8c2680c4956af190da9e809eaede21a22e0810b2107fe1a2387070c644247210b4aa8ee08e7907c190f42400829f019191895ec0c680a38220f54816e00e556a52184eb185400c2a10b220d9d2f7207750b59030c67d7d419e3a89400fbefa829012aadcac000a64da7030bf926de03005b4161030a22e712194876e303b209da0a010b90c1be0305fd9627220423714f0389a14119c93f77000040a62901b38092c8060b2df31a09c6490719cb2a0301f71ead0309f1890203056d7414210265b37404b8be370303cda4d821017b463301bb1af9030c67ee6f1a09c6490c558922030542e2442256e6e06c0c1e31270305fcd106210171f8530a09848d21030be47c05d483fd1ab4b3790bed89a21a0f424003bd2b6a19c966870400b3d20307711d662106916061007867af210423714f0b770726210129bdd202a8ec65030c23d02c0303a7fd1421037248d80ae580131a365cd807582e02192ceb66080621c5030b6353d31a0933fa0ba997881a2778340b14abe3030087fd6f2202a5692a0687ac4c2a010becb926002538d4030b209e8b030c38c9e10300147a2a210125a6100bd0d3f1030601f16219f0e4250c5e3ab31a64a2250c6803622102a5692a09f5f7f522138ce2000ba3650021012657db0b6f9f091a61dcbc04083fe21ab85abd079e007e21370b6ea00a69dcd21a0933fa0c680a3c030c680a3d030b9555fb1a2ffcfb0583c73d19a3f3980c67d4e61b08f6910c5b016d2109404e43098895ac03005b096503072296d01a3ca16d0aed19351a4c4b400c680920030007a0372a012aadcac0010e020719277834095fa58203080631ee03099761c11917f6de057dee3d0304c25216220f20476b09d445bc0300b1e84c2101542d7f0ada781d03005502ec19766e41002a6e1711d5c50ad615fa198b00e40c408f11030b60bd1c030b5511b521015d537b0693d68a212050fd28005483bd192fbf8f0572d34e1aaee69a090adbde03000024ee220201652e03420afc1938173909f9c6061a1735df0c6445fd1332bb0c4d126d03074fcb7b2102fbd5750c59d5082101876df40a15d24c1923cdd20bc86bf0210b4022e101a437d60a0109a10adf19204a9502cef4d01104900c680a400305865b30222791efee03798fec2101dfd52d0b33ecad03023a9b01030b54f6cd19d4b5300a2a8e9221012c9fdd01bba05b22032dab8c03fa370921083faf3f0bf8ff1703054757bf0303971d731953e83a09dd5306d1026cfffffffffffeaa06cbc6293affffffffffffaca8f0a9d0b309dd5308b9019945ca261fffffffffffffffffffdf6d885ca3d3875a09ec7a50b120fd2ed5c663fffffffffffffffffd5fe2ef583868390c44f30bb1014c03a6c93dffffffffffffffffffe59395970c4a2409e7ef9ab90176fc97a9905cffffffffffffffffe22811d7b48ed4fc0c4d079b030548b67542c9e3d2c58796218d056816ea59037bd927138600d49b4d90054da50d09270bcfaf66030a46e7f709010a4c1c3509010c5bd888090109dd531342209277a35c2c78a60b9aea8741209277a35c2c78a600fbefbcc99c000000000888a7d3486ee8fc000000000022acbedefb5f5e00fbefbd090100fbefbec21b12dafbfdd0c316fffffffffffffef05a67c16a75940000010863a35316a253dbddd58131cd890c451d92ba08a4df4daa4772ffffffffffffffffa925507d569c417e0c2e5ba34b2c0084229a1569df370c5ad8644b0ad5346741605a9e9e0c2e5badc20208bd18cfd592ddffffffffffffffeb8fad8f6a3a87d0bf088a0e048189944a67fd8887d4068d29134fc834980888d993890cffffffffffffffffffffede8081952df0c5d6a5e930db4fce241dd331254b45955e759b9906c9303d46f36c96100000000001de9d0a296be1400000000000716c5b3a2c19a03d46f38c201698cbb5c309527fffffffffffffff1d706c6154911e07f0c59b2efba1a61815fe4c121fffffffffffffffef77fefeff0aa5724042b6310c2014443fe6005c4b2fffffffffffffff34cd852b7ebecc0bc0c3c7a4faa14c2b90b47ffffffffffffffffffff2fd9791c6c700c5f9f89ba0ae726d98d041bffffffffffffffff92af5393f35e5c3d00906feb1903cd60009027a141053b66e17d45c8c7008ff8c42239a84a060c60fd6b29056986fc3e0c60fd6c29109c5d155b001a667129109c5d155b022d7a9a29056986fc3e04c2642a69029cd0dc9601befc844e95506604c25506891a99f2e294e69eec28fa1a6b7c16767aa104c2524fc1010000010000000000000000fcb116aeffa3332d33f00b0a04c25250aa7433e70f1b0000000000000000000000000000000006ff262b4162ebbefa000020c504c2642b11072404c255073148c27394fff2010f3865030ba64f0709010bb2404f090101053fb3030a64e88f030c68093e030a15fd4e19219c8d07af16c303029b17cb1a42fbfa0c68065a1a06a0980b93a5fd030c68093f0304cd9fb31ac47e710a0352ab1b0c4033060647131a4c4b400c680a59030c68025a1993c4db089ab9a3030b72b260191557fb0c675c0f29010bfbaba90574fbde1a4fc54f029b1a471acb2d600c67e71a030b4cedcd19b8848c0a854f6d2a010bfbaba9086b56cd1a97971d0a629a691942fbfa09f46490191ed0a907209e531a5b8d7f0b40a2c7030a38ffd6195b8d7f0b0966290309f70c441b155dc209cdb28a1a26f6c80ac4d56919debe710c68025b1a06a0980b0b3aed1a4d8e300b0374da1a2ec1ec0548b6ac1a121c1b0c680383198006490a6782ca1a01d3e80a74b18d130c16058bcb9111519008ce49311176c70589dffd210326ced0058bcb9311cf840a46cf031ac96a80022d539f215b6684f709f66f8b1b155dc20c526ed4191e65060b437916030b3cbdef19f8c4c00871fb5b1990b19f0c680a5c0303a7fd3922036a2d9b0903f15e1a93c4db081258dc030b17f0ff1a43236f08cddc0a22026ab6f00a0395bb1a12d6c40a4684f91a20807006dfec2219bb6a710c649e170309851e40030c680a5e030a3e1f4e2102c677480a0f266c1a23d3540bf2a29f030a41b3ba1b0c40330c67fd060305865b6f2127939b3707ab051c030c6806663357f30ffacdaf09e1e3004a02bbc099d5fb6d81280bc3a2e5030a4453cf41c05cdff5d0a500000c680667030bf2a48303005483f3192fbf8f005483f41a2fbf8d005483f511010f0054953531023d3340b4fa00549536310236c03d4e720c50989e0a080b243d32030c23c025090804b56839c2d6dd7ab17898f622ffffffffffffffff2fe802be26151a7504b5683aca02a36f4789fa589414fffffffffffffffadebcacb20b1ca37004cd9cb28952ffffffffffffffffab92170ec0b4ed7e04d06768190364bc0c5f584e0304b56840c99897a77f8874d8e4a400000000000000004f0f3a878a5763b104b56841d105e853ffcd7b1bd168c2000000000000000637c4cedc5e64372e04b56842895a00000000000005e853ffcd7b1bd168c204ccbf0331367f0f4bfb9d04b56849c904fbb05e29a93b2adf00000000000000017bdabc4ad2329b4e04b5684ac90fa55fc849c633e2170000000000000009617f4a280cc9c31b04b5684b89970000000000000001f4abf90938c67c4304d0f752190331710c09470709020c680a6a030c6808193b4139ff1553628200d0e46c3903fa9a0613d57700d1e3c34902cd764c1a0ed7dc560c68081b030c5e8ed2030c680a6d3b4139ff1553628205876e804119d3a06ac81297150748c19e390fea68184f55de0afdf701110f0f0bdbbf2609010c5fc9b8098c0c34d7b909030bd65d4609010b8f018c09a00c41d03109c80ba8fff709910bcff8da09010bbb02fd09a00bcff8dd095a0c160cfd09010c160cfe09c80b859b3d09040ba8f73809a50c22a7d009f90bdbbf2909640b4ba04e09020c2adcc209780bd65d4b09960ba8f73a09010c22fb4b09010c22a7d709010bc564d311012d0b859b3f098c0bdde48509010c5fc9ba09010c41d03709010c2adccf09010c3f69dc09780c34d7c509dc0bcff30909010b4ba05009d20bd00885095a0bcff30a095a0bbb030309010bd9f60609640b8f019409030bdde48809780b72178909010c22fb5309a00c3f69de09010bd9f60b09020bd0088809010b72178a09cd0bc564d709010ba90003090101b402314a21aa81897338b5df6b01b40232390d74cd588d20000c680822030903f17e1a91ae9e0c68094e313086afcfb4f50903f18259045b800d5c97ce6b72ff2d0903f183410f87ac9ab7364f710651176c390c4611e277011400bc17a53907b7a66cca943300bc17a641265ab3c59e3f772100bc0e576964988e2f9ca76382faf3a116e900bc0e584903eb051ef2a61f90a200b1e8a121014b1f0d00bc149a6964988e2f9ca76382faf3a116e90c68095159015c76b8474f383b7749e808e0734169fe412349a007b6375cbbe79c4300bc14a1590182b58557823d4bdad23500bc14a241312772e8ee420f780a3f945639011648c153fbfb0c68082559049f6505903d636cf816d20a3f8ca40b1300bc17ab694e7ca90becece971fd4fdc2b1d065117724902706925fa818562160c68095319191c050c6809542102556bc100b1e54f3910786ceaf8a97f0a3f8ca6590d32a56f7537d7531ed9f9004c69d14101192d1069581b94004c69d209330bf1cd0b09010bf6e4a809010c09473741bfc0251dd96f547a04d01b80c95a0000000000000000000000000000000000000000000000000575ababd1027100000000000000000000000000000000003a6925fc60e0cb057f68a4413b5870af75813c370b243d48b12f47d6f93de7d412d100000000006c8de48162dbfebe057fca93d1027100000000000000000000000000000000000000000000000004d01b83c9530000000000000000000000000000000000000000000000000c5f5872a9438b05bb3e6e93dd000000000021ed231b1b090ac40c67f860a90301b8236a6237e80000000003867715cb273eb69e0c67c289b10c1127813160048abb000000000667b65640bdebec310c23c0455103c530c2b6a7500687aa0575a294c95300000000000000000000000000000000000808ad477f33fb05f3ce5fc99500000000000000000000000000000000000bbcd5e2e44dff0575a3a1c95a00000000000000000000000000000000396e5a6e245acc860b4cee0db111ce59aaf96c4721660000000001a672584024dfb12f0019fb5f1106910019fb601106670b8e452049056bc75e2d631000000b7c17ab4a056bc75e2d631000000a5bbc6603054bb4ede9440000000000000000000000003d74fffffffffffffff69b8ecd66ab61054e942d5901bbcf019b03df3c7f5ef7054e942ea10a6b3d75f2eefc6aa202326efd8f51d74b6abe8c00002156412b85e64d385c77b302db8c563905bb61e44c980004b58913421a67835a763b5b6403d46f524201698cbb5c30952801e36ec70309a4f954391fd7ab9a6e37eb0046088439514dc947af5000000021a342051164cdb739473101b7aea939380d19109f100004c273d34202874b1e2668585904079c00391aa4cb2630f102036e1b56391b0028e44b00000153fb38390415337752896002f689273904bd39622592f2009038bf41053b66e17d45c8c7002f56bc4102b42e2f31b30b910574fc0d39072d8e0e5b01ac029b1abf391292226e1c5da00a21c7ed390b74c7f337512c000092dc3a01d02222f7ee1f00fa0594390aa87bee538000001997f13922482f38a444ea09dd535039019945ca26200001af997c030a38fff6390855b5b3b66a000b7e8cd83a05d7359402c0000300f3143a0217907aa9c46402db89a7410591d35e956cb2000548bc4542051803f39d8e974200ad94b403004b24863af5a61df9971b9404b2453f0302009a06390880740a6b53b200009e8e39017ecd4caaa38408ce4959312d694efcb4000155a23b3102183722066e0589e0193a491f01fe8f8e0705d043e9410b1db9419116800000fbf00e421b12dafbfdd0c31705fd992e3a03e28a86bcb5590597185f392e7a36d85e9d05054bb4f93a0964713299549f0c680524420570a9ec4ff40000041f14093a130fafb46a0eea0087fdc3393db799f395cba60978133103097545af41015c2a7b13fd00000000215731343ab826e82908cddc4839382f51fa27dfab000871f0411b12dafbfdd0c317010e033b3a03947134637b48000024f131122f43d0889706dfec2d3a1100df58f432a904c252ae4101601cdf0989b552005503383a0ad21d9f971e8d0054f28a31689786263000000024f2392ebbb29f84fa31072eabf33a243dbe4b2ae6a30a3e1f9a3a4079d386312e3102cef7e3293abb90b98001bba0d3394f9c26095777f203fa37b83abf35527b40a1e200ae0745490452103a5bf1b02e7f00ae074f095b00bc0f2d490452103a5bf1b02e7f0368c5df0901000040fe422a566631615bb5b8022d53e21af3d7450313f97a1a1e6506000040ff421f8503008a9ac636022d53e319f4193301b4028f4a21aa81897338b5df6b03940b233a12c6b337e1696b022d5696215a90d0ca0313f97b411b2e51a2b6e9dc56000f107d490193f24bd6d411000003940b24490121ca814a8dfd200002649cc51176240283205e4202fb6929df376a51000041002901645ec4560052ebc34a4ef98ffa61d1d121d40283205f4102fbe64019dd653503940b2549076b9d4ab2674d200002649cc611762401b402904a21aa81897338b5df6b01b40291390d74cd588d200000d1e4784902cd764c1a0ed7dc56002fc8fc4270f5f9d3e35fef0d0b835c84410142001cdc61ef77028320604102fbe64019dd65350052ebc42156e6e06c0b835c85410142001cdc61ef77000f77863a16148d3144922e022d53e6215b6684f70394b1ec22047eaa720030075f31144b3e47de000394b1ed490649d2c967d95000000052ebc54a4ef98ffa61d1d121d4000041012901b38092c800d1e4794902b247fa7757ee0000000f7787490193f24bd6d411000000d1e47a22030176dd0b835c86320f30e478b53502649cc73a11d560af29644a003007604270f5f9d3e35fef0d08d99f3919aee69a01aea5494207a8c34d91966e190c527b021140000b835c89320f30e478b5350b835c8a410142001cdc61ef77029b1adee9b2000000000000001292226e1c5d9fffffffffffffffffffffff34d2a002a03ab89910317b59202b2d6e21d33d0732911ead5ea68a02a03ab96107a49cf3be6f948543b4ccc60542e2b6421cb6f85fd2e7522305439e7c7972986ad9ec2171280133565bc1183405439e7e39aad11b19a6df110197db7a030300f320c9d80000000000002671bf58283800000000001ec0b7646c9f9b0300f322ba0217907aa9c463ffffffffffffffffeb0565f44b7d60000304746fba01d6319aea5c25ffffffffffffffffed94ee172678145809f94876b234709d919f80fffffffffffffffffdf222e716e36a6d0c512557b20cee422dc8bdffffffffffffffffff7e54f60e21e13d0030076631144b3e47de00003007674270f5f9d3e35fef0d0052ebc74a4ef98ffa61d1d121d40052ebc82156e6e06c00233b0e6903ba8b37847ef8c3ffc592bcb30019a07189019b7e6885e057cfd94de9a8455396707900198dc6a9030000000000000002f6e8d8c462d415303fb8cb4f001997f7aa072f4327ff0000000000000000000000000000000000198dc741071f5d130000025c00233b0f1133030019a0723116017a24a3e70446096c09010453eede09010589e04fc101fffffbfffffffffffffffc46bc87d924c806938931d1d40589e051610378b0bd195189daed0d205a0589e05211231c0ac46ae3b901c3d46bb287adcef6cd04830000006aac0d08000023570ac46ae4b901c0f8f4becece38ae623e3f0000006a030c1c0000231f0c00e4a23906d70b1e467ff10bf3e4ac3916d589b158c07c010e0342c9a7000000000000490c067e0d4700000000000a24b91278d9a9010e0344ba03947134637b47ffffffffffffffffdc1a47e1fef8c0000229c717ba03947134637b47ffffffffffffffffdc1a47e1fef8c00003971e0cc9c6000000000001fc8ccc254b6700000000001548988fb4860303971e0ec275b37816b749b66affffffffffffffffb3afd9060c1a60000bd60722c201c89324ecbf867bfffffffffffffffffed7f934dae423db03a3cc8fc2322b1eaac33cc0cfffffffffffffffffdf78fc98ec308bfa03c5d197c241bfc647074d6f20ffffffffffffffffd55ee3384505b02c01bba0fbe943000000000000004f9c26095777f1fffffffffffffffffffffcd25474021967c5990613ad4cd34479a71cf4baa64d89aa86b6ea11021967c66102e2b4482a1b2ee4eb879bf8038dbb085103e8523f4cf16f1ed2dd0b625d544a0649d2c967d950000003971e2e4275b37816b749b66b0c67dee11305f003940b8649076b9d4ab2674d20000c67edc25103e8523f4cf16f1ed2dd0c67e2224a0121ca814a8dfd20000b3335bc030548bc6daa160000000000000015fe6b958a1def7b2cf821ff360548c0a861042836d62935a138f80f16550548c0aa11795d0548bc6f3a01152fa1b7813a0548bc70a937b364758d0e8b5f112f00000000273e350000000d08d000b56b0a0fd0cafba3ad4f70e7d61fc008d000b683ee58ddd1b8c2f74c59836726c83e71df08d000b7000100035ecc00000000000000000c69eedb60a6ee1c1836839a00000a2e57af1808c13d946bf79ddf75276a1261d2e8a4010308c13d958a58f4196928a34ef1e38bf9eb6e563960a908c13d9600010064a2c900000000000000013ccd5ffdf7f0100da967dad4000132804b730c09055b19720223674047a2a8812fc88d18197409055b1a8a544052657586fb99171be9ecfa6ba5eeff09055b1b0001006b52a80000000000000001498433241b4acccd421a55860001470d47ecae08e377ed7201d1dcd6d55189793a0ee38e0ec608e377ee8a4885c9fc86ada7510bef25bf71909cfd3508e377ef0001007a995f00000000000000015503947fe177358efb0cb881000176413e64a6090324767201e00c46e709866855c283a59175090324778a4aa365bf20a7829253d7578b234b063f6b09032478000100776396000000000000000153d079eca60d3516f8003da200016c518e770208b722047201b3e0ce972f08a410243fda658d08b722058a4416ffa7197831ba6277e542ecea53978308b722060001007f46ff00000000000000015883aee435fa37cc54002db9000184b7a9f43a08e3886a7201d0385e2708b91b559228c33a9b08e3886b8a485fbde0216f13213f417922fee6b9d68708e3886c0001007acf02000000000000000155f77a5323ed060f907a6318000176e312076308bb0e1f72021c48d97beb9eeadbad85fcec3408bb0e208a534efab3eddd71418465305ccb9ceea7dd08bb0e210001006c582300000000000000014c0467b830910343bd1c33f100014a307ee20e0969582a720225910be4fb3f4c8c90bba311ca0969582b8a5489cf1ee3eb5b7f2dab62b0361a6423d70969582c0001006b1766000000000000000148fecd25bc194a292d88d184000146576fdc5402db89f8410591d35e956cb20002db89f91a24665e02db89fa094802db8c965105fb1c190a119235e19002db8c97292715d230a902db8c984119092b04800c5cf00a6db00609010a6de0d709010aa19645030313f9b01a1e65060313f9b1411b2e51a2b6e9dc560000410431011321bd066c000041052901645ec45600004106421f8503008a9ac63600004107310174f70cbeee001a672829109c5d155b09a2bbf70303f40749419f325f4a4e0a91d90045800f39cccbbc3146ce8f0776079a429fff2b067f516068077603f30308086d3e09f108086d3f09eb08086d406a7996fa92b74084cfbc0180000008086d41f10835000010ec000000f0ffffffffffffff1452f91da725adde013a80000008086d42692e8af4824003cf2d4fcc00000008086d43f1173600002a78000000f10000000000000024f02b92d0e798e0114f40000008d02cf9096e08d02cfa292554ab87ef09093f3811414a09096aaa1113960903f1c21a91ae9e0c680873030bc36da2030c0bcf6a0a010c680996030bc36da403034a0510391719c7cf8f2a0408056dc803005484943a0d22f86191843d078be0f4a205d500a31e181944dac576d7972bbb392118112d04b5893e421a2aed139a1a95d80c67f8a0c904da8c9cf14366ebf8fffffffffffffffffff429dd2cc03f540c23c0a8d10553ccb59180a60ba9e700000000000000000005ab45276bb00904ccd0ccc151b1e495bd8db569000000000000000000000000000000000c67c2cfc908d4a393227c5018f2ffffffffffffffffe5dad340290462b60c67e53a22dbab80ad0591735a09760591735b29280593239805ba423c2101ca15e405ba70301129320597c1083209b6307c9ef40c6806d71b1beb8c08cddc7722026ab6f008cddc7839382f51fa27dfab08cddc79093308cdde5231116faa2d3eba08cdde5351019551bbe22fa1be472c08ce49ab210132320508ce49ac397510d063daac210a854fb3c93600000000002a21baf2469b0a0000000000021560470250760a854fb5caf3baacc5ef4619951effffffffffffff0c52eb6bea66efa0000aed1a22ca056050d42eed289e11fffffffffffffffa9ffbf025b034674e0c5e5208c2164927047cf102a3ffffffffffffffffe9b817330b3535660afaec0fca03760813504663a0cafffffffffffffffc8a29584733bc29b60a854fb9caeace0ab76b959c539effffffffffffff153f0e0c4a77c9d999034a051ad10456fffffffffffb1e59cdfadb8c00000000000566bd9b48d92d034a051cc14c5daf3ddfd8afffffffffffffffffffb3afd83bf3f024640366358eb9349d903cb1fbb7ffffffffffffffffffcb6bc2115c695f03686a39c102aabbc81b6feccbfffffffffffffffffd55bd2c5c15dc930356e057c14957ae340c0a9462ffffffffffffffffb6b55025dc36c2f2034a0524b926a7b17bac3315ffffffffffffffffffd95f27aa471b7e076d04484a01f9e8e7ffc272bce30458953c090103d01fd809010a6c2ff10901097c00680901054140020901052e746509010548532c090103bb90c00901070018080901056c5a8d090104cda0511a0bce710c5f58d9d103919a5f7f294feaedd3ffffffffffffffffffffffffffe19f250b8eab2b09010b9516e709010394b2b222047eaa720394b2b3490649d2c967d950000005865bd0222791efee05865bd12127939b3705865bd20933058bc6fe310bd3b219ac7b058bc6ff310ce03235a917058bcc2a292e761055cd00afaeac09090087fe09c10100000f000000000000000ed816fc6e728422594141ea35008825f9810b6f402e8d0e272e8fb5d3bd03b809d60852cbbeb91faeea7320a15c982557889e0000018f8a41950000844f04d0eac819b4ad700b4cee5cd9015f85daf9d2acb27b5c9b00000000000000000000000000b4ad700bc3a9e90303fa3823e9b6ffffffffffffff40caad84bf5e1e00000000000000000000083faf3f03fa55849910a46349a418ab5b0a0224d1e45227f5c8b91703fa55856107dc4d22178d82fa15c449ac0ba762f2090106bb6a91790e6ea4e5039aa3aae12cb7bb080d0106ba30c589b700000000000000000000000000000000036e1bd8c980ffffffffffffa430d578ee330000000000048a400bccaae0036e1bdab91b0028e44afffffffffffffffffffef134a81ab02db3b40c3496a6b124c297188c93fffffffffffffffffe8f536e0937ec42038aff47b913c2ffdfc4550bffffffffffffffff39ce58e97e69b119036e1bdfb90466e443e5afa6ffffffffffffffffd3da3152f4db9069074b456fb902b18229886eb5ffffffffffffffffe4fcca7033b085ee09e8e64809010c6809b709010b8f6f2f09020b9514db090202649d383a11d560af29644a02649d391176240c5ff8cc03028320874102fbe64019dd6535000021684101b9eafba6c4c7180000216f41e9251466e89f46320000217031343ab826e82900002171096b000021d4599d0409205c3e4991d20274000021d5510889add24f258c858ff1000021d631047aedee1eba000024f4212e1850350c556aac4101b9eafba6c4c7180c556aad3101d0af4b8c110c556ab009010c556ab13929c7d10b73563f00fa0672f108f2000000000000000aa87bee537fffffffffffffffffffffffff8b5c8500fa08af99d082926bbdd876495e4d2fb6317dec698f960a00fa08b061623e58cc4f9e902575730d0a03940c773a12c6b337e1696b03940c78490121ca814a8dfd20000bcfa5c9030b1eaeee122d1a0b17c8fa120c8f0a739ef712762402db8a611a24665e0a445449127cba02649d4e117624054bb595113d75078d41db1167af041f1455117cba0c6809bc030ab89d8712105b06c715ac03072eac8011f06a0bcfa280030a7db2bb0305876ee90b720a3e1fdec96700000000000c0435e82b4c5300000000000265646355344d0a3e1fe0ba4079d386312e30fffffffffffffffd79d5c1e1172380000a3e28f4ba37135f7bd40e4bfffffffffffffffdd80ac5edac06bcfc0a47afa7ba078f75e3ec7266ffffffffffffffffb43b2f1620112bc00c3f0b9faa1ecf3267d2fffffffffffffffffffecb3c045ca24c0a528342ba01d6df573e45abffffffffffffffffed9101a146aef4fa0a46cf6809010a4b40d409010c32a294090100ae07a6094700ae07a72911ad41501500b2155a3115220ce302d900b33dd531065703ddb40d00b1e60e3910786ceaf8a97f065117b1390c4611e2770114005b121f030b82a0e009060b7194b509010bcf6882090108b8450709020c07007e09010a6c4f1a09260a5ad61f0901059188b4e95200000000000000000000000000000000000000000000000000000000059188b5e9460000000000000000000000000000000000000000000000000000000005917581e95a0000000000000000000000000000000000000000000000000000000005917585e9760000000000000000000000000000000000000000000000000000000004f2809c4902117006df65270000049e222b3b14473ca46fac5e06c5b9240a0109fd90710309fd90720309fd90730309fd9075030c67ef2a12485009f421e91158ba0c6783500309e1e370210301632b09e1e3714a02bbc099d5fb6d812809e1e37211017209e1e4f829a6e77c559009e1e4f9519a2c2bda491472f295bc09e1e4fa191f8645004608fe22037bfcd3004608ff39514dc947af50000046090009280054801e3102ed2c26c7c80054801f49443c9c50dcbd675c200b835cea410142001cdc61ef770ba5d886030458957a0901052e74bf090103d020310901097c00e409010700185909010a6c303309010548538709010541404d0901000024fae957000000000000002ebbb29f84fa30fffffffffffffffffffffdfe9ad2000026759907e0e6c231b1c5abd756e3cb1f008c1d3f0645000026766103c0b69ad26b7f2f4913e62f00bcf87f0300ae083603002fc9584270f5f9d3e35fef0d0a21c87df1013b000000000000000b74c7f337512bffffffffffff2e0c3ed6a21ee9530a29749d71112bed02928d501af17b6c3597300a29749e8916927103ab543f0b242e7af7ee27264563029b18e8e94effffffffffffffffffffffbd0406000000000000000000000043498202b27be3794ee242faabd1e8658d4bde25c2d48102b27be4794f1dc83b5b375b1fdf53dd62f6fcfd022d576c29d71734a637022d576d215a90d0ca01aea60e4207a8c34d91966e19022d576e3101035c52116f022d7bb829056986fc3e0c5569380304cc98db6939c2c91aa2d5a9b4bf3f4d5c2004c7e178a96d000000000000000000000002e7cc2b9e42ae0f5304c7e638b10ef70093cc33ffffffffffffffffffffffcc245218d904c7e179421453716ffffffbd004cc98dc1903e83c057005b6a265fb552603d446648c1c202229b94f4bc564cdd4057005b77a0235312ac1af335d27a8ac6c8daabf0574bc0ca1387308e6a4b8eef0007353243e832c19a9a544b00574bc0d712e457710d1a7940319e8f5c7ebee057e18c3a13a83aaea30b52b770e72fb39550265b01f34687a057e18c471365c41a707e69641f4fb8b67750c09a2310b71047cc88526c5259f4fcd6d78fcfd08a6213a69dba95cbdace21dd664ba87d30e086b5809999796e90000000000000000000000000000000009a2310c11099908a6213b1101d603f407a7420b9163762a3984a7078d420f410b9163762a3984a70bf2a7450300bc4e240902000021b22137b6a3a4000021b342051164cdb7394731000021b40948000021da3901033b2e7650dd000021db511796012d3ab7bfa14b19000021dc2102dcb2ee09ea37f2030590b470c90100000000000000000000000000000000000000000000000005d04b102105ac35dd05d056503904a42ab2449f7605d04489410b1db9419116800001f466f30300ff318b03026e527b094f026e527c000004a77024a7e000000000008583b00000000000000000000000003bd209222105876f404119d3a06ac812971505876f410b7205876f4211184506420aa349345346dea2241209ad06420aa4198ba5050748c2a9593d93c0053d3ed733008f5a05876f4409010c242a70410e2715c359ff41840c1672bc420211426309d2f7d00b206baa410211426309d2f7d00c35718403072ecb739105593d58e3fc248eb7e4793283971984f2fc072eceab6972ba271dcc5da04324dea3123f072e9af5a90800000000000000000000000436225502d4b96091072eacd3a172a50ddb00000000000000000000000000000000072e9af642014b3169ffffffac072ecb743104322092fe90072eceac095900001d5dc202c90edcfffffffffba6e944ffffffffffffffffffffffa500001d5e5a053d71ffffffffff1628b80003ca1ab20f6600000000000001f9ffffffffffffffffffffffae0003ca1b4a81000000000000001503a7fe72e934fffffffffffffffffffffc95d26500000000000000000000037248d803b9330279350407badbb2daa07a29d924a0382f03b933037934fbf893929cf7679fd8fcb61df57309e8418509010c680b1f0901064160b5424563918244f4000006416bde424563918244f40000062baeec414563918244f4000004eb76400a010c519e2b1a0494510c52df89030c519e2c030c5973b8030c66990d1238740c519e2e0a010c519e2f030bbc8f632a03976cec230c5e0217030b2002f509010574fd36e94d00000000000000072d8e0e5b01abffffffffffffffffffffffb03ab105752c659906e86ea21b569a723471ac9a2a2e25674ac5e205752c6661035a49b78f46345a266966e807712d2e190f424007712d2f4a01f9e8e7ffc272bce307af4b71410b9163762a3984a70872e8451990b19f07af4b72326eb7f3ec9be0078d42321167af07711e8b3a92fa674c665b28078d42333a0fc0c6843deddf0872e8473a0ced5c4f71eb1207711e8c2106821e2101b72c92a9380d19109f0ffffffffffffffffffffffffd9791180ba6849609010badd37509010b01b8484904e6167b4922d798500afacf5c31048c273950000c680b244107dcad3d7bcb91cc0b97e0f46202af45109a024eb79eff9cf40b8e83d86102af45109a024eb79eff9cf40c680b27030c680b2809010b97e0f63225f9791ce2ed0c680b29091d0bdcf9ed190d59a80b8e83d93125f9791ce2ed09b685b13202499633c34809b42bdc29265627c61409a8729e2942a2fd6d8209a9a4353202412140cd8609a9a437298778adbc3709a861b822015f4fd409a861b922015d537b09a861ba095109b685b229845e3d01ab09a4fa08391fd7ab9a6e37eb09a4fa09391fd7ab9a6e37eb09a4fa0a095b09a4fb7c095109a4fa0c095b09a861bb1901fc59090b8eb80307b23db809010b8e905e09010b8f887e0901", - "expected_outputs": { - "kzg_commitment": "b3155be333f11f4ac0655292d3f7baf6b7c50fe15f2ae7509d8ad93a98c32adb82899f72b32fbcf77898a3e4a082c4b0", - "opening_point": "00000000000000000000000000000000b65574fcce3da68c6d7199110f1b5358", - "opening_value": "2e9ea79020c009e69d1d480ea0726dec69f43ab688f08c7852270dccbc7456c4", - "opening_proof": "ac2355d9a024ee3d77b8577c299b5384adcea34c205231bebb08ffa9fb5e33f2327f11ecde4fe46e0aaadce36b31c87d", - "versioned_hash": "01a70111f6ade45fdff137f00119dd45284d80e1af6bdad336419c07cec93f41", - "blob_proof": "a370bebe32974312600170a180d02aa0f44394b96221f95af357e696fdbc11ef917fe3b643ffc62c98d333385ab25b43", - "pubdata_commitment": "b65574fcce3da68c6d7199110f1b53582e9ea79020c009e69d1d480ea0726dec69f43ab688f08c7852270dccbc7456c4b3155be333f11f4ac0655292d3f7baf6b7c50fe15f2ae7509d8ad93a98c32adb82899f72b32fbcf77898a3e4a082c4b0ac2355d9a024ee3d77b8577c299b5384adcea34c205231bebb08ffa9fb5e33f2327f11ecde4fe46e0aaadce36b31c87d" - } -} diff --git a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/tests/mod.rs b/core/lib/l1_contract_interface/src/i_executor/commit/kzg/tests/mod.rs deleted file mode 100644 index fb38bea90537..000000000000 --- a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/tests/mod.rs +++ /dev/null @@ -1,135 +0,0 @@ -//! Tests for KZG commitments. - -use kzg::{ - boojum::pairing::{bls12_381::G1Compressed, EncodedPoint}, - verify_kzg_proof, verify_proof_poly, - zkevm_circuits::eip_4844::ethereum_4844_data_into_zksync_pubdata, -}; -use serde::{Deserialize, Serialize}; - -use super::*; - -const KZG_TEST_JSON: &str = include_str!("kzg_test_0.json"); - -#[serde_with::serde_as] -#[derive(Debug, Serialize, Deserialize)] -struct ExpectedOutputs { - #[serde_as(as = "serde_with::hex::Hex")] - versioned_hash: Vec, - #[serde_as(as = "serde_with::hex::Hex")] - kzg_commitment: Vec, - #[serde_as(as = "serde_with::hex::Hex")] - opening_point: Vec, - #[serde_as(as = "serde_with::hex::Hex")] - opening_value: Vec, - #[serde_as(as = "serde_with::hex::Hex")] - opening_proof: Vec, - #[serde_as(as = "serde_with::hex::Hex")] - blob_proof: Vec, - #[serde_as(as = "serde_with::hex::Hex")] - pubdata_commitment: Vec, -} - -#[serde_with::serde_as] -#[derive(Debug, Serialize, Deserialize)] -struct KzgTest { - #[serde_as(as = "serde_with::hex::Hex")] - pubdata: Vec, - expected_outputs: ExpectedOutputs, -} - -/// Copy of function from https://github.com/matter-labs/era-zkevm_test_harness/blob/99956050a7705e26e0e5aa0729348896a27846c7/src/kzg/mod.rs#L339 -fn u8_repr_to_fr(bytes: &[u8]) -> Fr { - assert_eq!(bytes.len(), 32); - let mut ret = [0u64; 4]; - - for (i, chunk) in bytes.chunks(8).enumerate() { - let mut repr = [0u8; 8]; - repr.copy_from_slice(chunk); - ret[3 - i] = u64::from_be_bytes(repr); - } - - Fr::from_repr(FrRepr(ret)).unwrap() -} - -fn bytes_to_g1(data: &[u8]) -> G1Affine { - let mut compressed = G1Compressed::empty(); - let v = compressed.as_mut(); - v.copy_from_slice(data); - compressed.into_affine().unwrap() -} - -#[test] -fn kzg_test() { - let kzg_test: KzgTest = serde_json::from_str(KZG_TEST_JSON).unwrap(); - let kzg_info = KzgInfo::new(&kzg_test.pubdata); - - // Verify all the fields were correctly computed - assert_eq!( - hex::encode(kzg_info.kzg_commitment), - hex::encode(kzg_test.expected_outputs.kzg_commitment) - ); - assert_eq!( - hex::encode(kzg_info.opening_point), - hex::encode(kzg_test.expected_outputs.opening_point) - ); - assert_eq!( - hex::encode(kzg_info.opening_value), - hex::encode(kzg_test.expected_outputs.opening_value) - ); - assert_eq!( - hex::encode(kzg_info.opening_proof), - hex::encode(kzg_test.expected_outputs.opening_proof) - ); - assert_eq!( - hex::encode(kzg_info.versioned_hash), - hex::encode(kzg_test.expected_outputs.versioned_hash) - ); - assert_eq!( - hex::encode(kzg_info.blob_proof), - hex::encode(kzg_test.expected_outputs.blob_proof) - ); - - // Verify data we need for blob commitment on L1 returns the correct data - assert_eq!( - hex::encode(kzg_info.to_pubdata_commitment()), - hex::encode(kzg_test.expected_outputs.pubdata_commitment) - ); - - // Verify that the blob, commitment, and proofs are all valid - let blob = ethereum_4844_data_into_zksync_pubdata(&kzg_info.blob); - let mut poly = zksync_pubdata_into_monomial_form_poly(&blob); - fft(&mut poly); - bitreverse(&mut poly); - - let commitment = bytes_to_g1(&kzg_info.kzg_commitment); - let blob_proof = bytes_to_g1(&kzg_info.blob_proof); - - let valid_blob_proof = verify_proof_poly(&KZG_SETTINGS, &poly, &commitment, &blob_proof); - assert!(valid_blob_proof); - - let opening_point = u8_repr_to_fr(&kzg_info.opening_point); - let opening_value = u8_repr_to_fr(&kzg_info.opening_value); - let opening_proof = bytes_to_g1(&kzg_info.opening_proof); - - let valid_opening_proof = verify_kzg_proof( - &KZG_SETTINGS, - &commitment, - &opening_point, - &opening_value, - &opening_proof, - ); - assert!(valid_opening_proof); -} - -#[test] -fn bytes_test() { - let kzg_test: KzgTest = serde_json::from_str(KZG_TEST_JSON).unwrap(); - - let kzg_info = KzgInfo::new(&kzg_test.pubdata); - let encoded_info = kzg_info.to_bytes(); - assert_eq!(KzgInfo::SERIALIZED_SIZE, encoded_info.len()); - - let decoded_kzg_info = KzgInfo::from_slice(&encoded_info); - assert_eq!(kzg_info, decoded_kzg_info); -} diff --git a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/trusted_setup.rs b/core/lib/l1_contract_interface/src/i_executor/commit/kzg/trusted_setup.rs deleted file mode 100644 index 1aecea0fad39..000000000000 --- a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/trusted_setup.rs +++ /dev/null @@ -1,101 +0,0 @@ -use std::{convert::TryInto, iter}; - -use kzg::{ - boojum::pairing::{bls12_381::G2Compressed, EncodedPoint}, - zkevm_circuits::{ - boojum::pairing::{ - bls12_381::{Fr, FrRepr, G1Compressed}, - ff::{Field as _, PrimeField as _}, - CurveAffine, - }, - eip_4844::input::ELEMENTS_PER_4844_BLOCK, - }, - KzgSettings, -}; -use once_cell::sync::Lazy; - -const FIRST_ROOT_OF_UNITY: FrRepr = FrRepr([ - 0xe206da11a5d36306, - 0x0ad1347b378fbf96, - 0xfc3e8acfe0f8245f, - 0x564c0a11a0f704f4, -]); - -fn bit_reverse_slice_indices(array: &mut [T]) { - assert_eq!(array.len(), ELEMENTS_PER_4844_BLOCK); - for idx in 0..ELEMENTS_PER_4844_BLOCK { - let reversed_idx = idx.reverse_bits() >> (usize::BITS - ELEMENTS_PER_4844_BLOCK.ilog2()); - if idx < reversed_idx { - array.swap(idx, reversed_idx); - } - } -} - -pub(super) static KZG_SETTINGS: Lazy = Lazy::new(|| { - // Taken from the C KZG library: https://github.com/ethereum/c-kzg-4844/blob/main/src/trusted_setup.txt - const TRUSTED_SETUP_STR: &str = include_str!("trusted_setup.txt"); - - // Skip 2 first lines (number of G1 and G2 points). - let mut lines = TRUSTED_SETUP_STR.lines().skip(2); - - let first_root_of_unity = - Fr::from_repr(FIRST_ROOT_OF_UNITY).expect("invalid first root of unity"); - let mut roots_of_unity: Box<[_]> = iter::successors(Some(Fr::one()), |prev| { - let mut next = first_root_of_unity; - next.mul_assign(prev); - Some(next) - }) - .take(ELEMENTS_PER_4844_BLOCK) - .collect(); - bit_reverse_slice_indices(&mut roots_of_unity); - - let lagrange_setup = lines.by_ref().take(ELEMENTS_PER_4844_BLOCK).map(|line| { - let mut g1_bytes = [0_u8; 48]; - hex::decode_to_slice(line, &mut g1_bytes).expect("failed decoding G1 point from hex"); - let mut g1 = G1Compressed::empty(); - g1.as_mut().copy_from_slice(&g1_bytes); - g1.into_affine().expect("invalid G1 point") - }); - let mut lagrange_setup: Box<[_]> = lagrange_setup.collect(); - bit_reverse_slice_indices(&mut lagrange_setup); - - // Skip the 0th G2 point. - assert!( - lines.next().is_some(), - "KZG trusted setup doesn't contain G2 points" - ); - - let line = lines - .next() - .expect("KZG trusted setup doesn't contain G2 point #1"); - let mut g2_bytes = [0_u8; 96]; - hex::decode_to_slice(line, &mut g2_bytes).expect("failed decoding G2 point from hex"); - let mut setup_g2_1 = G2Compressed::empty(); - - setup_g2_1.as_mut().copy_from_slice(&g2_bytes); - let setup_g2_1 = setup_g2_1 - .into_affine() - .expect("invalid G2 point #1") - .into_projective(); - - KzgSettings { - roots_of_unity_brp: roots_of_unity.try_into().unwrap(), - setup_g2_1, - lagrange_setup_brp: lagrange_setup.try_into().unwrap(), - } -}); - -#[cfg(test)] -mod tests { - use super::*; - - #[test] - fn kzg_roots_of_unity_are_correct() { - let mut value = Fr::from_repr(FIRST_ROOT_OF_UNITY).unwrap(); - for _ in 0..ELEMENTS_PER_4844_BLOCK.ilog2() { - assert_ne!(value, Fr::one()); - value.mul_assign(&value.clone()); - } - assert_eq!(value, Fr::one()); - } -} diff --git a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/trusted_setup.txt b/core/lib/l1_contract_interface/src/i_executor/commit/kzg/trusted_setup.txt deleted file mode 100644 index d2519656fb2a..000000000000 --- a/core/lib/l1_contract_interface/src/i_executor/commit/kzg/trusted_setup.txt +++ /dev/null @@ -1,4163 +0,0 @@ -4096 -65 -a0413c0dcafec6dbc9f47d66785cf1e8c981044f7d13cfe3e4fcbb71b5408dfde6312493cb3c1d30516cb3ca88c03654 -8b997fb25730d661918371bb41f2a6e899cac23f04fc5365800b75433c0a953250e15e7a98fb5ca5cc56a8cd34c20c57 -83302852db89424d5699f3f157e79e91dc1380f8d5895c5a772bb4ea3a5928e7c26c07db6775203ce33e62a114adaa99 -a759c48b7e4a685e735c01e5aa6ef9c248705001f470f9ad856cd87806983e917a8742a3bd5ee27db8d76080269b7c83 -967f8dc45ebc3be14c8705f43249a30ff48e96205fb02ae28daeab47b72eb3f45df0625928582aa1eb4368381c33e127 -a418eb1e9fb84cb32b370610f56f3cb470706a40ac5a47c411c464299c45c91f25b63ae3fcd623172aa0f273c0526c13 -8f44e3f0387293bc7931e978165abbaed08f53acd72a0a23ac85f6da0091196b886233bcee5b4a194db02f3d5a9b3f78 -97173434b336be73c89412a6d70d416e170ea355bf1956c32d464090b107c090ef2d4e1a467a5632fbc332eeb679bf2d -a24052ad8d55ad04bc5d951f78e14213435681594110fd18173482609d5019105b8045182d53ffce4fc29fc8810516c1 -b950768136b260277590b5bec3f56bbc2f7a8bc383d44ce8600e85bf8cf19f479898bcc999d96dfbd2001ede01d94949 -92ab8077871037bd3b57b95cbb9fb10eb11efde9191690dcac655356986fd02841d8fdb25396faa0feadfe3f50baf56d -a79b096dff98038ac30f91112dd14b78f8ad428268af36d20c292e2b3b6d9ed4fb28480bb04e465071cc67d05786b6d1 -b9ff71461328f370ce68bf591aa7fb13027044f42a575517f3319e2be4aa4843fa281e756d0aa5645428d6dfa857cef2 -8d765808c00b3543ff182e2d159c38ae174b12d1314da88ea08e13bd9d1c37184cb515e6bf6420531b5d41767987d7ce -b8c9a837d20c3b53e6f578e4a257bb7ef8fc43178614ec2a154915b267ad2be135981d01ed2ee1b5fbd9d9bb27f0800a -a9773d92cf23f65f98ef68f6cf95c72b53d0683af2f9bf886bb9036e4a38184b1131b26fd24397910b494fbef856f3aa -b41ebe38962d112da4a01bf101cb248d808fbd50aaf749fc7c151cf332032eb3e3bdbd716db899724b734d392f26c412 -90fbb030167fb47dcc13d604a726c0339418567c1d287d1d87423fa0cb92eec3455fbb46bcbe2e697144a2d3972142e4 -b11d298bd167464b35fb923520d14832bd9ed50ed841bf6d7618424fd6f3699190af21759e351b89142d355952149da1 -8bc36066f69dc89f7c4d1e58d67497675050c6aa002244cebd9fc957ec5e364c46bab4735ea3db02b73b3ca43c96e019 -ab7ab92c5d4d773068e485aa5831941ebd63db7118674ca38089635f3b4186833af2455a6fb9ed2b745df53b3ce96727 -af191ca3089892cb943cd97cf11a51f38e38bd9be50844a4e8da99f27e305e876f9ed4ab0628e8ae3939066b7d34a15f -a3204c1747feabc2c11339a542195e7cb6628fd3964f846e71e2e3f2d6bb379a5e51700682ea1844eba12756adb13216 -903a29883846b7c50c15968b20e30c471aeac07b872c40a4d19eb1a42da18b649d5bbfde4b4cf6225d215a461b0deb6d -8e6e9c15ffbf1e16e5865a5fef7ed751dc81957a9757b535cb38b649e1098cda25d42381dc4f776778573cdf90c3e6e0 -a8f6dd26100b512a8c96c52e00715c4b2cb9ac457f17aed8ffe1cf1ea524068fe5a1ddf218149845fc1417b789ecfc98 -a5b0ffc819451ea639cfd1c18cbc9365cc79368d3b2e736c0ae54eba2f0801e6eb0ee14a5f373f4a70ca463bdb696c09 -879f91ccd56a1b9736fbfd20d8747354da743fb121f0e308a0d298ff0d9344431890e41da66b5009af3f442c636b4f43 -81bf3a2d9755e206b515a508ac4d1109bf933c282a46a4ae4a1b4cb4a94e1d23642fad6bd452428845afa155742ade7e -8de778d4742f945df40004964e165592f9c6b1946263adcdd5a88b00244bda46c7bb49098c8eb6b3d97a0dd46148a8ca -b7a57b21d13121907ee28c5c1f80ee2e3e83a3135a8101e933cf57171209a96173ff5037f5af606e9fd6d066de6ed693 -b0877d1963fd9200414a38753dffd9f23a10eb3198912790d7eddbc9f6b477019d52ddd4ebdcb9f60818db076938a5a9 -88da2d7a6611bc16adc55fc1c377480c828aba4496c645e3efe0e1a67f333c05a0307f7f1d2df8ac013602c655c6e209 -95719eb02e8a9dede1a888c656a778b1c69b7716fbe3d1538fe8afd4a1bc972183c7d32aa7d6073376f7701df80116d8 -8e8a1ca971f2444b35af3376e85dccda3abb8e8e11d095d0a4c37628dfe5d3e043a377c3de68289ef142e4308e9941a0 -b720caaff02f6d798ac84c4f527203e823ff685869e3943c979e388e1c34c3f77f5c242c6daa7e3b30e511aab917b866 -86040d55809afeec10e315d1ad950d269d37cfee8c144cd8dd4126459e3b15a53b3e68df5981df3c2346d23c7b4baaf4 -82d8cabf13ab853db0377504f0aec00dba3a5cd3119787e8ad378ddf2c40b022ecfc67c642b7acc8c1e3dd03ab50993e -b8d873927936719d2484cd03a6687d65697e17dcf4f0d5aed6f5e4750f52ef2133d4645894e7ebfc4ef6ce6788d404c8 -b1235594dbb15b674a419ff2b2deb644ad2a93791ca05af402823f87114483d6aa1689b7a9bea0f547ad12fe270e4344 -a53fda86571b0651f5affb74312551a082fffc0385cfd24c1d779985b72a5b1cf7c78b42b4f7e51e77055f8e5e915b00 -b579adcfd9c6ef916a5a999e77a0cb21d378c4ea67e13b7c58709d5da23a56c2e54218691fc4ac39a4a3d74f88cc31f7 -ab79e584011713e8a2f583e483a91a0c2a40771b77d91475825b5acbea82db4262132901cb3e4a108c46d7c9ee217a4e -a0fe58ea9eb982d7654c8aaf9366230578fc1362f6faae0594f8b9e659bcb405dff4aac0c7888bbe07f614ecf0d800a6 -867e50e74281f28ecd4925560e2e7a6f8911b135557b688254623acce0dbc41e23ac3e706a184a45d54c586edc416eb0 -89f81b61adda20ea9d0b387a36d0ab073dc7c7cbff518501962038be19867042f11fcc7ff78096e5d3b68c6d8dc04d9b -a58ee91bb556d43cf01f1398c5811f76dc0f11efdd569eed9ef178b3b0715e122060ec8f945b4dbf6eebfa2b90af6fa6 -ac460be540f4c840def2eef19fc754a9af34608d107cbadb53334cf194cc91138d53b9538fcd0ec970b5d4aa455b224a -b09b91f929de52c09d48ca0893be6eb44e2f5210a6c394689dc1f7729d4be4e11d0474b178e80cea8c2ac0d081f0e811 -8d37a442a76b06a02a4e64c2504aea72c8b9b020ab7bcc94580fe2b9603c7c50d7b1e9d70d2a7daea19c68667e8f8c31 -a9838d4c4e3f3a0075a952cf7dd623307ec633fcc81a7cf9e52e66c31780de33dbb3d74c320dc7f0a4b72f7a49949515 -a44766b6251af458fe4f5f9ed1e02950f35703520b8656f09fc42d9a2d38a700c11a7c8a0436ac2e5e9f053d0bb8ff91 -ad78d9481c840f5202546bea0d13c776826feb8b1b7c72e83d99a947622f0bf38a4208551c4c41beb1270d7792075457 -b619ffa8733b470039451e224b777845021e8dc1125f247a4ff2476cc774657d0ff9c5279da841fc1236047de9d81c60 -af760b0a30a1d6af3bc5cd6686f396bd41779aeeb6e0d70a09349bd5da17ca2e7965afc5c8ec22744198fbe3f02fb331 -a0cc209abdb768b589fcb7b376b6e1cac07743288c95a1cf1a0354b47f0cf91fca78a75c1fcafa6f5926d6c379116608 -864add673c89c41c754eeb3cd8dcff5cdde1d739fce65c30e474a082bb5d813cba6412e61154ce88fdb6c12c5d9be35b -b091443b0ce279327dc37cb484e9a5b69b257a714ce21895d67539172f95ffa326903747b64a3649e99aea7bb10d03f7 -a8c452b8c4ca8e0a61942a8e08e28f17fb0ef4c5b018b4e6d1a64038280afa2bf1169202f05f14af24a06ca72f448ccd -a23c24721d18bc48d5dcf70effcbef89a7ae24e67158d70ae1d8169ee75d9a051d34b14e9cf06488bac324fe58549f26 -92a730e30eb5f3231feb85f6720489dbb1afd42c43f05a1610c6b3c67bb949ec8fde507e924498f4ffc646f7b07d9123 -8dbe5abf4031ec9ba6bb06d1a47dd1121fb9e03b652804069250967fd5e9577d0039e233441b7f837a7c9d67ba18c28e -aa456bcfef6a21bb88181482b279df260297b3778e84594ebddbdf337e85d9e3d46ca1d0b516622fb0b103df8ec519b7 -a3b31ae621bd210a2b767e0e6f22eb28fe3c4943498a7e91753225426168b9a26da0e02f1dc5264da53a5ad240d9f51b -aa8d66857127e6e71874ce2202923385a7d2818b84cb73a6c42d71afe70972a70c6bdd2aad1a6e8c5e4ca728382a8ea8 -ac7e8e7a82f439127a5e40558d90d17990f8229852d21c13d753c2e97facf077cf59582b603984c3dd3faebd80aff4f5 -93a8bcf4159f455d1baa73d2ef2450dcd4100420de84169bbe28b8b7a5d1746273f870091a87a057e834f754f34204b1 -89d0ebb287c3613cdcae7f5acc43f17f09c0213fc40c074660120b755d664109ffb9902ed981ede79e018ddb0c845698 -a87ccbfad431406aadbee878d9cf7d91b13649d5f7e19938b7dfd32645a43b114eef64ff3a13201398bd9b0337832e5a -833c51d0d0048f70c3eefb4e70e4ff66d0809c41838e8d2c21c288dd3ae9d9dfaf26d1742bf4976dab83a2b381677011 -8bcd6b1c3b02fffead432e8b1680bad0a1ac5a712d4225e220690ee18df3e7406e2769e1f309e2e803b850bc96f0e768 -b61e3dbd88aaf4ff1401521781e2eea9ef8b66d1fac5387c83b1da9e65c2aa2a56c262dea9eceeb4ad86c90211672db0 -866d3090db944ecf190dd0651abf67659caafd31ae861bab9992c1e3915cb0952da7c561cc7e203560a610f48fae633b -a5e8971543c14274a8dc892b0be188c1b4fbc75c692ed29f166e0ea80874bc5520c2791342b7c1d2fb5dd454b03b8a5b -8f2f9fc50471bae9ea87487ebd1bc8576ef844cc42d606af5c4c0969670fdf2189afd643e4de3145864e7773d215f37f -b1bb0f2527db6d51f42b9224383c0f96048bbc03d469bf01fe1383173ef8b1cc9455d9dd8ba04d46057f46949bfc92b5 -aa7c99d906b4d7922296cfe2520473fc50137c03d68b7865c5bfb8adbc316b1034310ec4b5670c47295f4a80fb8d61e9 -a5d1da4d6aba555919df44cbaa8ff79378a1c9e2cfdfbf9d39c63a4a00f284c5a5724e28ecbc2d9dba27fe4ee5018bd5 -a8db53224f70af4d991b9aae4ffe92d2aa5b618ad9137784b55843e9f16cefbfd25ada355d308e9bbf55f6d2f7976fb3 -b6536c4232bb20e22af1a8bb12de76d5fec2ad9a3b48af1f38fa67e0f8504ef60f305a73d19385095bb6a9603fe29889 -87f7e371a1817a63d6838a8cf4ab3a8473d19ce0d4f40fd013c03d5ddd5f4985df2956531cc9f187928ef54c68f4f9a9 -ae13530b1dbc5e4dced9d909ea61286ec09e25c12f37a1ed2f309b0eb99863d236c3b25ed3484acc8c076ad2fa8cd430 -98928d850247c6f7606190e687d5c94a627550198dbdbea0161ef9515eacdb1a0f195cae3bb293112179082daccf8b35 -918528bb8e6a055ad4db6230d3a405e9e55866da15c4721f5ddd1f1f37962d4904aad7a419218fe6d906fe191a991806 -b71e31a06afe065773dd3f4a6e9ef81c3292e27a3b7fdfdd452d03e05af3b6dd654c355f7516b2a93553360c6681a73a -8870b83ab78a98820866f91ac643af9f3ff792a2b7fda34185a9456a63abdce42bfe8ad4dc67f08a6392f250d4062df4 -91eea1b668e52f7a7a5087fabf1cab803b0316f78d9fff469fbfde2162f660c250e4336a9eea4cb0450bd30ac067bc8b -8b74990946de7b72a92147ceac1bd9d55999a8b576e8df68639e40ed5dc2062cfcd727903133de482b6dca19d0aaed82 -8ebad537fece090ebbab662bdf2618e21ca30cf6329c50935e8346d1217dcbe3c1fe1ea28efca369c6003ce0a94703c1 -a8640479556fb59ebd1c40c5f368fbd960932fdbb782665e4a0e24e2bdb598fc0164ce8c0726d7759cfc59e60a62e182 -a9a52a6bf98ee4d749f6d38be2c60a6d54b64d5cbe4e67266633dc096cf28c97fe998596707d31968cbe2064b72256bf -847953c48a4ce6032780e9b39d0ed4384e0be202c2bbe2dfda3910f5d87aa5cd3c2ffbfcfae4dddce16d6ab657599b95 -b6f6e1485d3ec2a06abaecd23028b200b2e4a0096c16144d07403e1720ff8f9ba9d919016b5eb8dc5103880a7a77a1d3 -98dfc2065b1622f596dbe27131ea60bef7a193b12922cecb27f8c571404f483014f8014572e86ae2e341ab738e4887ef -acb0d205566bacc87bbe2e25d10793f63f7a1f27fd9e58f4f653ceae3ffeba511eaf658e068fad289eeb28f9edbeb35b -ae4411ed5b263673cee894c11fe4abc72a4bf642d94022a5c0f3369380fcdfc1c21e277f2902972252503f91ada3029a -ac4a7a27ba390a75d0a247d93d4a8ef1f0485f8d373a4af4e1139369ec274b91b3464d9738eeaceb19cd6f509e2f8262 -87379c3bf231fdafcf6472a79e9e55a938d851d4dd662ab6e0d95fd47a478ed99e2ad1e6e39be3c0fc4f6d996a7dd833 -81316904b035a8bcc2041199a789a2e6879486ba9fddcba0a82c745cc8dd8374a39e523b91792170cd30be7aa3005b85 -b8206809c6cd027ed019f472581b45f7e12288f89047928ba32b4856b6560ad30395830d71e5e30c556f6f182b1fe690 -88d76c028f534a62e019b4a52967bb8642ede6becfa3807be68fdd36d366fc84a4ac8dc176e80a68bc59eb62caf5dff9 -8c3b8be685b0f8aad131ee7544d0e12f223f08a6f8edaf464b385ac644e0ddc9eff7cc7cb5c1b50ab5d71ea0f41d2213 -8d91410e004f76c50fdc05784157b4d839cb5090022c629c7c97a5e0c3536eeafee17a527b54b1165c3cd81774bb54ce -b25c2863bc28ec5281ce800ddf91a7e1a53f4c6d5da1e6c86ef4616e93bcf55ed49e297216d01379f5c6e7b3c1e46728 -865f7b09ac3ca03f20be90c48f6975dd2588838c2536c7a3532a6aa5187ed0b709cd03d91ff4048061c10d0aa72b69ce -b3f7477c90c11596eb4f8bbf34adbcb832638c4ff3cdd090d4d477ee50472ac9ddaf5be9ad7eca3f148960d362bbd098 -8db35fd53fca04faecd1c76a8227160b3ab46ac1af070f2492445a19d8ff7c25bbaef6c9fa0c8c088444561e9f7e4eb2 -a478b6e9d058a2e01d2fc053b739092e113c23a6a2770a16afbef044a3709a9e32f425ace9ba7981325f02667c3f9609 -98caa6bd38916c08cf221722a675a4f7577f33452623de801d2b3429595f988090907a7e99960fff7c076d6d8e877b31 -b79aaaacefc49c3038a14d2ac468cfec8c2161e88bdae91798d63552cdbe39e0e02f9225717436b9b8a40a022c633c6e -845a31006c680ee6a0cc41d3dc6c0c95d833fcf426f2e7c573fa15b2c4c641fbd6fe5ebb0e23720cc3467d6ee1d80dc4 -a1bc287e272cf8b74dbf6405b3a5190883195806aa351f1dc8e525aa342283f0a35ff687e3b434324dedee74946dd185 -a4fd2dc8db75d3783a020856e2b3aa266dc6926e84f5c491ef739a3bddd46dc8e9e0fc1177937839ef1b18d062ffbb9e -acbf0d3c697f57c202bb8c5dc4f3fc341b8fc509a455d44bd86acc67cad2a04495d5537bcd3e98680185e8aa286f2587 -a5caf423a917352e1b8e844f5968a6da4fdeae467d10c6f4bbd82b5eea46a660b82d2f5440d3641c717b2c3c9ed0be52 -8a39d763c08b926599ab1233219c49c825368fad14d9afc7c0c039224d37c00d8743293fd21645bf0b91eaf579a99867 -b2b53a496def0ba06e80b28f36530fbe0fb5d70a601a2f10722e59abee529369c1ae8fd0f2db9184dd4a2519bb832d94 -a73980fcef053f1b60ebbb5d78ba6332a475e0b96a0c724741a3abf3b59dd344772527f07203cf4c9cb5155ebed81fa0 -a070d20acce42518ece322c9db096f16aed620303a39d8d5735a0df6e70fbeceb940e8d9f5cc38f3314b2240394ec47b -a50cf591f522f19ca337b73089557f75929d9f645f3e57d4f241e14cdd1ea3fb48d84bcf05e4f0377afbb789fbdb5d20 -82a5ffce451096aca8eeb0cd2ae9d83db3ed76da3f531a80d9a70a346359bf05d74863ce6a7c848522b526156a5e20cd -88e0e84d358cbb93755a906f329db1537c3894845f32b9b0b691c29cbb455373d9452fadd1e77e20a623f6eaf624de6f -aa07ac7b84a6d6838826e0b9e350d8ec75e398a52e9824e6b0da6ae4010e5943fec4f00239e96433f291fef9d1d1e609 -ac8887bf39366034bc63f6cc5db0c26fd27307cbc3d6cce47894a8a019c22dd51322fb5096edc018227edfafc053a8f6 -b7d26c26c5b33f77422191dca94977588ab1d4b9ce7d0e19c4a3b4cd1c25211b78c328dbf81e755e78cd7d1d622ad23e -99a676d5af49f0ba44047009298d8474cabf2d5bca1a76ba21eff7ee3c4691a102fdefea27bc948ccad8894a658abd02 -b0d09a91909ab3620c183bdf1d53d43d39eb750dc7a722c661c3de3a1a5d383ad221f71bae374f8a71867505958a3f76 -84681a883de8e4b93d68ac10e91899c2bbb815ce2de74bb48a11a6113b2a3f4df8aceabda1f5f67bc5aacac8c9da7221 -9470259957780fa9b43521fab3644f555f5343281c72582b56d2efd11991d897b3b481cafa48681c5aeb80c9663b68f7 -ab1b29f7ece686e6fa968a4815da1d64f3579fed3bc92e1f3e51cd13a3c076b6cf695ed269d373300a62463dc98a4234 -8ab415bfcd5f1061f7687597024c96dd9c7cb4942b5989379a7a3b5742f7d394337886317659cbeacaf030234a24f972 -b9b524aad924f9acc63d002d617488f31b0016e0f0548f050cada285ce7491b74a125621638f19e9c96eabb091d945be -8c4c373e79415061837dd0def4f28a2d5d74d21cb13a76c9049ad678ca40228405ab0c3941df49249847ecdefc1a5b78 -a8edf4710b5ab2929d3db6c1c0e3e242261bbaa8bcec56908ddadd7d2dad2dca9d6eb9de630b960b122ebeea41040421 -8d66bb3b50b9df8f373163629f9221b3d4b6980a05ea81dc3741bfe9519cf3ebba7ab98e98390bae475e8ede5821bd5c -8d3c21bae7f0cfb97c56952bb22084b58e7bb718890935b73103f33adf5e4d99cd262f929c6eeab96209814f0dbae50a -a5c66cfab3d9ebf733c4af24bebc97070e7989fe3c73e79ac85fb0e4d40ae44fb571e0fad4ad72560e13ed453900d14f -9362e6b50b43dbefbc3254471372297b5dcce809cd3b60bf74a1268ab68bdb50e46e462cbd78f0d6c056330e982846af -854630d08e3f0243d570cc2e856234cb4c1a158d9c1883bf028a76525aaa34be897fe918d5f6da9764a3735fa9ebd24a -8c7d246985469ff252c3f4df6c7c9196fc79f05c1c66a609d84725c78001d0837c7a7049394ba5cf7e863e2d58af8417 -ae050271e01b528925302e71903f785b782f7bf4e4e7a7f537140219bc352dc7540c657ed03d3a297ad36798ecdb98cd -8d2ae9179fcf2b0c69850554580b52c1f4a5bd865af5f3028f222f4acad9c1ad69a8ef6c7dc7b03715ee5c506b74325e -b8ef8de6ce6369a8851cd36db0ccf00a85077e816c14c4e601f533330af9e3acf0743a95d28962ed8bfcfc2520ef3cfe -a6ecad6fdfb851b40356a8b1060f38235407a0f2706e7b8bb4a13465ca3f81d4f5b99466ac2565c60af15f022d26732e -819ff14cdea3ab89d98e133cd2d0379361e2e2c67ad94eeddcdb9232efd509f51d12f4f03ebd4dd953bd262a886281f7 -8561cd0f7a6dbcddd83fcd7f472d7dbcba95b2d4fb98276f48fccf69f76d284e626d7e41314b633352df8e6333fd52a1 -b42557ccce32d9a894d538c48712cb3e212d06ac05cd5e0527ccd2db1078ee6ae399bf6a601ffdab1f5913d35fc0b20c -89b4008d767aad3c6f93c349d3b956e28307311a5b1cec237e8d74bb0dee7e972c24f347fd56afd915a2342bd7bc32f0 -877487384b207e53f5492f4e36c832c2227f92d1bb60542cfeb35e025a4a7afc2b885fae2528b33b40ab09510398f83e -8c411050b63c9053dd0cd81dacb48753c3d7f162028098e024d17cd6348482703a69df31ad6256e3d25a8bbf7783de39 -a8506b54a88d17ac10fb1b0d1fe4aa40eae7553a064863d7f6b52ccc4236dd4b82d01dca6ba87da9a239e3069ba879fb -b1a24caef9df64750c1350789bb8d8a0db0f39474a1c74ea9ba064b1516db6923f00af8d57c632d58844fb8786c3d47a -959d6e255f212b0708c58a2f75cb1fe932248c9d93424612c1b8d1e640149656059737e4db2139afd5556bcdacf3eda2 -84525af21a8d78748680b6535bbc9dc2f0cf9a1d1740d12f382f6ecb2e73811d6c1da2ad9956070b1a617c61fcff9fe5 -b74417d84597a485d0a8e1be07bf78f17ebb2e7b3521b748f73935b9afbbd82f34b710fb7749e7d4ab55b0c7f9de127d -a4a9aecb19a6bab167af96d8b9d9aa5308eab19e6bfb78f5a580f9bf89bdf250a7b52a09b75f715d651cb73febd08e84 -9777b30be2c5ffe7d29cc2803a562a32fb43b59d8c3f05a707ab60ec05b28293716230a7d264d7cd9dd358fc031cc13e -95dce7a3d4f23ac0050c510999f5fbf8042f771e8f8f94192e17bcbfa213470802ebdbe33a876cb621cf42e275cbfc8b -b0b963ebcbbee847ab8ae740478544350b3ac7e86887e4dfb2299ee5096247cd2b03c1de74c774d9bde94ae2ee2dcd59 -a4ab20bafa316030264e13f7ef5891a2c3b29ab62e1668fcb5881f50a9acac6adbe3d706c07e62f2539715db768f6c43 -901478a297669d608e406fe4989be75264b6c8be12169aa9e0ad5234f459ca377f78484ffd2099a2fe2db5e457826427 -88c76e5c250810c057004a03408b85cd918e0c8903dc55a0dd8bb9b4fc2b25c87f9b8cf5943eb19fbbe99d36490050c5 -91607322bbad4a4f03fc0012d0821eff5f8c516fda45d1ec1133bface6f858bf04b25547be24159cab931a7aa08344d4 -843203e07fce3c6c81f84bc6dc5fb5e9d1c50c8811ace522dc66e8658433a0ef9784c947e6a62c11bf705307ef05212e -91dd8813a5d6dddcda7b0f87f672b83198cd0959d8311b2b26fb1fae745185c01f796fbd03aad9db9b58482483fdadd8 -8d15911aacf76c8bcd7136e958febd6963104addcd751ce5c06b6c37213f9c4fb0ffd4e0d12c8e40c36d658999724bfd -8a36c5732d3f1b497ebe9250610605ee62a78eaa9e1a45f329d09aaa1061131cf1d9df00f3a7d0fe8ad614a1ff9caaae -a407d06affae03660881ce20dab5e2d2d6cddc23cd09b95502a9181c465e57597841144cb34d22889902aff23a76d049 -b5fd856d0578620a7e25674d9503be7d97a2222900e1b4738c1d81ff6483b144e19e46802e91161e246271f90270e6cf -91b7708869cdb5a7317f88c0312d103f8ce90be14fb4f219c2e074045a2a83636fdc3e69e862049fc7c1ef000e832541 -b64719cc5480709d1dae958f1d3082b32a43376da446c8f9f64cb02a301effc9c34d9102051733315a8179aed94d53cc -94347a9542ff9d18f7d9eaa2f4d9b832d0e535fe49d52aa2de08aa8192400eddabdb6444a2a78883e27c779eed7fdf5a -840ef44a733ff1376466698cd26f82cf56bb44811e196340467f932efa3ae1ef9958a0701b3b032f50fd9c1d2aed9ab5 -90ab3f6f67688888a31ffc2a882bb37adab32d1a4b278951a21646f90d03385fc976715fc639a785d015751171016f10 -b56f35d164c24b557dbcbc8a4bfa681ec916f8741ffcb27fb389c164f4e3ed2be325210ef5bdaeae7a172ca9599ab442 -a7921a5a80d7cf6ae81ba9ee05e0579b18c20cd2852762c89d6496aa4c8ca9d1ca2434a67b2c16d333ea8e382cdab1e3 -a506bcfbd7e7e5a92f68a1bd87d07ad5fe3b97aeee40af2bf2cae4efcd77fff03f872732c5b7883aa6584bee65d6f8cb -a8c46cff58931a1ce9cbe1501e1da90b174cddd6d50f3dfdfb759d1d4ad4673c0a8feed6c1f24c7af32865a7d6c984e5 -b45686265a83bff69e312c5149db7bb70ac3ec790dc92e392b54d9c85a656e2bf58596ce269f014a906eafc97461aa5f -8d4009a75ccb2f29f54a5f16684b93202c570d7a56ec1a8b20173269c5f7115894f210c26b41e8d54d4072de2d1c75d0 -aef8810af4fc676bf84a0d57b189760ddc3375c64e982539107422e3de2580b89bd27aa6da44e827b56db1b5555e4ee8 -888f0e1e4a34f48eb9a18ef4de334c27564d72f2cf8073e3d46d881853ac1424d79e88d8ddb251914890588937c8f711 -b64b0aa7b3a8f6e0d4b3499fe54e751b8c3e946377c0d5a6dbb677be23736b86a7e8a6be022411601dd75012012c3555 -8d57776f519f0dd912ea14f79fbab53a30624e102f9575c0bad08d2dc754e6be54f39b11278c290977d9b9c7c0e1e0ad -a018fc00d532ceb2e4de908a15606db9b6e0665dd77190e2338da7c87a1713e6b9b61554e7c1462f0f6d4934b960b15c -8c932be83ace46f65c78e145b384f58e41546dc0395270c1397874d88626fdeda395c8a289d602b4c312fe98c1311856 -89174838e21639d6bdd91a0621f04dc056907b88e305dd66e46a08f6d65f731dea72ae87ca5e3042d609e8de8de9aa26 -b7b7f508bb74f7a827ac8189daa855598ff1d96fa3a02394891fd105d8f0816224cd50ac4bf2ed1cf469ace516c48184 -b31877ad682583283baadd68dc1bebd83f5748b165aadd7fe9ef61a343773b88bcd3a022f36d6c92f339b7bfd72820a9 -b79d77260b25daf9126dab7a193df2d7d30542786fa1733ffaf6261734770275d3ca8bae1d9915d1181a78510b3439db -91894fb94cd4c1dd2ceaf9c53a7020c5799ba1217cf2d251ea5bc91ed26e1159dd758e98282ebe35a0395ef9f1ed15a0 -ab59895cdafd33934ceedfc3f0d5d89880482cba6c99a6db93245f9e41987efd76e0640e80aef31782c9a8c7a83fccec -aa22ea63654315e033e09d4d4432331904a6fc5fb1732557987846e3c564668ca67c60a324b4af01663a23af11a9ce4b -b53ba3ef342601467e1f71aa280e100fbabbd38518fa0193e0099505036ee517c1ac78e96e9baeb549bb6879bb698fb0 -943fd69fd656f37487cca3605dc7e5a215fddd811caf228595ec428751fc1de484a0cb84c667fe4d7c35599bfa0e5e34 -9353128b5ebe0dddc555093cf3e5942754f938173541033e8788d7331fafc56f68d9f97b4131e37963ab7f1c8946f5f1 -a76cd3c566691f65cfb86453b5b31dbaf3cab8f84fe1f795dd1e570784b9b01bdd5f0b3c1e233942b1b5838290e00598 -983d84b2e53ffa4ae7f3ba29ef2345247ea2377686b74a10479a0ef105ecf90427bf53b74c96dfa346d0f842b6ffb25b -92e0fe9063306894a2c6970c001781cff416c87e87cb5fbac927a3192655c3da4063e6fa93539f6ff58efac6adcc5514 -b00a81f03c2b8703acd4e2e4c21e06973aba696415d0ea1a648ace2b0ea19b242fede10e4f9d7dcd61c546ab878bc8f9 -b0d08d880f3b456a10bf65cff983f754f545c840c413aea90ce7101a66eb0a0b9b1549d6c4d57725315828607963f15a -90cb64d03534f913b411375cce88a9e8b1329ce67a9f89ca5df8a22b8c1c97707fec727dbcbb9737f20c4cf751359277 -8327c2d42590dfcdb78477fc18dcf71608686ad66c49bce64d7ee874668be7e1c17cc1042a754bbc77c9daf50b2dae07 -8532171ea13aa7e37178e51a6c775da469d2e26ec854eb16e60f3307db4acec110d2155832c202e9ba525fc99174e3b0 -83ca44b15393d021de2a511fa5511c5bd4e0ac7d67259dce5a5328f38a3cce9c3a269405959a2486016bc27bb140f9ff -b1d36e8ca812be545505c8214943b36cabee48112cf0de369957afa796d37f86bf7249d9f36e8e990f26f1076f292b13 -9803abf45be5271e2f3164c328d449efc4b8fc92dfc1225d38e09630909fe92e90a5c77618daa5f592d23fc3ad667094 -b268ad68c7bf432a01039cd889afae815c3e120f57930d463aece10af4fd330b5bd7d8869ef1bcf6b2e78e4229922edc -a4c91a0d6f16b1553264592b4cbbbf3ca5da32ab053ffbdd3dbb1aed1afb650fb6e0dc5274f71a51d7160856477228db -ad89d043c2f0f17806277ffdf3ecf007448e93968663f8a0b674254f36170447b7527d5906035e5e56f4146b89b5af56 -8b6964f757a72a22a642e4d69102951897e20c21449184e44717bd0681d75f7c5bfa5ee5397f6e53febf85a1810d6ed1 -b08f5cdaabec910856920cd6e836c830b863eb578423edf0b32529488f71fe8257d90aed4a127448204df498b6815d79 -af26bb3358be9d280d39b21d831bb53145c4527a642446073fee5a86215c4c89ff49a3877a7a549486262f6f57a0f476 -b4010b37ec4d7c2af20800e272539200a6b623ae4636ecbd0e619484f4ab9240d02bc5541ace3a3fb955dc0a3d774212 -82752ab52bdcc3cc2fc405cb05a2e694d3df4a3a68f2179ec0652536d067b43660b96f85f573f26fbd664a9ef899f650 -96d392dde067473a81faf2d1fea55b6429126b88b160e39b4210d31d0a82833ffd3a80e07d24d495aea2d96be7251547 -a76d8236d6671204d440c33ac5b8deb71fa389f6563d80e73be8b043ec77d4c9b06f9a586117c7f957f4af0331cbc871 -b6c90961f68b5e385d85c9830ec765d22a425f506904c4d506b87d8944c2b2c09615e740ed351df0f9321a7b93979cae -a6ec5ea80c7558403485b3b1869cdc63bde239bafdf936d9b62a37031628402a36a2cfa5cfbb8e26ac922cb0a209b3ba -8c3195bbdbf9bc0fc95fa7e3d7f739353c947f7767d1e3cb24d8c8602d8ea0a1790ac30b815be2a2ba26caa5227891e2 -a7f8a63d809f1155722c57f375ea00412b00147776ae4444f342550279ef4415450d6f400000a326bf11fea6c77bf941 -97fa404df48433a00c85793440e89bb1af44c7267588ae937a1f5d53e01e1c4d4fc8e4a6d517f3978bfdd6c2dfde012f -a984a0a3836de3d8d909c4629a2636aacb85393f6f214a2ef68860081e9db05ad608024762db0dc35e895dc00e2d4cdd -9526cf088ab90335add1db4d3a4ac631b58cbfbe88fa0845a877d33247d1cfeb85994522e1eb8f8874651bfb1df03e2a -ac83443fd0afe99ad49de9bf8230158c118e2814c9c89db5ac951c240d6c2ce45e7677221279d9e97848ec466b99aafe -aeeefdbaba612e971697798ceaf63b247949dc823a0ad771ae5b988a5e882b338a98d3d0796230f49d533ec5ba411b39 -ae3f248b5a7b0f92b7820a6c5ae21e5bd8f4265d4f6e21a22512079b8ee9be06393fd3133ce8ebac0faf23f4f8517e36 -a64a831b908eee784b8388b45447d2885ec0551b26b0c2b15e5f417d0a12c79e867fb7bd3d008d0af98b44336f8ec1ad -b242238cd8362b6e440ba21806905714dd55172db25ec7195f3fc4937b2aba146d5cbf3cf691a1384b4752dc3b54d627 -819f97f337eea1ffb2a678cc25f556f1aab751c6b048993a1d430fe1a3ddd8bb411c152e12ca60ec6e057c190cd1db9a -b9d7d187407380df54ee9fef224c54eec1bfabf17dc8abf60765b7951f538f59aa26fffd5846cfe05546c35f59b573f4 -aa6e3c14efa6a5962812e3f94f8ce673a433f4a82d07a67577285ea0eaa07f8be7115853122d12d6d4e1fdf64c504be1 -82268bee9c1662d3ddb5fb785abfae6fb8b774190f30267f1d47091d2cd4b3874db4372625aa36c32f27b0eee986269b -b236459565b7b966166c4a35b2fa71030b40321821b8e96879d95f0e83a0baf33fa25721f30af4a631df209e25b96061 -8708d752632d2435d2d5b1db4ad1fa2558d776a013655f88e9a3556d86b71976e7dfe5b8834fdec97682cd94560d0d0d -ae1424a68ae2dbfb0f01211f11773732a50510b5585c1fb005cb892b2c6a58f4a55490b5c5b4483c6fce40e9d3236a52 -b3f5f722af9dddb07293c871ce97abbccba0093ca98c8d74b1318fa21396fc1b45b69c15084f63d728f9908442024506 -9606f3ce5e63886853ca476dc0949e7f1051889d529365c0cb0296fdc02abd088f0f0318ecd2cf36740a3634132d36f6 -b11a833a49fa138db46b25ff8cdda665295226595bc212c0931b4931d0a55c99da972c12b4ef753f7e37c6332356e350 -afede34e7dab0a9e074bc19a7daddb27df65735581ca24ad70c891c98b1349fcebbcf3ba6b32c2617fe06a5818dabc2d -97993d456e459e66322d01f8eb13918979761c3e8590910453944bdff90b24091bb018ac6499792515c9923be289f99f -977e3e967eff19290a192cd11df3667d511b398fb3ac9a5114a0f3707e25a0edcb56105648b1b85a8b7519fc529fc6f6 -b873a7c88bf58731fe1bf61ff6828bf114cf5228f254083304a4570e854e83748fc98683ddba62d978fff7909f2c5c47 -ad4b2691f6f19da1d123aaa23cca3e876247ed9a4ab23c599afdbc0d3aa49776442a7ceaa996ac550d0313d9b9a36cee -b9210713c78e19685608c6475bfa974b57ac276808a443f8b280945c5d5f9c39da43effa294bfb1a6c6f7b6b9f85bf6c -a65152f376113e61a0e468759de38d742caa260291b4753391ee408dea55927af08a4d4a9918600a3bdf1df462dffe76 -8bf8c27ad5140dde7f3d2280fd4cc6b29ab76537e8d7aa7011a9d2796ee3e56e9a60c27b5c2da6c5e14fc866301dc195 -92fde8effc9f61393a2771155812b863cff2a0c5423d7d40aa04d621d396b44af94ddd376c28e7d2f53c930aea947484 -97a01d1dd9ee30553ce676011aea97fa93d55038ada95f0057d2362ae9437f3ed13de8290e2ff21e3167dd7ba10b9c3f -89affffaa63cb2df3490f76f0d1e1d6ca35c221dd34057176ba739fa18d492355e6d2a5a5ad93a136d3b1fed0bb8aa19 -928b8e255a77e1f0495c86d3c63b83677b4561a5fcbbe5d3210f1e0fc947496e426d6bf3b49394a5df796c9f25673fc4 -842a0af91799c9b533e79ee081efe2a634cac6c584c2f054fb7d1db67dde90ae36de36cbf712ec9cd1a0c7ee79e151ea -a65b946cf637e090baf2107c9a42f354b390e7316beb8913638130dbc67c918926eb87bec3b1fe92ef72bc77a170fa3b -aafc0f19bfd71ab5ae4a8510c7861458b70ad062a44107b1b1dbacbfa44ba3217028c2824bd7058e2fa32455f624040b -95269dc787653814e0be899c95dba8cfa384f575a25e671c0806fd80816ad6797dc819d30ae06e1d0ed9cb01c3950d47 -a1e760f7fa5775a1b2964b719ff961a92083c5c617f637fc46e0c9c20ab233f8686f7f38c3cb27d825c54dd95e93a59b -ac3b8a7c2317ea967f229eddc3e23e279427f665c4705c7532ed33443f1243d33453c1088f57088d2ab1e3df690a9cc9 -b787beeddfbfe36dd51ec4efd9cf83e59e84d354c3353cc9c447be53ae53d366ed1c59b686e52a92f002142c8652bfe0 -b7a64198300cb6716aa7ac6b25621f8bdec46ad5c07a27e165b3f774cdf65bcfdbf31e9bae0c16b44de4b00ada7a4244 -b8ae9f1452909e0c412c7a7fe075027691ea8df1347f65a5507bc8848f1d2c833d69748076db1129e5b4fb912f65c86c -9682e41872456b9fa67def89e71f06d362d6c8ca85c9c48536615bc401442711e1c9803f10ab7f8ab5feaec0f9df20a6 -88889ff4e271dc1c7e21989cc39f73cde2f0475acd98078281591ff6c944fadeb9954e72334319050205d745d4df73df -8f79b5b8159e7fd0d93b0645f3c416464f39aec353b57d99ecf24f96272df8a068ad67a6c90c78d82c63b40bb73989bb -838c01a009a3d8558a3f0bdd5e22de21af71ca1aefc8423c91dc577d50920e9516880e87dce3e6d086e11cd45c9052d9 -b97f1c6eee8a78f137c840667cc288256e39294268a3009419298a04a1d0087c9c9077b33c917c65caf76637702dda8a -972284ce72f96a61c899260203dfa06fc3268981732bef74060641c1a5068ead723e3399431c247ca034b0dae861e8df -945a8d52d6d3db6663dbd3110c6587f9e9c44132045eeffba15621576d178315cb52870fa5861669f84f0bee646183fe -a0a547b5f0967b1c3e5ec6c6a9a99f0578521489180dfdfbb5561f4d166baac43a2f06f950f645ce991664e167537eed -a0592cda5cdddf1340033a745fd13a6eff2021f2e26587116c61c60edead067e0f217bc2bef4172a3c9839b0b978ab35 -b9c223b65a3281587fa44ec829e609154b32f801fd1de6950e01eafb07a8324243b960d5735288d0f89f0078b2c42b5b -99ebfc3b8f9f98249f4d37a0023149ed85edd7a5abe062c8fb30c8c84555258b998bdcdd1d400bc0fa2a4aaa8b224466 -955b68526e6cb3937b26843270f4e60f9c6c8ece2fa9308fe3e23afa433309c068c66a4bc16ee2cf04220f095e9afce4 -b766caeafcc00378135ae53397f8a67ed586f5e30795462c4a35853de6681b1f17401a1c40958de32b197c083b7279c1 -921bf87cad947c2c33fa596d819423c10337a76fe5a63813c0a9dc78a728207ae7b339407a402fc4d0f7cba3af6da6fc -a74ba1f3bc3e6c025db411308f49b347ec91da1c916bda9da61e510ec8d71d25e0ac0f124811b7860e5204f93099af27 -a29b4d144e0bf17a7e8353f2824cef0ce85621396babe8a0b873ca1e8a5f8d508b87866cf86da348470649fceefd735c -a8040e12ffc3480dd83a349d06741d1572ef91932c46f5cf03aee8454254156ee95786fd013d5654725e674c920cec32 -8c4cf34ca60afd33923f219ffed054f90cd3f253ffeb2204a3b61b0183417e366c16c07fae860e362b0f2bfe3e1a1d35 -8195eede4ddb1c950459df6c396b2e99d83059f282b420acc34220cadeed16ab65c856f2c52568d86d3c682818ed7b37 -91fff19e54c15932260aa990c7fcb3c3c3da94845cc5aa8740ef56cf9f58d19b4c3c55596f8d6c877f9f4d22921d93aa -a3e0bf7e5d02a80b75cf75f2db7e66cb625250c45436e3c136d86297d652590ec97c2311bafe407ad357c79ab29d107b -81917ff87e5ed2ae4656b481a63ced9e6e5ff653b8aa6b7986911b8bc1ee5b8ef4f4d7882c3f250f2238e141b227e510 -915fdbe5e7de09c66c0416ae14a8750db9412e11dc576cf6158755fdcaf67abdbf0fa79b554cac4fe91c4ec245be073f -8df27eafb5c3996ba4dc5773c1a45ca77e626b52e454dc1c4058aa94c2067c18332280630cc3d364821ee53bf2b8c130 -934f8a17c5cbb827d7868f5c8ca00cb027728a841000a16a3428ab16aa28733f16b52f58c9c4fbf75ccc45df72d9c4df -b83f4da811f9183c25de8958bc73b504cf790e0f357cbe74ef696efa7aca97ad3b7ead1faf76e9f982c65b6a4d888fc2 -87188213c8b5c268dc2b6da413f0501c95749e953791b727450af3e43714149c115b596b33b63a2f006a1a271b87efd0 -83e9e888ab9c3e30761de635d9aabd31248cdd92f7675fc43e4b21fd96a03ec1dc4ad2ec94fec857ffb52683ac98e360 -b4b9a1823fe2d983dc4ec4e3aaea297e581c3fc5ab4b4af5fa1370caa37af2d1cc7fc6bfc5e7da60ad8fdce27dfe4b24 -856388bc78aef465dbcdd1f559252e028c9e9a2225c37d645c138e78f008f764124522705822a61326a6d1c79781e189 -a6431b36db93c3b47353ba22e7c9592c9cdfb9cbdd052ecf2cc3793f5b60c1e89bc96e6bae117bfd047f2308da00dd2f -b619972d48e7e4291542dcde08f7a9cdc883c892986ded2f23ccb216e245cd8d9ad1d285347b0f9d7611d63bf4cee2bc -8845cca6ff8595955f37440232f8e61d5351500bd016dfadd182b9d39544db77a62f4e0102ff74dd4173ae2c181d24ef -b2f5f7fa26dcd3b6550879520172db2d64ee6aaa213cbef1a12befbce03f0973a22eb4e5d7b977f466ac2bf8323dcedd -858b7f7e2d44bdf5235841164aa8b4f3d33934e8cb122794d90e0c1cac726417b220529e4f896d7b77902ab0ccd35b3a -80b0408a092dae2b287a5e32ea1ad52b78b10e9c12f49282976cd738f5d834e03d1ad59b09c5ccaccc39818b87d06092 -b996b0a9c6a2d14d984edcd6ab56bc941674102980d65b3ad9733455f49473d3f587c8cbf661228a7e125ddbe07e3198 -90224fcebb36865293bd63af786e0c5ade6b67c4938d77eb0cbae730d514fdd0fe2d6632788e858afd29d46310cf86df -b71351fdfff7168b0a5ec48397ecc27ac36657a8033d9981e97002dcca0303e3715ce6dd3f39423bc8ef286fa2e9e669 -ae2a3f078b89fb753ce4ed87e0c1a58bb19b4f0cfb6586dedb9fcab99d097d659a489fb40e14651741e1375cfc4b6c5f -8ef476b118e0b868caed297c161f4231bbeb863cdfa5e2eaa0fc6b6669425ce7af50dc374abceac154c287de50c22307 -92e46ab472c56cfc6458955270d3c72b7bde563bb32f7d4ab4d959db6f885764a3d864e1aa19802fefaa5e16b0cb0b54 -96a3f68323d1c94e73d5938a18a377af31b782f56212de3f489d22bc289cf24793a95b37f1d6776edf88114b5c1fa695 -962cc068cfce6faaa27213c4e43e44eeff0dfbb6d25b814e82c7da981fb81d7d91868fa2344f05fb552362f98cfd4a72 -895d4e4c4ad670abf66d43d59675b1add7afad7438ada8f42a0360c704cee2060f9ac15b4d27e9b9d0996bb801276fe3 -b3ad18d7ece71f89f2ef749b853c45dc56bf1c796250024b39a1e91ed11ca32713864049c9aaaea60cde309b47486bbf -8f05404e0c0258fdbae50e97ccb9b72ee17e0bd2400d9102c0dad981dac8c4c71585f03e9b5d50086d0a2d3334cb55d1 -8bd877e9d4591d02c63c6f9fc9976c109de2d0d2df2bfa5f6a3232bab5b0b8b46e255679520480c2d7a318545efa1245 -8d4c16b5d98957c9da13d3f36c46f176e64e5be879f22be3179a2c0e624fe4758a82bf8c8027410002f973a3b84cd55a -86e2a8dea86427b424fa8eada881bdff896907084a495546e66556cbdf070b78ba312bf441eb1be6a80006d25d5097a3 -8608b0c117fd8652fdab0495b08fadbeba95d9c37068e570de6fddfef1ba4a1773b42ac2be212836141d1bdcdef11a17 -a13d6febf5fb993ae76cae08423ca28da8b818d6ef0fde32976a4db57839cd45b085026b28ee5795f10a9a8e3098c683 -8e261967fa6de96f00bc94a199d7f72896a6ad8a7bbb1d6187cca8fad824e522880e20f766620f4f7e191c53321d70f9 -8b8e8972ac0218d7e3d922c734302803878ad508ca19f5f012bc047babd8a5c5a53deb5fe7c15a4c00fd6d1cb9b1dbd0 -b5616b233fb3574a2717d125a434a2682ff68546dccf116dd8a3b750a096982f185614b9fb6c7678107ff40a451f56fa -aa6adf9b0c3334b0d0663f583a4914523b2ac2e7adffdb026ab9109295ff6af003ef8357026dbcf789896d2afded8d73 -acb72df56a0b65496cd534448ed4f62950bb1e11e50873b6ed349c088ee364441821294ce0f7c61bd7d38105bea3b442 -abae12df83e01ec947249fedd0115dc501d2b03ff7232092979eda531dbbca29ace1d46923427c7dde4c17bdf3fd7708 -820b4fc2b63a9fda7964acf5caf19a2fc4965007cb6d6b511fcafcb1f71c3f673a1c0791d3f86e3a9a1eb6955b191cc0 -af277259d78c6b0f4f030a10c53577555df5e83319ddbad91afbd7c30bc58e7671c56d00d66ec3ab5ef56470cd910cee -ad4a861c59f1f5ca1beedd488fb3d131dea924fffd8e038741a1a7371fad7370ca5cf80dc01f177fbb9576713bb9a5b3 -b67a5162982ce6a55ccfb2f177b1ec26b110043cf18abd6a6c451cf140b5af2d634591eb4f28ad92177d8c7e5cd0a5e8 -96176d0a83816330187798072d449cbfccff682561e668faf6b1220c9a6535b32a6e4f852e8abb00f79abb87493df16b -b0afe6e7cb672e18f0206e4423f51f8bd0017bf464c4b186d46332c5a5847647f89ff7fa4801a41c1b0b42f6135bcc92 -8fc5e7a95ef20c1278c645892811f6fe3f15c431ebc998a32ec0da44e7213ea934ed2be65239f3f49b8ec471e9914160 -b7793e41adda6c82ba1f2a31f656f6205f65bf8a3d50d836ee631bc7ce77c153345a2d0fc5c60edf8b37457c3729c4ec -a504dd7e4d6b2f4379f22cc867c65535079c75ccc575955f961677fa63ecb9f74026fa2f60c9fb6323c1699259e5e9c8 -ab899d00ae693649cc1afdf30fb80d728973d2177c006e428bf61c7be01e183866614e05410041bc82cb14a33330e69c -8a3bd8b0b1be570b65c4432a0f6dc42f48a2000e30ab089cf781d38f4090467b54f79c0d472fcbf18ef6a00df69cc6f3 -b4d7028f7f76a96a3d7803fca7f507ae11a77c5346e9cdfccb120a833a59bda1f4264e425aa588e7a16f8e7638061d84 -b9c7511a76ea5fb105de905d44b02edb17008335766ee357ed386b7b3cf19640a98b38785cb14603c1192bee5886c9b6 -8563afb12e53aed71ac7103ab8602bfa8371ae095207cb0d59e8fd389b6ad1aff0641147e53cb6a7ca16c7f37c9c5e6b -8e108be614604e09974a9ed90960c28c4ea330a3d9a0cb4af6dd6f193f84ab282b243ecdf549b3131036bebc8905690c -b794d127fbedb9c5b58e31822361706ffac55ce023fbfe55716c3c48c2fd2f2c7660a67346864dfe588812d369cb50b6 -b797a3442fc3b44f41baefd30346f9ac7f96e770d010d53c146ce74ce424c10fb62758b7e108b8abfdc5fafd89d745cb -993bb71e031e8096442e6205625e1bfddfe6dd6a83a81f3e2f84fafa9e5082ab4cad80a099f21eff2e81c83457c725c3 -8711ab833fc03e37acf2e1e74cfd9133b101ff4144fe30260654398ae48912ab46549d552eb9d15d2ea57760d35ac62e -b21321fd2a12083863a1576c5930e1aecb330391ef83326d9d92e1f6f0d066d1394519284ddab55b2cb77417d4b0292f -877d98f731ffe3ee94b0b5b72d127630fa8a96f6ca4f913d2aa581f67732df6709493693053b3e22b0181632ac6c1e3b -ae391c12e0eb8c145103c62ea64f41345973311c3bf7281fa6bf9b7faafac87bcf0998e5649b9ef81e288c369c827e07 -b83a2842f36998890492ab1cd5a088d9423d192681b9a3a90ec518d4c541bce63e6c5f4df0f734f31fbfdd87785a2463 -a21b6a790011396e1569ec5b2a423857b9bec16f543e63af28024e116c1ea24a3b96e8e4c75c6537c3e4611fd265e896 -b4251a9c4aab3a495da7a42e684ba4860dbcf940ad1da4b6d5ec46050cbe8dab0ab9ae6b63b5879de97b905723a41576 -8222f70aebfe6ac037f8543a08498f4cadb3edaac00336fc00437eb09f2cba758f6c38e887cc634b4d5b7112b6334836 -86f05038e060594c46b5d94621a1d9620aa8ba59a6995baf448734e21f58e23c1ea2993d3002ad5250d6edd5ba59b34f -a7c0c749baef811ab31b973c39ceb1d94750e2bc559c90dc5eeb20d8bb6b78586a2b363c599ba2107d6be65cd435f24e -861d46a5d70b38d6c1cd72817a2813803d9f34c00320c8b62f8b9deb67f5b5687bc0b37c16d28fd017367b92e05da9ca -b3365d3dab639bffbe38e35383686a435c8c88b397b717cd4aeced2772ea1053ceb670f811f883f4e02975e5f1c4ac58 -a5750285f61ab8f64cd771f6466e2c0395e01b692fd878f2ef2d5c78bdd8212a73a3b1dfa5e4c8d9e1afda7c84857d3b -835a10809ccf939bc46cf950a33b36d71be418774f51861f1cd98a016ade30f289114a88225a2c11e771b8b346cbe6ef -a4f59473a037077181a0a62f1856ec271028546ca9452b45cedfcb229d0f4d1aabfc13062b07e536cc8a0d4b113156a2 -95cd14802180b224d44a73cc1ed599d6c4ca62ddcaa503513ccdc80aaa8be050cc98bd4b4f3b639549beb4587ac6caf9 -973b731992a3e69996253d7f36dd7a0af1982b5ed21624b77a7965d69e9a377b010d6dabf88a8a97eec2a476259859cc -af8a1655d6f9c78c8eb9a95051aa3baaf9c811adf0ae8c944a8d3fcba87b15f61021f3baf6996fa0aa51c81b3cb69de1 -835aad5c56872d2a2d6c252507b85dd742bf9b8c211ccb6b25b52d15c07245b6d89b2a40f722aeb5083a47cca159c947 -abf4e970b02bef8a102df983e22e97e2541dd3650b46e26be9ee394a3ea8b577019331857241d3d12b41d4eacd29a3ac -a13c32449dbedf158721c13db9539ae076a6ce5aeaf68491e90e6ad4e20e20d1cdcc4a89ed9fd49cb8c0dd50c17633c1 -8c8f78f88b7e22dd7e9150ab1c000f10c28e696e21d85d6469a6fe315254740f32e73d81ab1f3c1cf8f544c86df506e8 -b4b77f2acfe945abf81f2605f906c10b88fb4d28628487fb4feb3a09f17f28e9780445dfcee4878349d4c6387a9d17d4 -8d255c235f3812c6ecc646f855fa3832be5cb4dbb9c9e544989fafdf3f69f05bfd370732eaf954012f0044aa013fc9c6 -b982efd3f34b47df37c910148ac56a84e8116647bea24145a49e34e0a6c0176e3284d838dae6230cb40d0be91c078b85 -983f365aa09bd85df2a6a2ad8e4318996b1e27d02090755391d4486144e40d80b1fbfe1c798d626db92f52e33aa634da -95fd1981271f3ea3a41d654cf497e6696730d9ff7369f26bc4d7d15c7adb4823dd0c42e4a005a810af12d234065e5390 -a9f5219bd4b913c186ef30c02f995a08f0f6f1462614ea5f236964e02bdaa33db9d9b816c4aee5829947840a9a07ba60 -9210e6ceb05c09b46fd09d036287ca33c45124ab86315e5d6911ff89054f1101faaa3e83d123b7805056d388bcec6664 -8ed9cbf69c6ff3a5c62dd9fe0d7264578c0f826a29e614bc2fb4d621d90c8c9992438accdd7a614b1dca5d1bb73dc315 -85cf2a8cca93e00da459e3cecd22c342d697eee13c74d5851634844fc215f60053cf84b0e03c327cb395f48d1c71a8a4 -8818a18e9a2ec90a271b784400c1903089ffb0e0b40bc5abbbe12fbebe0f731f91959d98c5519ef1694543e31e2016d4 -8dabc130f296fa7a82870bf9a8405aaf542b222ed9276bba9bd3c3555a0f473acb97d655ee7280baff766a827a8993f0 -ac7952b84b0dc60c4d858f034093b4d322c35959605a3dad2b806af9813a4680cb038c6d7f4485b4d6b2ff502aaeca25 -ad65cb6d57b48a2602568d2ec8010baed0eb440eec7638c5ec8f02687d764e9de5b5d42ad5582934e592b48471c22d26 -a02ab8bd4c3d114ea23aebdd880952f9495912817da8c0c08eabc4e6755439899d635034413d51134c72a6320f807f1c -8319567764b8295402ec1ebef4c2930a138480b37e6d7d01c8b4c9cd1f2fc3f6e9a44ae6e380a0c469b25b06db23305f -afec53b2301dc0caa8034cd9daef78c48905e6068d692ca23d589b84a6fa9ddc2ed24a39480597e19cb3e83eec213b3f -ac0b4ffdb5ae08e586a9cdb98f9fe56f4712af3a97065e89e274feacfb52b53c839565aee93c4cfaaccfe51432c4fab0 -8972cbf07a738549205b1094c5987818124144bf187bc0a85287c94fdb22ce038c0f11df1aa16ec5992e91b44d1af793 -b7267aa6f9e3de864179b7da30319f1d4cb2a3560f2ea980254775963f1523b44c680f917095879bebfa3dc2b603efcf -80f68f4bfc337952e29504ee5149f15093824ea7ab02507efd1317a670f6cbc3611201848560312e3e52e9d9af72eccf -8897fee93ce8fc1e1122e46b6d640bba309384dbd92e46e185e6364aa8210ebf5f9ee7e5e604b6ffba99aa80a10dd7d0 -b58ea6c02f2360be60595223d692e82ee64874fda41a9f75930f7d28586f89be34b1083e03bbc1575bbfdda2d30db1ea -85a523a33d903280d70ac5938770453a58293480170c84926457ac2df45c10d5ff34322ab130ef4a38c916e70d81af53 -a2cbf045e1bed38937492c1f2f93a5ba41875f1f262291914bc1fc40c60bd0740fb3fea428faf6da38b7c180fe8ac109 -8c09328770ed8eb17afc6ac7ddd87bb476de18ed63cab80027234a605806895959990c47bd10d259d7f3e2ecb50074c9 -b4b9e19edb4a33bde8b7289956568a5b6b6557404e0a34584b5721fe6f564821091013fbb158e2858c6d398293bb4b59 -8a47377df61733a2aa5a0e945fce00267f8e950f37e109d4487d92d878fb8b573317bb382d902de515b544e9e233458d -b5804c9d97efeff5ca94f3689b8088c62422d92a1506fd1d8d3b1b30e8a866ad0d6dad4abfa051dfc4471250cac4c5d9 -9084a6ee8ec22d4881e9dcc8a9eb3c2513523d8bc141942370fd191ad2601bf9537a0b1e84316f3209b3d8a54368051e -85447eea2fa26656a649f8519fa67279183044791d61cf8563d0783d46d747d96af31d0a93507bbb2242666aa87d3720 -97566a84481027b60116c751aec552adfff2d9038e68d48c4db9811fb0cbfdb3f1d91fc176a0b0d988a765f8a020bce1 -ae87e5c1b9e86c49a23dceda4ecfd1dcf08567f1db8e5b6ec752ebd45433c11e7da4988573cdaebbb6f4135814fc059e -abee05cf9abdbc52897ac1ce9ed157f5466ed6c383d6497de28616238d60409e5e92619e528af8b62cc552bf09970dc2 -ae6d31cd7bf9599e5ee0828bab00ceb4856d829bba967278a73706b5f388465367aa8a6c7da24b5e5f1fdd3256ef8e63 -ac33e7b1ee47e1ee4af472e37ab9e9175260e506a4e5ce449788075da1b53c44cb035f3792d1eea2aa24b1f688cc6ed3 -80f65b205666b0e089bb62152251c48c380a831e5f277f11f3ef4f0d52533f0851c1b612267042802f019ec900dc0e8f -858520ad7aa1c9fed738e3b583c84168f2927837ad0e1d326afe9935c26e9b473d7f8c382e82ef1fe37d2b39bb40a1ee -b842dd4af8befe00a97c2d0f0c33c93974761e2cb9e5ab8331b25170318ddd5e4bdbc02d8f90cbfdd5f348f4f371c1f7 -8bf2cb79bc783cb57088aae7363320cbeaabd078ffdec9d41bc74ff49e0043d0dad0086a30e5112b689fd2f5a606365d -982eb03bbe563e8850847cd37e6a3306d298ab08c4d63ab6334e6b8c1fa13fce80cf2693b09714c7621d74261a0ff306 -b143edb113dec9f1e5105d4a93fbe502b859e587640d3db2f628c09a17060e6aec9e900e2c8c411cda99bc301ff96625 -af472d9befa750dcebc5428fe1a024f18ec1c07bca0f95643ce6b5f4189892a910285afb03fd7ed7068fbe614e80d33c -a97e3bc57ede73ecd1bbf02de8f51b4e7c1a067da68a3cd719f4ba26a0156cbf1cef2169fd35a18c5a4cced50d475998 -a862253c937cf3d75d7183e5f5be6a4385d526aeda5171c1c60a8381fea79f88f5f52a4fab244ecc70765d5765e6dfd5 -90cb776f8e5a108f1719df4a355bebb04bf023349356382cae55991b31720f0fd03206b895fa10c56c98f52453be8778 -a7614e8d0769dccd520ea4b46f7646e12489951efaef5176bc889e9eb65f6e31758df136b5bf1e9107e68472fa9b46ec -ac3a9b80a3254c42e5ed3a090a0dd7aee2352f480de96ad187027a3bb6c791eddfc3074b6ffd74eea825188f107cda4d -82a01d0168238ef04180d4b6e0a0e39024c02c2d75b065017c2928039e154d093e1af4503f4d1f3d8a948917abb5d09f -8fab000a2b0eef851a483aec8d2dd85fe60504794411a2f73ed82e116960547ac58766cb73df71aea71079302630258d -872451a35c6db61c63e9b8bb9f16b217f985c20be4451c14282c814adb29d7fb13f201367c664435c7f1d4d9375d7a58 -887d9ff54cc96b35d562df4a537ff972d7c4b3fd91ab06354969a4cfede0b9fc68bbffb61d0dbf1a58948dc701e54f5a -8cb5c2a6bd956875d88f41ae24574434f1308514d44057b55c9c70f13a3366ed054150eed0955a38fda3f757be73d55f -89ad0163cad93e24129d63f8e38422b7674632a8d0a9016ee8636184cab177659a676c4ee7efba3abe1a68807c656d60 -b9ec01c7cab6d00359b5a0b4a1573467d09476e05ca51a9227cd16b589a9943d161eef62dcc73f0de2ec504d81f4d252 -8031d17635d39dfe9705c485d2c94830b6fc9bc67b91300d9d2591b51e36a782e77ab5904662effa9382d9cca201f525 -8be5a5f6bc8d680e5092d6f9a6585acbaaaa2ddc671da560dcf5cfa4472f4f184b9597b5b539438accd40dda885687cc -b1fc0f052fae038a2e3de3b3a96b0a1024b009de8457b8b3adb2d315ae68a89af905720108a30038e5ab8d0d97087785 -8b8bdc77bd3a6bc7ca5492b6f8c614852c39a70d6c8a74916eaca0aeb4533b11898b8820a4c2620a97bf35e275480029 -af35f4dc538d4ad5cdf710caa38fd1eb496c3fa890a047b6a659619c5ad3054158371d1e88e0894428282eed9f47f76b -8166454a7089cc07758ad78724654f4e7a1a13e305bbf88ddb86f1a4b2904c4fc8ab872d7da364cdd6a6c0365239e2ad -ab287c7d3addce74ce40491871c768abe01daaa0833481276ff2e56926b38a7c6d2681ffe837d2cc323045ad1a4414f9 -b90317f4505793094d89365beb35537f55a6b5618904236258dd04ca61f21476837624a2f45fef8168acf732cab65579 -98ae5ea27448e236b6657ab5ef7b1cccb5372f92ab25f5fa651fbac97d08353a1dae1b280b1cd42b17d2c6a70a63ab9d -adcf54e752d32cbaa6cb98fbca48d8cd087b1db1d131d465705a0d8042c8393c8f4d26b59006eb50129b21e6240f0c06 -b591a3e4db18a7345fa935a8dd7994bbac5cc270b8ebd84c8304c44484c7a74afb45471fdbe4ab22156a30fae1149b40 -806b53ac049a42f1dcc1d6335505371da0bf27c614f441b03bbf2e356be7b2fb4eed7117eabcce9e427a542eaa2bf7d8 -800482e7a772d49210b81c4a907f5ce97f270b959e745621ee293cf8c71e8989363d61f66a98f2d16914439544ca84c7 -99de9eafdad3617445312341644f2bb888680ff01ce95ca9276b1d2e5ef83fa02dab5e948ebf66c17df0752f1bd37b70 -961ee30810aa4c93ae157fbe9009b8e443c082192bd36a73a6764ff9b2ad8b0948fe9a73344556e01399dd77badb4257 -ae0a361067c52efbe56c8adf982c00432cd478929459fc7f74052c8ee9531cd031fe1335418fde53f7c2ef34254eb7ac -a3503d16b6b27eb20c1b177bcf90d13706169220523a6271b85b2ce35a9a2b9c5bed088540031c0a4ebfdae3a4c6ab04 -909420122c3e723289ca4e7b81c2df5aff312972a2203f4c45821b176e7c862bf9cac7f7df3adf1d59278f02694d06e7 -989f42380ae904b982f85d0c6186c1aef5d6bcba29bcfbb658e811b587eb2749c65c6e4a8cc6409c229a107499a4f5d7 -8037a6337195c8e26a27ea4ef218c6e7d79a9720aaab43932d343192abc2320fe72955f5e431c109093bda074103330a -b312e168663842099b88445e940249cc508f080ab0c94331f672e7760258dbd86be5267e4cf25ea25facb80bff82a7e9 -aaa3ff8639496864fcdbfdda1ac97edc4f08e3c9288b768f6c8073038c9fbbf7e1c4bea169b4d45c31935cdf0680d45e -97dbd3df37f0b481a311dfc5f40e59227720f367912200d71908ef6650f32cc985cb05b981e3eea38958f7e48d10a15d -a89d49d1e267bb452d6cb621b9a90826fe55e9b489c0427b94442d02a16f390eed758e209991687f73f6b5a032321f42 -9530dea4e0e19d6496f536f2e75cf7d814d65fde567055eb20db48fd8d20d501cd2a22fb506db566b94c9ee10f413d43 -81a7009b9e67f1965fa7da6a57591c307de91bf0cd35ab4348dc4a98a4961e096d004d7e7ad318000011dc4342c1b809 -83440a9402b766045d7aca61a58bba2aa29cac1cf718199e472ba086f5d48093d9dda4d135292ba51d049a23964eceae -a06c9ce5e802df14f6b064a3d1a0735d429b452f0e2e276042800b0a4f16df988fd94cf3945921d5dd3802ab2636f867 -b1359e358b89936dee9e678a187aad3e9ab14ac40e96a0a68f70ee2583cdcf467ae03bef4215e92893f4e12f902adec8 -835304f8619188b4d14674d803103d5a3fa594d48e96d9699e653115dd05fdc2dda6ba3641cf7ad53994d448da155f02 -8327cba5a9ff0d3f5cd0ae55e77167448926d5fcf76550c0ad978092a14122723090c51c415e88e42a2b62eb07cc3981 -b373dcdaea85f85ce9978b1426a7ef4945f65f2d3467a9f1cc551a99766aac95df4a09e2251d3f89ca8c9d1a7cfd7b0e -ab1422dc41af2a227b973a6fd124dfcb2367e2a11a21faa1d381d404f51b7257e5bc82e9cf20cd7fe37d7ae761a2ab37 -a93774a03519d2f20fdf2ef46547b0a5b77c137d6a3434b48d56a2cbef9e77120d1b85d0092cf8842909213826699477 -8eb967a495a38130ea28711580b7e61bcd1d051cd9e4f2dbf62f1380bd86e0d60e978d72f6f31e909eb97b3b9a2b867c -ae8213378da1287ba1fe4242e1acaec19b877b6fe872400013c6eac1084b8d03156792fa3020201725b08228a1e80f49 -b143daf6893d674d607772b3b02d8ac48f294237e2f2c87963c0d4e26d9227d94a2a13512457c3d5883544bbc259f0ef -b343bd2aca8973888e42542218924e2dda2e938fd1150d06878af76f777546213912b7c7a34a0f94186817d80ffa185c -b188ebc6a8c3007001aa347ae72cc0b15d09bc6c19a80e386ee4b334734ec0cc2fe8b493c2422f38d1e6d133cc3db6fe -b795f6a8b9b826aaeee18ccd6baf6c5adeeec85f95eb5b6d19450085ec7217e95a2d9e221d77f583b297d0872073ba0e -b1c7dbd998ad32ae57bfa95deafa147024afd57389e98992c36b6e52df915d3d5a39db585141ec2423173e85d212fed8 -812bcdeb9fe5f12d0e1df9964798056e1f1c3de3b17b6bd2919b6356c4b86d8e763c01933efbe0224c86a96d5198a4be -b19ebeda61c23d255cbf472ef0b8a441f4c55b70f0d8ed47078c248b1d3c7c62e076b43b95c00a958ec8b16d5a7cb0d7 -b02adc9aaa20e0368a989c2af14ff48b67233d28ebee44ff3418bb0473592e6b681af1cc45450bd4b175df9051df63d9 -8d87f0714acee522eb58cec00360e762adc411901dba46adc9227124fa70ee679f9a47e91a6306d6030dd4eb8de2f3c1 -8be54cec21e74bcc71de29dc621444263737db15f16d0bb13670f64e42f818154e04b484593d19ef95f2ee17e4b3fe21 -ab8e20546c1db38d31493b5d5f535758afb17e459645c1b70813b1cf7d242fd5d1f4354a7c929e8f7259f6a25302e351 -89f035a1ed8a1e302ac893349ba8ddf967580fcb6e73d44af09e3929cde445e97ff60c87dafe489e2c0ab9c9986cfa00 -8b2b0851a795c19191a692af55f7e72ad2474efdc5401bc3733cfdd910e34c918aaebe69d5ea951bdddf3c01cabbfc67 -a4edb52c2b51495ccd1ee6450fc14b7b3ede8b3d106808929d02fb31475bacb403e112ba9c818d2857651e508b3a7dd1 -9569341fded45d19f00bcf3cbf3f20eb2b4d82ef92aba3c8abd95866398438a2387437e580d8b646f17cf6fde8c5af23 -aa4b671c6d20f72f2f18a939a6ff21cc37e0084b44b4a717f1be859a80b39fb1be026b3205adec2a66a608ec2bcd578f -94902e980de23c4de394ad8aec91b46f888d18f045753541492bfbb92c59d3daa8de37ae755a6853744af8472ba7b72b -af651ef1b2a0d30a7884557edfad95b6b5d445a7561caebdc46a485aedd25932c62c0798465c340a76f6feaa196dd712 -b7b669b8e5a763452128846dd46b530dca4893ace5cc5881c7ddcd3d45969d7e73fbebdb0e78aa81686e5f7b22ec5759 -82507fd4ebe9fa656a7f2e084d64a1fa6777a2b0bc106d686e2d9d2edafc58997e58cb6bfd0453b2bf415704aa82ae62 -b40bce2b42b88678400ecd52955bbdadd15f8b9e1b3751a1a3375dc0efb5ca3ee258cf201e1140b3c09ad41217d1d49e -b0210d0cbb3fbf3b8cdb39e862f036b0ff941cd838e7aaf3a8354e24246e64778d22f3de34572e6b2a580614fb6425be -876693cba4301b251523c7d034108831df3ce133d8be5a514e7a2ca494c268ca0556fa2ad8310a1d92a16b55bcd99ea9 -8660281406d22a4950f5ef050bf71dd3090edb16eff27fa29ef600cdea628315e2054211ed2cc6eaf8f2a1771ef689fd -a610e7e41e41ab66955b809ba4ade0330b8e9057d8efc9144753caed81995edeb1a42a53f93ce93540feca1fae708dac -a49e2c176a350251daef1218efaccc07a1e06203386ede59c136699d25ca5cb2ac1b800c25b28dd05678f14e78e51891 -83e0915aa2b09359604566080d411874af8c993beba97d4547782fdbe1a68e59324b800ff1f07b8db30c71adcbd102a8 -a19e84e3541fb6498e9bb8a099c495cbfcad113330e0262a7e4c6544495bb8a754b2208d0c2d895c93463558013a5a32 -87f2bd49859a364912023aca7b19a592c60214b8d6239e2be887ae80b69ebdeb59742bdebcfa73a586ab23b2c945586c -b8e8fdddae934a14b57bc274b8dcd0d45ebb95ddbaabef4454e0f6ce7d3a5a61c86181929546b3d60c447a15134d08e1 -87e0c31dcb736ea4604727e92dc1d9a3cf00adcff79df3546e02108355260f3dd171531c3c0f57be78d8b28058fcc8c0 -9617d74e8f808a4165a8ac2e30878c349e1c3d40972006f0787b31ea62d248c2d9f3fc3da83181c6e57e95feedfd0e8c -8949e2cee582a2f8db86e89785a6e46bc1565c2d8627d5b6bf43ba71ffadfab7e3c5710f88dcb5fb2fc6edf6f4fae216 -ad3fa7b0edceb83118972a2935a09f409d09a8db3869f30be3a76f67aa9fb379cabb3a3aff805ba023a331cad7d7eb64 -8c95718a4112512c4efbd496be38bf3ca6cdcaad8a0d128f32a3f9aae57f3a57bdf295a3b372a8c549fda8f4707cffed -88f3261d1e28a58b2dee3fcc799777ad1c0eb68b3560f9b4410d134672d9533532a91ea7be28a041784872632d3c9d80 -b47472a41d72dd2e8b72f5c4f8ad626737dde3717f63d6bc776639ab299e564cbad0a2ad5452a07f02ff49a359c437e5 -9896d21dc2e8aad87b76d6df1654f10cd7bceed4884159d50a818bea391f8e473e01e14684814c7780235f28e69dca6e -82d47c332bbd31bbe83b5eb44a23da76d4a7a06c45d7f80f395035822bc27f62f59281d5174e6f8e77cc9b5c3193d6f0 -95c74cd46206e7f70c9766117c34c0ec45c2b0f927a15ea167901a160e1530d8522943c29b61e03568aa0f9c55926c53 -a89d7757825ae73a6e81829ff788ea7b3d7409857b378ebccd7df73fdbe62c8d9073741cf038314971b39af6c29c9030 -8c1cd212d0b010905d560688cfc036ae6535bc334fa8b812519d810b7e7dcf1bb7c5f43deaa40f097158358987324a7f -b86993c383c015ed8d847c6b795164114dd3e9efd25143f509da318bfba89389ea72a420699e339423afd68b6512fafb -8d06bd379c6d87c6ed841d8c6e9d2d0de21653a073725ff74be1934301cc3a79b81ef6dd0aad4e7a9dc6eac9b73019bc -81af4d2d87219985b9b1202d724fe39ef988f14fef07dfe3c3b11714e90ffba2a97250838e8535eb63f107abfe645e96 -8c5e0af6330a8becb787e4b502f34f528ef5756e298a77dc0c7467433454347f3a2e0bd2641fbc2a45b95e231c6e1c02 -8e2a8f0f04562820dc8e7da681d5cad9fe2e85dd11c785fb6fba6786c57a857e0b3bd838fb849b0376c34ce1665e4837 -a39be8269449bfdfc61b1f62077033649f18dae9bef7c6163b9314ca8923691fb832f42776f0160b9e8abd4d143aa4e1 -8c154e665706355e1cc98e0a4cabf294ab019545ba9c4c399d666e6ec5c869ca9e1faf8fb06cd9c0a5c2f51a7d51b70a -a046a7d4de879d3ebd4284f08f24398e9e3bf006cd4e25b5c67273ade248689c69affff92ae810c07941e4904296a563 -afd94c1cb48758e5917804df03fb38a6da0e48cd9b6262413ea13b26973f9e266690a1b7d9d24bbaf7e82718e0e594b0 -859e21080310c8d6a38e12e2ac9f90a156578cdeb4bb2e324700e97d9a5511cd6045dc39d1d0de3f94aeed043a24119d -a219fb0303c379d0ab50893264919f598e753aac9065e1f23ef2949abc992577ab43c636a1d2c089203ec9ddb941e27d -b0fdb639d449588a2ca730afcba59334e7c387342d56defdfb7ef79c493f7fd0e5277eff18e7203e756c7bdda5803047 -87f9c3b7ed01f54368aca6dbcf2f6e06bff96e183c4b2c65f8baa23b377988863a0a125d5cdd41a072da8462ced4c070 -99ef7a5d5ac2f1c567160e1f8c95f2f38d41881850f30c461a205f7b1b9fb181277311333839b13fb3ae203447e17727 -aeaca9b1c2afd24e443326cc68de67b4d9cedb22ad7b501a799d30d39c85bb2ea910d4672673e39e154d699e12d9b3dc -a11675a1721a4ba24dd3d0e4c3c33a6edf4cd1b9f6b471070b4386c61f77452266eae6e3f566a40cfc885eada9a29f23 -b228334445e37b9b49cb4f2cc56b454575e92173ddb01370a553bba665adadd52df353ad74470d512561c2c3473c7bb9 -a18177087c996572d76f81178d18ed1ceebc8362a396348ce289f1d8bd708b9e99539be6fccd4acb1112381cfc5749b4 -8e7b8bf460f0d3c99abb19803b9e43422e91507a1c0c22b29ee8b2c52d1a384da4b87c292e28eff040db5be7b1f8641f -b03d038d813e29688b6e6f444eb56fec3abba64c3d6f890a6bcf2e916507091cdb2b9d2c7484617be6b26552ed1c56cb -a1c88ccd30e934adfc5494b72655f8afe1865a84196abfb376968f22ddc07761210b6a9fb7638f1413d1b4073d430290 -961b714faebf172ad2dbc11902461e286e4f24a99a939152a53406117767682a571057044decbeb3d3feef81f4488497 -a03dc4059b46effdd786a0a03cc17cfee8585683faa35bb07936ded3fa3f3a097f518c0b8e2db92fd700149db1937789 -adf60180c99ca574191cbcc23e8d025b2f931f98ca7dfcebfc380226239b6329347100fcb8b0fcb12db108c6ad101c07 -805d4f5ef24d46911cbf942f62cb84b0346e5e712284f82b0db223db26d51aabf43204755eb19519b00e665c7719fcaa -8dea7243e9c139662a7fe3526c6c601eee72fd8847c54c8e1f2ad93ef7f9e1826b170afe58817dac212427164a88e87f -a2ba42356606d651b077983de1ad643650997bb2babb188c9a3b27245bb65d2036e46667c37d4ce02cb1be5ae8547abe -af2ae50b392bdc013db2d12ce2544883472d72424fc767d3f5cb0ca2d973fc7d1f425880101e61970e1a988d0670c81b -98e6bec0568d3939b31d00eb1040e9b8b2a35db46ddf4369bdaee41bbb63cc84423d29ee510a170fb5b0e2df434ba589 -822ff3cd12fbef4f508f3ca813c04a2e0b9b799c99848e5ad3563265979e753ee61a48f6adc2984a850f1b46c1a43d35 -891e8b8b92a394f36653d55725ef514bd2e2a46840a0a2975c76c2a935577f85289026aaa74384da0afe26775cbddfb9 -b2a3131a5d2fe7c8967047aa66e4524babae941d90552171cc109527f345f42aa0df06dcbb2fa01b33d0043917bbed69 -80c869469900431f3eeefafdbe07b8afd8cee7739e659e6d0109b397cacff85a88247698f87dc4e2fe39a592f250ac64 -9091594f488b38f9d2bb5df49fd8b4f8829d9c2f11a197dd1431ed5abbc5c954bbde3387088f9ee3a5a834beb7619bce -b472e241e6956146cca57b97a8a204668d050423b4e76f857bad5b47f43b203a04c8391ba9d9c3e95093c071f9d376a1 -b7dd2de0284844392f7dfb56fe7ca3ede41e27519753ffc579a0a8d2d65ceb8108d06b6b0d4c3c1a2588951297bd1a1e -902116ce70d0a079ac190321c1f48701318c05f8e69ee09694754885d33a835a849cafe56f499a2f49f6cda413ddf9a7 -b18105cc736787fafaf7c3c11c448bce9466e683159dff52723b7951dff429565e466e4841d982e3aaa9ee2066838666 -97ab9911f3f659691762d568ae0b7faa1047b0aed1009c319fa79d15d0db8db9f808fc385dc9a68fa388c10224985379 -b2a2cba65f5b927e64d2904ba412e2bac1cf18c9c3eda9c72fb70262497ecf505b640827e2afebecf10eebbcf48ccd3e -b36a3fd677baa0d3ef0dac4f1548ff50a1730286b8c99d276a0a45d576e17b39b3cbadd2fe55e003796d370d4be43ce3 -a5dfec96ca3c272566e89dc453a458909247e3895d3e44831528130bc47cc9d0a0dac78dd3cad680a4351d399d241967 -8029382113909af6340959c3e61db27392531d62d90f92370a432aec3eb1e4c36ae1d4ef2ba8ec6edb4d7320c7a453f6 -971d85121ea108e6769d54f9c51299b0381ece8b51d46d49c89f65bedc123bab4d5a8bc14d6f67f4f680077529cbae4c -98ff6afc01d0bec80a278f25912e1b1ebff80117adae72e31d5b9fa4d9624db4ba2065b444df49b489b0607c45e26c4c -8fa29be10fb3ab30ce25920fec0187e6e91e458947009dabb869aade7136c8ba23602682b71e390c251f3743164cbdaa -b3345c89eb1653418fe3940cf3e56a9a9c66526389b98f45ca02dd62bfb37baa69a4baaa7132d7320695f8ea6ad1fd94 -b72c7f5541c9ac6b60a7ec9f5415e7fb14da03f7164ea529952a29399f3a071576608dbbcc0d45994f21f92ddbeb1e19 -aa3450bb155a5f9043d0ef95f546a2e6ade167280bfb75c9f09c6f9cdb1fffb7ce8181436161a538433afa3681c7a141 -92a18fecaded7854b349f441e7102b638ababa75b1b0281dd0bded6541abe7aa37d96693595be0b01fe0a2e2133d50f9 -980756ddf9d2253cfe6c94960b516c94889d09e612810935150892627d2ecee9a2517e04968eea295d0106850c04ca44 -ae68c6ccc454318cdd92f32b11d89116a3b8350207a36d22a0f626718cad671d960090e054c0c77ac3162ae180ecfd4b -99f31f66eaaa551749ad91d48a0d4e3ff4d82ef0e8b28f3184c54e852422ba1bdafd53b1e753f3a070f3b55f3c23b6a2 -a44eaeaa6589206069e9c0a45ff9fc51c68da38d4edff1d15529b7932e6f403d12b9387019c44a1488a5d5f27782a51f -b80b5d54d4b344840e45b79e621bd77a3f83fb4ce6d8796b7d6915107b3f3c34d2e7d95bdafd120f285669e5acf2437a -b36c069ec085a612b5908314d6b84c00a83031780261d1c77a0384c406867c9847d5b0845deddfa512cc04a8df2046fb -b09dbe501583220f640d201acea7ee3e39bf9eda8b91aa07b5c50b7641d86d71acb619b38d27835ce97c3759787f08e9 -87403d46a2bf63170fff0b857acacf42ee801afe9ccba8e5b4aea967b68eac73a499a65ca46906c2eb4c8f27bc739faa -82b93669f42a0a2aa5e250ffe6097269da06a9c02fcd1801abbad415a7729a64f830754bafc702e64600ba47671c2208 -8e3a3029be7edb8dd3ab1f8216664c8dc50d395f603736061d802cef77627db7b859ef287ed850382c13b4d22d6a2d80 -968e9ec7194ff424409d182ce0259acd950c384c163c04463bc8700a40b79beba6146d22b7fa7016875a249b7b31c602 -8b42c984bbe4996e0c20862059167c6bdc5164b1ffcd928f29512664459212d263e89f0f0e30eed4e672ffa5ed0b01b5 -96bac54062110dada905363211133f1f15dc7e4fd80a4c6e4a83bc9a0bcbbaba11cd2c7a13debcf0985e1a954c1da66b -a16dc8a653d67a7cd7ae90b2fffac0bf1ca587005430fe5ba9403edd70ca33e38ba5661d2ed6e9d2864400d997626a62 -a68ab11a570a27853c8d67e491591dcba746bfbee08a2e75ae0790399130d027ed387f41ef1d7de8df38b472df309161 -92532b74886874447c0300d07eda9bbe4b41ed25349a3da2e072a93fe32c89d280f740d8ff70d5816793d7f2b97373cc -88e35711b471e89218fd5f4d0eadea8a29405af1cd81974427bc4a5fb26ed60798daaf94f726c96e779b403a2cd82820 -b5c72aa4147c19f8c4f3a0a62d32315b0f4606e0a7025edc5445571eaf4daff64f4b7a585464821574dd50dbe1b49d08 -9305d9b4095258e79744338683fd93f9e657367b3ab32d78080e51d54eec331edbc224fad5093ebf8ee4bd4286757eb8 -b2a17abb3f6a05bcb14dc7b98321fa8b46d299626c73d7c6eb12140bf4c3f8e1795250870947af817834f033c88a59d6 -b3477004837dbd8ba594e4296f960fc91ab3f13551458445e6c232eb04b326da803c4d93e2e8dcd268b4413305ff84da -924b4b2ebaafdcfdfedb2829a8bf46cd32e1407d8d725a5bd28bdc821f1bafb3614f030ea4352c671076a63494275a3f -8b81b9ef6125c82a9bece6fdcb9888a767ac16e70527753428cc87c56a1236e437da8be4f7ecfe57b9296dc3ae7ba807 -906e19ec8b8edd58bdf9ae05610a86e4ea2282b1bbc1e8b00b7021d093194e0837d74cf27ac9916bdb8ec308b00da3da -b41c5185869071760ac786078a57a2ab4e2af60a890037ac0c0c28d6826f15c2cf028fddd42a9b6de632c3d550bfbc14 -a646e5dec1b713ae9dfdf7bdc6cd474d5731a320403c7dfcfd666ffc9ae0cff4b5a79530e8df3f4aa9cb80568cb138e9 -b0efad22827e562bd3c3e925acbd0d9425d19057868608d78c2209a531cccd0f2c43dc5673acf9822247428ffa2bb821 -a94c19468d14b6f99002fc52ac06bbe59e5c472e4a0cdb225144a62f8870b3f10593749df7a2de0bd3c9476ce682e148 -803864a91162f0273d49271dafaab632d93d494d1af935aefa522768af058fce52165018512e8d6774976d52bd797e22 -a08711c2f7d45c68fb340ac23597332e1bcaec9198f72967b9921204b9d48a7843561ff318f87908c05a44fc35e3cc9d -91c3cad94a11a3197ae4f9461faab91a669e0dddb0371d3cab3ed9aeb1267badc797d8375181130e461eadd05099b2a2 -81bdaaf48aae4f7b480fc13f1e7f4dd3023a41439ba231760409ce9292c11128ab2b0bdbbf28b98af4f97b3551f363af -8d60f9df9fd303f625af90e8272c4ecb95bb94e6efc5da17b8ab663ee3b3f673e9f6420d890ccc94acf4d2cae7a860d8 -a7b75901520c06e9495ab983f70b61483504c7ff2a0980c51115d11e0744683ce022d76e3e09f4e99e698cbd21432a0d -82956072df0586562fda7e7738226f694e1c73518dd86e0799d2e820d7f79233667192c9236dcb27637e4c65ef19d493 -a586beb9b6ffd06ad200957490803a7cd8c9bf76e782734e0f55e04a3dc38949de75dc607822ec405736c576cf83bca3 -a179a30d00def9b34a7e85607a447eea0401e32ab5abeee1a281f2acd1cf6ec81a178020666f641d9492b1bdf66f05a3 -83e129705c538787ed8e0fdc1275e6466a3f4ee21a1e6abedd239393b1df72244723b92f9d9d9339a0cab6ebf28f5a16 -811bd8d1e3722b64cd2f5b431167e7f91456e8bba2cc669d3fbbce7d553e29c3c19f629fcedd2498bc26d33a24891d17 -a243c030c858f1f60cccd26b45b024698cc6d9d9e6198c1ed4964a235d9f8d0baf9cde10c8e63dfaa47f8e74e51a6e85 -ab839eb82e23ca52663281f863b55b0a3d6d4425c33ffb4eeb1d7979488ab068bf99e2a60e82cea4dc42c56c26cbfebe -8b896f9bb21d49343e67aec6ad175b58c0c81a3ca73d44d113ae4354a0065d98eb1a5cafedaf232a2bb9cdc62152f309 -af6230340cc0b66f5bf845540ed4fc3e7d6077f361d60762e488d57834c3e7eb7eacc1b0ed73a7d134f174a01410e50c -88975e1b1af678d1b5179f72300a30900736af580dd748fd9461ef7afccc91ccd9bed33f9da55c8711a7635b800e831f -a97486bb9047391661718a54b8dd5a5e363964e495eae6c692730264478c927cf3e66dd3602413189a3699fbeae26e15 -a5973c161ab38732885d1d2785fd74bf156ba34881980cba27fe239caef06b24a533ffe6dbbbeca5e6566682cc00300a -a24776e9a840afda0003fa73b415d5bd6ecd9b5c2cc842b643ee51b8c6087f4eead4d0bfbd987eb174c489a7b952ff2a -a8a6ee06e3af053b705a12b59777267c546f33ba8a0f49493af8e6df4e15cf8dd2d4fb4daf7e84c6b5d3a7363118ff03 -a28e59ce6ad02c2ce725067c0123117e12ac5a52c8f5af13eec75f4a9efc4f696777db18a374fa33bcae82e0734ebd16 -86dfc3b78e841c708aff677baa8ee654c808e5d257158715097c1025d46ece94993efe12c9d188252ad98a1e0e331fec -a88d0275510f242eab11fdb0410ff6e1b9d7a3cbd3658333539815f1b450a84816e6613d15aa8a8eb15d87cdad4b27a2 -8440acea2931118a5b481268ff9f180ee4ede85d14a52c026adc882410825b8275caa44aff0b50c2b88d39f21b1a0696 -a7c3182eab25bd6785bacf12079d0afb0a9b165d6ed327814e2177148539f249eb9b5b2554538f54f3c882d37c0a8abe -85291fbe10538d7da38efdd55a7acebf03b1848428a2f664c3ce55367aece60039f4f320b1771c9c89a35941797f717c -a2c6414eeb1234728ab0de94aa98fc06433a58efa646ca3fcbd97dbfb8d98ae59f7ce6d528f669c8149e1e13266f69c9 -840c8462785591ee93aee2538d9f1ec44ba2ca61a569ab51d335ac873f5d48099ae8d7a7efa0725d9ff8f9475bfa4f56 -a7065a9d02fb3673acf7702a488fbc01aa69580964932f6f40b6c2d1c386b19e50b0e104fcac24ea26c4e723611d0238 -b72db6d141267438279e032c95e6106c2ccb3164b842ba857a2018f3a35f4b040da92680881eb17cd61d0920d5b8f006 -a8005d6c5960e090374747307ef0be2871a7a43fa4e76a16c35d2baab808e9777b496e9f57a4218b23390887c33a0b55 -8e152cea1e00a451ca47c20a1e8875873419700af15a5f38ee2268d3fbc974d4bd5f4be38008fa6f404dbdedd6e6e710 -a3391aed1fcd68761f06a7d1008ec62a09b1cb3d0203cd04e300a0c91adfed1812d8bc1e4a3fd7976dc0aae0e99f52f1 -967eb57bf2aa503ee0c6e67438098149eac305089c155f1762cf5e84e31f0fbf27c34a9af05621e34645c1ec96afaec8 -88af97ddc4937a95ec0dcd25e4173127260f91c8db2f6eac84afb789b363705fb3196235af631c70cafd09411d233589 -a32df75b3f2c921b8767638fd289bcfc61e08597170186637a7128ffedd52c798c434485ac2c7de07014f9e895c2c3d8 -b0a783832153650aa0d766a3a73ec208b6ce5caeb40b87177ffc035ab03c7705ecdd1090b6456a29f5fb7e90e2fa8930 -b59c8e803b4c3486777d15fc2311b97f9ded1602fa570c7b0200bada36a49ee9ef4d4c1474265af8e1c38a93eb66b18b -982f2c85f83e852022998ff91bafbb6ff093ef22cf9d5063e083a48b29175ccbd51b9c6557151409e439096300981a6c -939e3b5989fefebb9d272a954659a4eb125b98c9da6953f5e628d26266bd0525ec38304b8d56f08d65abc4d6da4a8dbb -8898212fe05bc8de7d18503cb84a1c1337cc2c09d1eeef2b475aa79185b7322bf1f8e065f1bf871c0c927dd19faf1f6d -94b0393a41cd00f724aee2d4bc72103d626a5aecb4b5486dd1ef8ac27528398edf56df9db5c3d238d8579af368afeb09 -96ac564450d998e7445dd2ea8e3fc7974d575508fa19e1c60c308d83b645864c029f2f6b7396d4ff4c1b24e92e3bac37 -8adf6638e18aff3eb3b47617da696eb6c4bdfbecbbc3c45d3d0ab0b12cbad00e462fdfbe0c35780d21aa973fc150285e -b53f94612f818571b5565bbb295e74bada9b5f9794b3b91125915e44d6ddcc4da25510eab718e251a09c99534d6042d9 -8b96462508d77ee083c376cd90807aebad8de96bca43983c84a4a6f196d5faf6619a2351f43bfeec101864c3bf255519 -aeadf34657083fc71df33bd44af73bf5281c9ca6d906b9c745536e1819ea90b56107c55e2178ebad08f3ba75b3f81c86 -9784ba29b2f0057b5af1d3ab2796d439b8753f1f749c73e791037461bdfc3f7097394283105b8ab01788ea5255a96710 -8756241bda159d4a33bf74faba0d4594d963c370fb6a18431f279b4a865b070b0547a6d1613cf45b8cfb5f9236bbf831 -b03ebfd6b71421dfd49a30460f9f57063eebfe31b9ceaa2a05c37c61522b35bdc09d7db3ad75c76c253c00ba282d3cd2 -b34e7e6341fa9d854b2d3153bdda0c4ae2b2f442ab7af6f99a0975d45725aa48e36ae5f7011edd249862e91f499687d4 -b462ee09dc3963a14354244313e3444de5cc37ea5ccfbf14cd9aca8027b59c4cb2a949bc30474497cab8123e768460e6 -aea753290e51e2f6a21a9a0ee67d3a2713f95c2a5c17fe41116c87d3aa77b1683761264d704df1ac34f8b873bc88ef7b -98430592afd414394f98ddfff9f280fcb1c322dbe3510f45e1e9c4bb8ee306b3e0cf0282c0ee73ebb8ba087d4d9e0858 -b95d3b5aaf54ffca11f4be8d57f76e14afdb20afc859dc7c7471e0b42031e8f3d461b726ecb979bdb2f353498dfe95ea -984d17f9b11a683132e0b5a9ee5945e3ff7054c2d5c716be73b29078db1d36f54c6e652fd2f52a19da313112e97ade07 -ab232f756b3fff3262be418a1af61a7e0c95ceebbc775389622a8e10610508cd6784ab7960441917a83cc191c58829ea -a28f41678d6e60de76b0e36ab10e4516e53e02e9c77d2b5af3cfeee3ce94cfa30c5797bd1daab20c98e1cad83ad0f633 -b55395fca84dd3ccc05dd480cb9b430bf8631ff06e24cb51d54519703d667268c2f8afcde4ba4ed16bece8cc7bc8c6e0 -8a8a5392a0e2ea3c7a8c51328fab11156004e84a9c63483b64e8f8ebf18a58b6ffa8fe8b9d95af0a2f655f601d096396 -ab480000fe194d23f08a7a9ec1c392334e9c687e06851f083845121ce502c06b54dda8c43092bcc1035df45cc752fe9b -b265644c29f628d1c7e8e25a5e845cabb21799371814730a41a363e1bda8a7be50fee7c3996a365b7fcba4642add10db -b8a915a3c685c2d4728f6931c4d29487cad764c5ce23c25e64b1a3259ac27235e41b23bfe7ae982921b4cb84463097df -8efa7338442a4b6318145a5440fc213b97869647eeae41b9aa3c0a27ee51285b73e3ae3b4a9423df255e6add58864aa9 -9106d65444f74d217f4187dfc8fcf3810b916d1e4275f94f6a86d1c4f3565b131fd6cde1fa708bc05fe183c49f14941a -948252dac8026bbbdb0a06b3c9d66ec4cf9532163bab68076fda1bd2357b69e4b514729c15aaa83b5618b1977bbc60c4 -ae6596ccfdf5cbbc5782efe3bb0b101bb132dbe1d568854ca24cacc0b2e0e9fabcb2ca7ab42aecec412efd15cf8cb7a2 -84a0b6c198ff64fd7958dfd1b40eac9638e8e0b2c4cd8cf5d8cdf80419baee76a05184bce6c5b635f6bf2d30055476a7 -8893118be4a055c2b3da593dbca51b1ae2ea2469911acfb27ee42faf3e6c3ad0693d3914c508c0b05b36a88c8b312b76 -b097479e967504deb6734785db7e60d1d8034d6ca5ba9552887e937f5e17bb413fccac2c1d1082154ed76609127860ad -a0294e6b9958f244d29943debf24b00b538b3da1116269b6e452bb12dc742226712fd1a15b9c88195afeb5d2415f505c -b3cc15f635080bc038f61b615f62b5b5c6f2870586191f59476e8368a73641d6ac2f7d0c1f54621982defdb318020230 -99856f49b9fe1604d917c94d09cc0ed753d13d015d30587a94e6631ffd964b214e607deb8a69a8b5e349a7edf4309206 -a8571e113ea22b4b4fce41a094da8c70de37830ae32e62c65c2fa5ad06a9bc29e884b945e73d448c72b176d6ecebfb58 -a9e9c6e52beb0013273c29844956b3ce291023678107cdc785f7b44eff5003462841ad8780761b86aefc6b734adde7cf -80a784b0b27edb51ef2bad3aee80e51778dcaa0f3f5d3dcb5dc5d4f4b2cf7ae35b08de6680ea9dac53f8438b92eb09ef -827b543e609ea328e97e373f70ad72d4915a2d1daae0c60d44ac637231070e164c43a2a58db80a64df1c624a042b38f9 -b449c65e8195202efdcb9bdb4e869a437313b118fef8b510cbbf8b79a4e99376adb749b37e9c20b51b31ed3310169e27 -8ea3028f4548a79a94c717e1ed28ad4d8725b8d6ab18b021063ce46f665c79da3c49440c6577319dab2d036b7e08f387 -897798431cfb17fe39f08f5f854005dc37b1c1ec1edba6c24bc8acb3b88838d0534a75475325a5ea98b326ad47dbad75 -89cf232e6303b0751561960fd4dea5754a28c594daf930326b4541274ffb03c7dd75938e411eb9a375006a70ce38097f -9727c6ae7f0840f0b6c8bfb3a1a5582ceee705e0b5c59b97def7a7a2283edd4d3f47b7971e902a3a2079e40b53ff69b8 -b76ed72b122c48679d221072efc0eeea063cb205cbf5f9ef0101fd10cb1075b8628166c83577cced654e1c001c7882f7 -ae908c42d208759da5ee9b405df85a6532ea35c6f0f6a1288d22870f59d98edc896841b8ac890a538e6c8d1e8b02d359 -809d12fe4039a0ec80dc9be6a89acaab7797e5f7f9b163378f52f9a75a1d73b2e9ae6e3dd49e32ced439783c1cabbef5 -a4149530b7f85d1098ba534d69548c6c612c416e8d35992fc1f64f4deeb41e09e49c6cf7aadbed7e846b91299358fe2d -a49342eacd1ec1148b8df1e253b1c015f603c39de11fa0a364ccb86ea32d69c34fd7aa6980a1fadcd8e785a57fa46f60 -87d43eff5a006dc4dddcf76cc96c656a1f3a68f19f124181feab86c6cc9a52cb9189cdbb423414defdd9bb0ca8ff1ddc -861367e87a9aa2f0f68296ba50aa5dbc5713008d260cc2c7e62d407c2063064749324c4e8156dc21b749656cfebce26b -b5303c2f72e84e170e66ae1b0fbd51b8c7a6f27476eaf5694b64e8737d5c84b51fe90100b256465a4c4156dd873cddb0 -b62849a4f891415d74f434cdc1d23c4a69074487659ca96e1762466b2b7a5d8525b056b891d0feea6fe6845cba8bc7fb -923dd9e0d6590a9307e8c4c23f13bae3306b580e297a937711a8b13e8de85e41a61462f25b7d352b682e8437bf2b4ab3 -9147379860cd713cd46c94b8cdf75125d36c37517fbecf81ace9680b98ce6291cd1c3e472f84249cc3b2b445e314b1b6 -a808a4f17ac21e3fb5cfef404e61fae3693ca3e688d375f99b6116779696059a146c27b06de3ac36da349b0649befd56 -87787e9322e1b75e66c1f0d9ea0915722a232770930c2d2a95e9478c4b950d15ab767e30cea128f9ed65893bfc2d0743 -9036a6ee2577223be105defe1081c48ea7319e112fff9110eb9f61110c319da25a6cea0464ce65e858635b079691ef1f -af5548c7c24e1088c23b57ee14d26c12a83484c9fd9296edf1012d8dcf88243f20039b43c8c548c265ef9a1ffe9c1c88 -a0fff520045e14065965fb8accd17e878d3fcaf9e0af2962c8954e50be6683d31fa0bf4816ab68f08630dbac6bfce52a -b4c1b249e079f6ae1781af1d97a60b15855f49864c50496c09c91fe1946266915b799f0406084d7783f5b1039116dd8b -8b0ffa5e7c498cb3879dddca34743b41eee8e2dea3d4317a6e961b58adb699ef0c92400c068d5228881a2b08121226bf -852ae8b19a1d80aa8ae5382e7ee5c8e7670ceb16640871c56b20b96b66b3b60e00015a3dde039446972e57b49a999ddd -a49942f04234a7d8492169da232cfff8051df86e8e1ba3db46aede02422c689c87dc1d99699c25f96cb763f5ca0983e5 -b04b597b7760cf5dcf411ef896d1661e6d5b0db3257ac2cf64b20b60c6cc18fa10523bb958a48d010b55bac7b02ab3b1 -a494591b51ea8285daecc194b5e5bd45ae35767d0246ac94fae204d674ee180c8e97ff15f71f28b7aeb175b8aea59710 -97d2624919e78406e7460730680dea8e71c8571cf988e11441aeea54512b95bd820e78562c99372d535d96f7e200d20d -ac693ddb00e48f76e667243b9b6a7008424043fb779e4f2252330285232c3fccac4da25cbd6d95fe9ad959ff305a91f6 -8d20ca0a71a64a3f702a0825bb46bd810d03bebfb227683680d474a52f965716ff99e19a165ebaf6567987f4f9ee3c94 -a5c516a438f916d1d68ca76996404792e0a66e97b7f18fc54c917bf10cf3211b62387932756e39e67e47b0bd6e88385a -b089614d830abc0afa435034cec7f851f2f095d479cacf1a3fb57272da826c499a52e7dcbc0eb85f4166fb94778e18e9 -a8dacc943765d930848288192f4c69e2461c4b9bc6e79e30eeef9a543318cf9ae9569d6986c65c5668a89d49993f8e07 -ab5a9361fa339eec8c621bdad0a58078983abd8942d4282b22835d7a3a47e132d42414b7c359694986f7db39386c2e19 -94230517fb57bd8eb26c6f64129b8b2abd0282323bf7b94b8bac7fab27b4ecc2c4290c294275e1a759de19f2216134f3 -b8f158ea5006bc3b90b285246625faaa6ac9b5f5030dc69701b12f3b79a53ec7e92eeb5a63bbd1f9509a0a3469ff3ffc -8b6944fd8cb8540957a91a142fdcda827762aa777a31e8810ca6d026e50370ee1636fc351724767e817ca38804ebe005 -82d1ee40fe1569c29644f79fa6c4033b7ed45cd2c3b343881f6eb0de2e79548fded4787fae19bed6ee76ed76ff9f2f11 -a8924c7035e99eaed244ca165607e7e568b6c8085510dcdbaf6ebdbed405af2e6c14ee27d94ffef10d30aa52a60bf66d -956f82a6c2ae044635e85812581e4866c5fa2f427b01942047d81f6d79a14192f66fbbe77c9ffeaef4e6147097fdd2b5 -b1100255a1bcf5e05b6aff1dfeb6e1d55b5d68d43a7457ba10cc76b61885f67f4d0d5179abda786e037ae95deb8eea45 -99510799025e3e5e8fbf06dedb14c060c6548ba2bda824f687d3999dc395e794b1fb6514b9013f3892b6cf65cb0d65aa -8f9091cebf5e9c809aab415942172258f894e66e625d7388a05289183f01b8d994d52e05a8e69f784fba41db9ea357f0 -a13d2eeb0776bdee9820ecb6693536720232848c51936bb4ef4fe65588d3f920d08a21907e1fdb881c1ad70b3725e726 -a68b8f18922d550284c5e5dc2dda771f24c21965a6a4d5e7a71678178f46df4d8a421497aad8fcb4c7e241aba26378a0 -8b7601f0a3c6ad27f03f2d23e785c81c1460d60100f91ea9d1cab978aa03b523150206c6d52ce7c7769c71d2c8228e9e -a8e02926430813caa851bb2b46de7f0420f0a64eb5f6b805401c11c9091d3b6d67d841b5674fa2b1dce0867714124cd8 -b7968ecba568b8193b3058400af02c183f0a6df995a744450b3f7e0af7a772454677c3857f99c140bbdb2a09e832e8e0 -8f20b1e9ba87d0a3f35309b985f3c18d2e8800f1ca7f0c52cadef773f1496b6070c936eea48c4a1cae83fd2524e9d233 -88aef260042db0d641a51f40639dbeeefa9e9811df30bee695f3791f88a2f84d318f04e8926b7f47bf25956cb9e3754f -9725345893b647e9ba4e6a29e12f96751f1ae25fcaec2173e9a259921a1a7edb7a47159b3c8767e44d9e2689f5aa0f72 -8c281e6f72752cb11e239e4df9341c45106eb7993c160e54423c2bffe10bc39d42624b45a1f673936ef2e1a02fc92f1a -90aba2f68bddb2fcce6c51430dacdfeec43ea8dc379660c99095df11017691ccf5faa27665cf4b9f0eea7728ae53c327 -b7022695c16521c5704f49b7ddbdbec9b5f57ce0ceebe537bc0ebb0906d8196cc855a9afeb8950a1710f6a654464d93f -8fe1b9dd3c6a258116415d36e08374e094b22f0afb104385a5da48be17123e86fb8327baacc4f0d9ebae923d55d99bb5 -817e85d8e3d19a4cbc1dec31597142c2daa4871bda89c2177fa719c00eda3344eb08b82eb92d4aa91a9eaacb3fc09783 -b59053e1081d2603f1ca0ba553804d6fa696e1fd996631db8f62087b26a40dfef02098b0326bb75f99ec83b9267ca738 -990a173d857d3ba81ff3789b931bfc9f5609cde0169b7f055fa3cb56451748d593d62d46ba33f80f9cafffe02b68dd14 -b0c538dbba4954b809ab26f9f94a3cf1dcb77ce289eaec1d19f556c0ae4be1fa03af4a9b7057837541c3cc0a80538736 -ac3ba42f5f44f9e1fc453ce49c4ab79d0e1d5c42d3b30b1e098f3ab3f414c4c262fa12fb2be249f52d4aaf3c5224beb9 -af47467eb152e59870e21f0d4da2f43e093daf40180ab01438030684b114d025326928eaab12c41b81a066d94fce8436 -98d1b58ba22e7289b1c45c79a24624f19b1d89e00f778eef327ec4856a9a897278e6f1a9a7e673844b31dde949153000 -97ccb15dfadc7c59dca08cfe0d22df2e52c684cf97de1d94bc00d7ba24e020025130b0a39c0f4d46e4fc872771ee7875 -b699e4ed9a000ff96ca296b2f09dce278832bc8ac96851ff3cff99ed3f6f752cfc0fea8571be28cd9b5a7ec36f1a08ee -b9f49f0edb7941cc296435ff0a912e3ad16848ee8765ab5f60a050b280d6ea585e5b34051b15f6b8934ef01ceb85f648 -ac3893df7b4ceab23c6b9054e48e8ba40d6e5beda8fbe90b814f992f52494186969b35d8c4cdc3c99890a222c9c09008 -a41293ad22fae81dea94467bc1488c3707f3d4765059173980be93995fa4fcc3c9340796e3eed0beeb0ba0d9bb4fa3aa -a0543e77acd2aeecde13d18d258aeb2c7397b77f17c35a1992e8666ea7abcd8a38ec6c2741bd929abba2f766138618cc -92e79b22bc40e69f6527c969500ca543899105837b6b1075fa1796755c723462059b3d1b028e0b3df2559fa440e09175 -a1fa1eac8f41a5197a6fb4aa1eae1a031c89f9c13ff9448338b222780cf9022e0b0925d930c37501a0ef7b2b00fdaf83 -b3cb29ff73229f0637335f28a08ad8c5f166066f27c6c175164d0f26766a927f843b987ee9b309ed71cbf0a65d483831 -84d4ab787f0ac00f104f4a734dc693d62d48c2aeb03913153da62c2ae2c27d11b1110dcef8980368dd84682ea2c1a308 -ab6a8e4bbc78d4a7b291ad3e9a8fe2d65f640524ba3181123b09d2d18a9e300e2509ccf7000fe47e75b65f3e992a2e7e -b7805ebe4f1a4df414003dc10bca805f2ab86ca75820012653e8f9b79c405196b0e2cab099f2ab953d67f0d60d31a0f9 -b12c582454148338ea605d22bd00a754109063e22617f1f8ac8ddf5502c22a181c50c216c3617b9852aa5f26af56b323 -86333ad9f898947e31ce747728dc8c887479e18d36ff3013f69ebef807d82c6981543b5c3788af93c4d912ba084d3cba -b514efa310dc4ad1258add138891e540d8c87142a881b5f46563cc58ecd1488e6d3a2fca54c0b72a929f3364ca8c333e -aa0a30f92843cf2f484066a783a1d75a7aa6f41f00b421d4baf20a6ac7886c468d0eea7ca8b17dd22f4f74631b62b640 -b3b7dc63baec9a752e8433c0cdee4d0f9bc41f66f2b8d132faf925eef9cf89aae756fc132c45910f057122462605dc10 -b9b8190dac5bfdeb59fd44f4da41a57e7f1e7d2c21faba9da91fa45cbeca06dcf299c9ae22f0c89ece11ac46352d619f -89f8cf36501ad8bdfeab863752a9090e3bfda57cf8fdeca2944864dc05925f501e252c048221bcc57136ab09a64b64b2 -b0cbfaf317f05f97be47fc9d69eda2dd82500e00d42612f271a1fe24626408c28881f171e855bd5bd67409f9847502b4 -a7c21a8fcede581bfd9847b6835eda62ba250bea81f1bb17372c800a19c732abe03064e64a2f865d974fb636cab4b859 -95f9df524ba7a4667351696c4176b505d8ea3659f5ff2701173064acc624af69a0fad4970963736383b979830cb32260 -856a74fe8b37a2e3afeac858c8632200485d438422a16ae3b29f359e470e8244995c63ad79c7e007ed063f178d0306fd -b37faa4d78fdc0bb9d403674dbea0176c2014a171c7be8527b54f7d1a32a76883d3422a3e7a5f5fcc5e9b31b57822eeb -8d37234d8594ec3fe75670b5c9cc1ec3537564d4739b2682a75b18b08401869a4264c0f264354219d8d896cded715db4 -b5289ee5737f0e0bde485d32096d23387d68dab8f01f47821ab4f06cc79a967afe7355e72dc0c751d96b2747b26f6255 -9085e1fdf9f813e9c3b8232d3c8863cd84ab30d45e8e0d3d6a0abd9ebc6fd70cdf749ff4d04390000e14c7d8c6655fc7 -93a388c83630331eca4da37ea4a97b3b453238af474817cc0a0727fd3138dcb4a22de38c04783ec829c22cb459cb4e8e -a5377116027c5d061dbe24c240b891c08cdd8cd3f0899e848d682c873aff5b8132c1e7cfe76d2e5ed97ee0eb1d42cb68 -a274c84b04338ed28d74683e2a7519c2591a3ce37c294d6f6e678f7d628be2db8eff253ede21823e2df7183e6552f622 -8bc201147a842453a50bec3ac97671397bc086d6dfc9377fa38c2124cdc286abda69b7324f47d64da094ae011d98d9d9 -9842d0c066c524592b76fbec5132bc628e5e1d21c424bec4555efca8619cc1fd8ea3161febcb8b9e8ab54702f4e815e2 -a19191b713a07efe85c266f839d14e25660ee74452e6c691cd9997d85ae4f732052d802d3deb018bdd847caa298a894b -a24f71fc0db504da4e287dd118a4a74301cbcd16033937ba2abc8417956fcb4ae19b8e63b931795544a978137eff51cb -a90eec4a6a3a4b8f9a5b93d978b5026fcf812fe65585b008d7e08c4aaf21195a1d0699f12fc16f79b6a18a369af45771 -8b551cf89737d7d06d9b3b9c4c1c73b41f2ea0af4540999c70b82dabff8580797cf0a3caf34c86c59a7069eb2e38f087 -b8d312e6c635e7a216a1cda075ae77ba3e1d2fd501dc31e83496e6e81ed5d9c7799f8e578869c2e0e256fb29f5de10a7 -8d144bdb8cae0b2cdb5b33d44bbc96984a5925202506a8cc65eb67ac904b466f5a7fe3e1cbf04aa785bbb7348c4bb73c -a101b3d58b7a98659244b88de0b478b3fb87dc5fc6031f6e689b99edf498abd43e151fd32bd4bbd240e0b3e59c440359 -907453abca7d8e7151a05cc3d506c988007692fe7401395dc93177d0d07d114ab6cca0cc658eb94c0223fe8658295cad -825329ffbe2147ddb68f63a0a67f32d7f309657b8e5d9ab5bb34b3730bfa2c77a23eaaadb05def7d9f94a9e08fdc1e96 -88ee923c95c1dac99ae7ed6067906d734d793c5dc5d26339c1bb3314abe201c5dccb33b9007351885eb2754e9a8ea06c -98bc9798543f5f1adc9f2cfcfa72331989420e9c3f6598c45269f0dc9b7c8607bbeaf03faa0aea2ddde2b8f17fdceff5 -8ee87877702a79aef923ab970db6fa81561b3c07d5bf1a072af0a7bad765b4cbaec910afe1a91703feacc7822fa38a94 -8060b9584aa294fe8adc2b22f67e988bc6da768eae91e429dcc43ddc53cfcc5d6753fdc1b420b268c7eb2fb50736a970 -b344a5524d80a2f051870c7001f74fcf348a70fcf78dbd20c6ff9ca85d81567d2318c8b8089f2c4f195d6aec9fc15fa6 -8f5a5d893e1936ed062149d20eb73d98b62b7f50ab5d93a6429c03656b36688d1c80cb5010e4977491e51fa0d7dd35d5 -86fa32ebbf97328c5f5f15564e1238297e289ec3219b9a741724e9f3ae8d5c15277008f555863a478b247ba5dc601d44 -9557e55377e279f4b6b5e0ffe01eca037cc13aac242d67dfcd0374a1e775c5ed5cb30c25fe21143fee54e3302d34a3ea -8cb6bcbc39372d23464a416ea7039f57ba8413cf3f00d9a7a5b356ab20dcb8ed11b3561f7bce372b8534d2870c7ee270 -b5d59075cb5abde5391f64b6c3b8b50adc6e1f654e2a580b6d6d6eff3f4fbdd8fffc92e06809c393f5c8eab37f774c4b -afcfb6903ef13e493a1f7308675582f15af0403b6553e8c37afb8b2808ad21b88b347dc139464367dc260df075fea1ad -810fbbe808375735dd22d5bc7fc3828dc49fdd22cc2d7661604e7ac9c4535c1df578780affb3b895a0831640a945bcad -8056b0c678803b416f924e09a6299a33cf9ad7da6fe1ad7accefe95c179e0077da36815fde3716711c394e2c5ea7127f -8b67403702d06979be19f1d6dc3ec73cc2e81254d6b7d0cc49cd4fdda8cd51ab0835c1d2d26fc0ecab5df90585c2f351 -87f97f9e6d4be07e8db250e5dd2bffdf1390665bc5709f2b631a6fa69a7fca958f19bd7cc617183da1f50ee63e9352b5 -ae151310985940471e6803fcf37600d7fa98830613e381e00dab943aec32c14162d51c4598e8847148148000d6e5af5c -81eb537b35b7602c45441cfc61b27fa9a30d3998fad35a064e05bc9479e9f10b62eba2b234b348219eea3cadcaac64bb -8a441434934180ab6f5bc541f86ebd06eadbee01f438836d797e930fa803a51510e005c9248cecc231a775b74d12b5e9 -81f3c250a27ba14d8496a5092b145629eb2c2e6a5298438670375363f57e2798207832c8027c3e9238ad94ecdadfc4df -a6217c311f2f3db02ceaa5b6096849fe92b6f4b6f1491535ef8525f6ccee6130bed2809e625073ecbaddd4a3eb3df186 -82d1c396f0388b942cf22b119d7ef1ad03d3dad49a74d9d01649ee284f377c8daddd095d596871669e16160299a210db -a40ddf7043c5d72a7246bd727b07f7fff1549f0e443d611de6f9976c37448b21664c5089c57f20105102d935ab82f27b -b6c03c1c97adf0c4bf4447ec71366c6c1bff401ba46236cd4a33d39291e7a1f0bb34bd078ba3a18d15c98993b153a279 -8a94f5f632068399c359c4b3a3653cb6df2b207379b3d0cdace51afdf70d6d5cce6b89a2b0fee66744eba86c98fb21c2 -b2f19e78ee85073f680c3bba1f07fd31b057c00b97040357d97855b54a0b5accb0d3b05b2a294568fcd6a4be6f266950 -a74632d13bbe2d64b51d7a9c3ae0a5a971c19f51cf7596a807cea053e6a0f3719700976d4e394b356c0329a2dced9aa2 -afef616d341a9bc94393b8dfba68ff0581436aa3a3adb7c26a1bbf2cf19fa877066191681f71f17f3cd6f9cf6bf70b5a -8ce96d93ae217408acf7eb0f9cbb9563363e5c7002e19bbe1e80760bc9d449daee2118f3878b955163ed664516b97294 -8414f79b496176bc8b8e25f8e4cfee28f4f1c2ddab099d63d2aca1b6403d26a571152fc3edb97794767a7c4686ad557c -b6c61d01fd8ce087ef9f079bf25bf10090db483dd4f88c4a786d31c1bdf52065651c1f5523f20c21e75cea17df69ab73 -a5790fd629be70545093631efadddc136661f63b65ec682609c38ef7d3d7fa4e56bdf94f06e263bc055b90cb1c6bcefe -b515a767e95704fb7597bca9e46f1753abacdc0e56e867ee3c6f4cd382643c2a28e65312c05ad040eaa3a8cbe7217a65 -8135806a02ead6aa92e9adb6fefb91349837ab73105aaa7be488ef966aa8dfaafdfa64bbae30fcbfa55dd135a036a863 -8f22435702716d76b1369750694540742d909d5e72b54d0878245fab7c269953b1c6f2b29c66f08d5e0263ca3a731771 -8e0f8a8e8753e077dac95848212aeffd51c23d9b6d611df8b102f654089401954413ecbedc6367561ca599512ae5dda7 -815a9084e3e2345f24c5fa559deec21ee1352fb60f4025c0779be65057f2d528a3d91593bd30d3a185f5ec53a9950676 -967e6555ccba395b2cc1605f8484c5112c7b263f41ce8439a99fd1c71c5ed14ad02684d6f636364199ca48afbbde13be -8cd0ccf17682950b34c796a41e2ea7dd5367aba5e80a907e01f4cdc611e4a411918215e5aebf4292f8b24765d73314a6 -a58bf1bbb377e4b3915df6f058a0f53b8fb8130fdec8c391f6bc82065694d0be59bb67ffb540e6c42cc8b380c6e36359 -92af3151d9e6bfb3383d85433e953c0160859f759b0988431ec5893542ba40288f65db43c78a904325ef8d324988f09d -8011bbb05705167afb47d4425065630f54cb86cd462095e83b81dfebf348f846e4d8fbcf1c13208f5de1931f81da40b9 -81c743c104fc3cb047885c9fa0fb9705c3a83ee24f690f539f4985509c3dafd507af3f6a2128276f45d5939ef70c167f -a2c9679b151c041aaf5efeac5a737a8f70d1631d931609fca16be1905682f35e291292874cb3b03f14994f98573c6f44 -a4949b86c4e5b1d5c82a337e5ce6b2718b1f7c215148c8bfb7e7c44ec86c5c9476048fc5c01f57cb0920876478c41ad6 -86c2495088bd1772152e527a1da0ef473f924ea9ab0e5b8077df859c28078f73c4e22e3a906b507fdf217c3c80808b5c -892e0a910dcf162bcea379763c3e2349349e4cda9402949255ac4a78dd5a47e0bf42f5bd0913951576b1d206dc1e536a -a7009b2c6b396138afe4754b7cc10dee557c51c7f1a357a11486b3253818531f781ea8107360c8d4c3b1cd96282353c0 -911763ef439c086065cc7b4e57484ed6d693ea44acee4b18c9fd998116da55fbe7dcb8d2a0f0f9b32132fca82d73dff6 -a722000b95a4a2d40bed81870793f15ba2af633f9892df507f2842e52452e02b5ea8dea6a043c2b2611d82376e33742a -9387ac49477bd719c2f92240d0bdfcf9767aad247ca93dc51e56106463206bc343a8ec855eb803471629a66fffb565d6 -92819a1fa48ab4902939bb72a0a4e6143c058ea42b42f9bc6cea5df45f49724e2530daf3fc4f097cceefa2a8b9db0076 -98eac7b04537653bc0f4941aae732e4b1f84bd276c992c64a219b8715eb1fb829b5cbd997d57feb15c7694c468f95f70 -b275e7ba848ce21bf7996e12dbeb8dadb5d0e4f1cb5a0248a4f8f9c9fe6c74e3c93f4b61edbcb0a51af5a141e1c14bc7 -97243189285aba4d49c53770c242f2faf5fd3914451da4931472e3290164f7663c726cf86020f8f181e568c72fd172d1 -839b0b3c25dd412bee3dc24653b873cc65454f8f16186bb707bcd58259c0b6765fa4c195403209179192a4455c95f3b8 -8689d1a870514568a074a38232e2ceb4d7df30fabeb76cff0aed5b42bf7f02baea12c5fadf69f4713464dbd52aafa55f -8958ae7b290f0b00d17c3e9fdb4dbf168432b457c7676829299dd428984aba892de1966fc106cfc58a772862ecce3976 -a422bc6bd68b8870cfa5bc4ce71781fd7f4368b564d7f1e0917f6013c8bbb5b240a257f89ecfdbecb40fe0f3aa31d310 -aa61f78130cebe09bc9a2c0a37f0dd57ed2d702962e37d38b1df7f17dc554b1d4b7a39a44182a452ce4c5eb31fa4cfcc -b7918bd114f37869bf1a459023386825821bfadce545201929d13ac3256d92a431e34f690a55d944f77d0b652cefeffc -819bba35fb6ace1510920d4dcff30aa682a3c9af9022e287751a6a6649b00c5402f14b6309f0aeef8fce312a0402915e -8b7c9ad446c6f63c11e1c24e24014bd570862b65d53684e107ba9ad381e81a2eaa96731b4b33536efd55e0f055071274 -8fe79b53f06d33386c0ec7d6d521183c13199498594a46d44a8a716932c3ec480c60be398650bbfa044fa791c4e99b65 -9558e10fb81250b9844c99648cf38fa05ec1e65d0ccbb18aa17f2d1f503144baf59d802c25be8cc0879fff82ed5034ad -b538a7b97fbd702ba84645ca0a63725be1e2891c784b1d599e54e3480e4670d0025526674ef5cf2f87dddf2290ba09f0 -92eafe2e869a3dd8519bbbceb630585c6eb21712b2f31e1b63067c0acb5f9bdbbcbdb612db4ea7f9cc4e7be83d31973f -b40d21390bb813ab7b70a010dff64c57178418c62685761784e37d327ba3cb9ef62df87ecb84277c325a637fe3709732 -b349e6fbf778c4af35fbed33130bd8a7216ed3ba0a79163ebb556e8eb8e1a7dad3456ddd700dad9d08d202491c51b939 -a8fdaedecb251f892b66c669e34137f2650509ade5d38fbe8a05d9b9184bb3b2d416186a3640429bd1f3e4b903c159dd -ac6167ebfee1dbab338eff7642f5e785fc21ef0b4ddd6660333fe398068cbd6c42585f62e81e4edbb72161ce852a1a4f -874b1fbf2ebe140c683bd7e4e0ab017afa5d4ad38055aaa83ee6bbef77dbc88a6ce8eb0dcc48f0155244af6f86f34c2d -903c58e57ddd9c446afab8256a6bb6c911121e6ccfb4f9b4ed3e2ed922a0e500a5cb7fa379d5285bc16e11dac90d1fda -8dae7a0cffa2fd166859cd1bf10ff82dd1932e488af377366b7efc0d5dec85f85fe5e8150ff86a79a39cefc29631733a -aa047857a47cc4dfc08585f28640420fcf105b881fd59a6cf7890a36516af0644d143b73f3515ab48faaa621168f8c31 -864508f7077c266cc0cb3f7f001cb6e27125ebfe79ab57a123a8195f2e27d3799ff98413e8483c533b46a816a3557f1f -8bcd45ab1f9cbab36937a27e724af819838f66dfeb15923f8113654ff877bd8667c54f6307aaf0c35027ca11b6229bfd -b21aa34da9ab0a48fcfdd291df224697ce0c1ebc0e9b022fdee8750a1a4b5ba421c419541ed5c98b461eecf363047471 -a9a18a2ab2fae14542dc336269fe612e9c1af6cf0c9ac933679a2f2cb77d3c304114f4d219ca66fe288adde30716775b -b5205989b92c58bdda71817f9a897e84100b5c4e708de1fced5c286f7a6f01ae96b1c8d845f3a320d77c8e2703c0e8b1 -a364059412bbcc17b8907d43ac8e5df90bc87fd1724b5f99832d0d24559fae6fa76a74cff1d1eac8cbac6ec80b44af20 -ae709f2c339886b31450834cf29a38b26eb3b0779bd77c9ac269a8a925d1d78ea3837876c654b61a8fe834b3b6940808 -8802581bba66e1952ac4dab36af371f66778958f4612901d95e5cac17f59165e6064371d02de8fb6fccf89c6dc8bd118 -a313252df653e29c672cbcfd2d4f775089cb77be1077381cf4dc9533790e88af6cedc8a119158e7da5bf6806ad9b91a1 -992a065b4152c7ef11515cd54ba9d191fda44032a01aed954acff3443377ee16680c7248d530b746b8c6dee2d634e68c -b627b683ee2b32c1ab4ccd27b9f6cce2fe097d96386fa0e5c182ad997c4c422ab8dfc03870cd830b8c774feb66537282 -b823cf8a9aee03dadd013eb9efe40a201b4b57ef67efaae9f99683005f5d1bf55e950bf4af0774f50859d743642d3fea -b8a7449ffac0a3f206677097baf7ce00ca07a4d2bd9b5356fbcb83f3649b0fda07cfebad220c1066afba89e5a52abf4b -b2dd1a2f986395bb4e3e960fbbe823dbb154f823284ebc9068502c19a7609790ec0073d08bfa63f71e30c7161b6ef966 -98e5236de4281245234f5d40a25b503505af140b503a035fc25a26159a9074ec81512b28f324c56ea2c9a5aa7ce90805 -89070847dc8bbf5bc4ed073aa2e2a1f699cf0c2ca226f185a0671cecc54e7d3e14cd475c7752314a7a8e7476829da4bc -a9402dc9117fdb39c4734c0688254f23aed3dce94f5f53f5b7ef2b4bf1b71a67f85ab1a38ec224a59691f3bee050aeb3 -957288f9866a4bf56a4204218ccc583f717d7ce45c01ea27142a7e245ad04a07f289cc044f8cf1f21d35e67e39299e9c -b2fb31ccb4e69113763d7247d0fc8edaae69b550c5c56aecacfd780c7217dc672f9fb7496edf4aba65dacf3361268e5b -b44a4526b2f1d6eb2aa8dba23bfa385ff7634572ab2afddd0546c3beb630fbfe85a32f42dd287a7fec069041411537f7 -8db5a6660c3ac7fd7a093573940f068ee79a82bc17312af900b51c8c439336bc86ca646c6b7ab13aaaa008a24ca508ab -8f9899a6d7e8eb4367beb5c060a1f8e94d8a21099033ae582118477265155ba9e72176a67f7f25d7bad75a152b56e21a -a67de0e91ade8d69a0e00c9ff33ee2909b8a609357095fa12319e6158570c232e5b6f4647522efb7345ce0052aa9d489 -82eb2414898e9c3023d57907a2b17de8e7eea5269029d05a94bfd7bf5685ac4a799110fbb375eb5e0e2bd16acf6458ae -94451fc7fea3c5a89ba701004a9693bab555cb622caf0896b678faba040409fdfd14a978979038b2a81e8f0abc4994d2 -ac879a5bb433998e289809a4a966bd02b4bf6a9c1cc276454e39c886efcf4fc68baebed575826bde577ab5aa71d735a9 -880c0f8f49c875dfd62b4ddedde0f5c8b19f5687e693717f7e5c031bc580e58e13ab497d48b4874130a18743c59fdce3 -b582af8d8ff0bf76f0a3934775e0b54c0e8fed893245d7d89cae65b03c8125b7237edc29dc45b4fe1a3fe6db45d280ee -89f337882ed3ae060aaee98efa20d79b6822bde9708c1c5fcee365d0ec9297f694cae37d38fd8e3d49717c1e86f078e7 -826d2c1faea54061848b484e288a5f4de0d221258178cf87f72e14baaa4acc21322f8c9eab5dde612ef497f2d2e1d60b -a5333d4f227543e9cd741ccf3b81db79f2f03ca9e649e40d6a6e8ff9073e06da83683566d3b3c8d7b258c62970fb24d1 -a28f08c473db06aaf4c043a2fae82b3c8cfaa160bce793a4c208e4e168fb1c65115ff8139dea06453c5963d95e922b94 -8162546135cc5e124e9683bdfaa45833c18553ff06a0861c887dc84a5b12ae8cd4697f6794c7ef6230492c32faba7014 -b23f0d05b74c08d6a7df1760792be83a761b36e3f8ae360f3c363fb196e2a9dd2de2e492e49d36561366e14daa77155c -b6f70d6c546722d3907c708d630dbe289771d2c8bf059c2e32b77f224696d750b4dda9b3a014debda38e7d02c9a77585 -83bf4c4a9f3ca022c631017e7a30ea205ba97f7f5927cba8fc8489a4646eac6712cb821c5668c9ffe94d69d524374a27 -b0371475425a8076d0dd5f733f55aabbe42d20a7c8ea7da352e736d4d35a327b2beb370dfcb05284e22cfd69c5f6c4cc -a0031ba7522c79211416c2cca3aa5450f96f8fee711552a30889910970ba13608646538781a2c08b834b140aadd7166f -99d273c80c7f2dc6045d4ed355d9fc6f74e93549d961f4a3b73cd38683f905934d359058cd1fc4da8083c7d75070487f -b0e4b0efa3237793e9dcce86d75aafe9879c5fa23f0d628649aef2130454dcf72578f9bf227b9d2b9e05617468e82588 -a5ab076fa2e1c5c51f3ae101afdd596ad9d106bba7882b359c43d8548b64f528af19afa76cd6f40da1e6c5fca4def3fa -8ce2299e570331d60f6a6eff1b271097cd5f1c0e1113fc69b89c6a0f685dabea3e5bc2ac6bd789aa492ab189f89be494 -91b829068874d911a310a5f9dee001021f97471307b5a3de9ec336870ec597413e1d92010ce320b619f38bed7c4f7910 -b14fe91f4b07bf33b046e9285b66cb07927f3a8da0af548ac2569b4c4fb1309d3ced76d733051a20814e90dd5b75ffd1 -abaab92ea6152d40f82940277c725aa768a631ee0b37f5961667f82fb990fc11e6d3a6a2752b0c6f94563ed9bb28265c -b7fe28543eca2a716859a76ab9092f135337e28109544f6bd2727728d0a7650428af5713171ea60bfc273d1c821d992c -8a4917b2ab749fc7343fc64bdf51b6c0698ff15d740cc7baf248c030475c097097d5a473bcc00d8c25817563fe0447b4 -aa96156d1379553256350a0a3250166add75948fb9cde62aa555a0a9dc0a9cb7f2f7b8428aff66097bf6bfedaf14bbe2 -ae4ffeb9bdc76830d3eca2b705f30c1bdede6412fa064260a21562c8850c7fb611ec62bc68479fe48f692833e6f66d8d -b96543caaba9d051600a14997765d49e4ab10b07c7a92cccf0c90b309e6da334fdd6d18c96806cbb67a7801024fbd3c7 -97b2b9ad76f19f500fcc94ca8e434176249f542ac66e5881a3dccd07354bdab6a2157018b19f8459437a68d8b86ba8e0 -a8d206f6c5a14c80005849474fde44b1e7bcf0b2d52068f5f97504c3c035b09e65e56d1cf4b5322791ae2c2fdbd61859 -936bad397ad577a70cf99bf9056584a61bd7f02d2d5a6cf219c05d770ae30a5cd902ba38366ce636067fc1dd10108d31 -a77e30195ee402b84f3882e2286bf5380c0ed374a112dbd11e16cef6b6b61ab209d4635e6f35cdaaa72c1a1981d5dabe -a46ba4d3947188590a43c180757886a453a0503f79cc435322d92490446f37419c7b999fdf868a023601078070e03346 -80d8d4c5542f223d48240b445d4d8cf6a75d120b060bc08c45e99a13028b809d910b534d2ac47fb7068930c54efd8da9 -803be9c68c91b42b68e1f55e58917a477a9a6265e679ca44ee30d3eb92453f8c89c64eafc04c970d6831edd33d066902 -b14b2b3d0dfe2bb57cee4cd72765b60ac33c1056580950be005790176543826c1d4fbd737f6cfeada6c735543244ab57 -a9e480188bba1b8fb7105ff12215706665fd35bf1117bacfb6ab6985f4dbc181229873b82e5e18323c2b8f5de03258e0 -a66a0f0779436a9a3999996d1e6d3000f22c2cac8e0b29cddef9636393c7f1457fb188a293b6c875b05d68d138a7cc4a -848397366300ab40c52d0dbbdafbafef6cd3dadf1503bb14b430f52bb9724188928ac26f6292a2412bc7d7aa620763c8 -95466cc1a78c9f33a9aaa3829a4c8a690af074916b56f43ae46a67a12bb537a5ac6dbe61590344a25b44e8512355a4a7 -8b5f7a959f818e3baf0887f140f4575cac093d0aece27e23b823cf421f34d6e4ff4bb8384426e33e8ec7b5eed51f6b5c -8d5e1368ec7e3c65640d216bcc5d076f3d9845924c734a34f3558ac0f16e40597c1a775a25bf38b187213fbdba17c93b -b4647c1b823516880f60d20c5cc38c7f80b363c19d191e8992226799718ee26b522a12ecb66556ed3d483aa4824f3326 -ac3abaea9cd283eb347efda4ed9086ea3acf495043e08d0d19945876329e8675224b685612a6badf8fd72fb6274902b1 -8eae1ce292d317aaa71bcf6e77e654914edd5090e2e1ebab78b18bb41b9b1bc2e697439f54a44c0c8aa0d436ebe6e1a9 -94dc7d1aec2c28eb43d93b111fa59aaa0d77d5a09501220bd411768c3e52208806abf973c6a452fd8292ff6490e0c9e2 -8fd8967f8e506fef27d17b435d6b86b232ec71c1036351f12e6fb8a2e12daf01d0ee04451fb944d0f1bf7fd20e714d02 -824e6865be55d43032f0fec65b3480ea89b0a2bf860872237a19a54bc186a85d2f8f9989cc837fbb325b7c72d9babe2c -8bd361f5adb27fd6f4e3f5de866e2befda6a8454efeb704aacc606f528c03f0faae888f60310e49440496abd84083ce2 -b098a3c49f2aaa28b6b3e85bc40ce6a9cdd02134ee522ae73771e667ad7629c8d82c393fba9f27f5416986af4c261438 -b385f5ca285ff2cfe64dcaa32dcde869c28996ed091542600a0b46f65f3f5a38428cca46029ede72b6cf43e12279e3d3 -8196b03d011e5be5288196ef7d47137d6f9237a635ab913acdf9c595fa521d9e2df722090ec7eb0203544ee88178fc5f -8ed1270211ef928db18e502271b7edf24d0bbd11d97f2786aee772d70c2029e28095cf8f650b0328cc8a4c38d045316d -a52ab60e28d69b333d597a445884d44fd2a7e1923dd60f763951e1e45f83e27a4dac745f3b9eff75977b3280e132c15d -91e9fe78cdac578f4a4687f71b800b35da54b824b1886dafec073a3c977ce7a25038a2f3a5b1e35c2c8c9d1a7312417c -a42832173f9d9491c7bd93b21497fbfa4121687cd4d2ab572e80753d7edcbb42cfa49f460026fbde52f420786751a138 -97b947126d84dcc70c97be3c04b3de3f239b1c4914342fa643b1a4bb8c4fe45c0fcb585700d13a7ed50784790c54bef9 -860e407d353eac070e2418ef6cb80b96fc5f6661d6333e634f6f306779651588037be4c2419562c89c61f9aa2c4947f5 -b2c9d93c3ba4e511b0560b55d3501bf28a510745fd666b3cb532db051e6a8617841ea2f071dda6c9f15619c7bfd2737f -8596f4d239aeeac78311207904d1bd863ef68e769629cc379db60e019aaf05a9d5cd31dc8e630b31e106a3a93e47cbc5 -8b26e14e2e136b65c5e9e5c2022cee8c255834ea427552f780a6ca130a6446102f2a6f334c3f9a0308c53df09e3dba7e -b54724354eb515a3c8bed0d0677ff1db94ac0a07043459b4358cb90e3e1aa38ac23f2caa3072cf9647275d7cd61d0e80 -b7ce9fe0e515e7a6b2d7ddcb92bc0196416ff04199326aea57996eef8c5b1548bd8569012210da317f7c0074691d01b7 -a1a13549c82c877253ddefa36a29ea6a23695ee401fdd48e65f6f61e5ebd956d5e0edeff99484e9075cb35071fec41e2 -838ba0c1e5bd1a6da05611ff1822b8622457ebd019cb065ece36a2d176bd2d889511328120b8a357e44569e7f640c1e6 -b916eccff2a95519400bbf76b5f576cbe53cf200410370a19d77734dc04c05b585cfe382e8864e67142d548cd3c4c2f4 -a610447cb7ca6eea53a6ff1f5fe562377dcb7f4aaa7300f755a4f5e8eba61e863c51dc2aa9a29b35525b550fbc32a0fe -9620e8f0f0ee9a4719aa9685eeb1049c5c77659ba6149ec4c158f999cfd09514794b23388879931fe26fea03fa471fd3 -a9dcf8b679e276583cf5b9360702a185470d09aea463dc474ee9c8aee91ef089dacb073e334e47fbc78ec5417c90465c -8c9adee8410bdd99e5b285744cee61e2593b6300ff31a8a83b0ec28da59475a5c6fb9346fe43aadea2e6c3dad2a8e30a -97d5afe9b3897d7b8bb628b7220cf02d8ee4e9d0b78f5000d500aaf4c1df9251aaaabfd1601626519f9d66f00a821d4e -8a382418157b601ce4c3501d3b8409ca98136a4ef6abcbf62885e16e215b76b035c94d149cc41ff92e42ccd7c43b9b3d -b64b8d11fb3b01abb2646ac99fdb9c02b804ce15d98f9fe0fbf1c9df8440c71417487feb6cdf51e3e81d37104b19e012 -849d7d044f9d8f0aab346a9374f0b3a5d14a9d1faa83dbacccbdc629ad1ef903a990940255564770537f8567521d17f0 -829dbb0c76b996c2a91b4cbbe93ba455ca0d5729755e5f0c92aaee37dff7f36fcdc06f33aca41f1b609c784127b67d88 -85a7c0069047b978422d264d831ab816435f63938015d2e977222b6b5746066c0071b7f89267027f8a975206ed25c1b0 -84b9fbc1cfb302df1acdcf3dc5d66fd1edfe7839f7a3b2fb3a0d5548656249dd556104d7c32b73967bccf0f5bdcf9e3b -972220ac5b807f53eac37dccfc2ad355d8b21ea6a9c9b011c09fe440ddcdf7513e0b43d7692c09ded80d7040e26aa28f -855885ed0b21350baeca890811f344c553cf9c21024649c722453138ba29193c6b02c4b4994cd414035486f923472e28 -841874783ae6d9d0e59daea03e96a01cbbe4ecaced91ae4f2c8386e0d87b3128e6d893c98d17c59e4de1098e1ad519dd -827e50fc9ce56f97a4c3f2f4cbaf0b22f1c3ce6f844ff0ef93a9c57a09b8bf91ebfbd2ba9c7f83c442920bffdaf288cc -a441f9136c7aa4c08d5b3534921b730e41ee91ab506313e1ba5f7c6f19fd2d2e1594e88c219834e92e6fb95356385aa7 -97d75b144471bf580099dd6842b823ec0e6c1fb86dd0da0db195e65524129ea8b6fd4a7a9bbf37146269e938a6956596 -a4b6fa87f09d5a29252efb2b3aaab6b3b6ea9fab343132a651630206254a25378e3e9d6c96c3d14c150d01817d375a8e -a31a671876d5d1e95fe2b8858dc69967231190880529d57d3cab7f9f4a2b9b458ac9ee5bdaa3289158141bf18f559efb -90bee6fff4338ba825974021b3b2a84e36d617e53857321f13d2b3d4a28954e6de3b3c0e629d61823d18a9763313b3bf -96b622a63153f393bb419bfcf88272ea8b3560dbd46b0aa07ada3a6223990d0abdd6c2adb356ef4be5641688c8d83941 -84c202adeaff9293698022bc0381adba2cd959f9a35a4e8472288fd68f96f6de8be9da314c526d88e291c96b1f3d6db9 -8ca01a143b8d13809e5a8024d03e6bc9492e22226073ef6e327edf1328ef4aff82d0bcccee92cb8e212831fa35fe1204 -b2f970dbad15bfbefb38903c9bcc043d1367055c55dc1100a850f5eb816a4252c8c194b3132c929105511e14ea10a67d -a5e36556472a95ad57eb90c3b6623671b03eafd842238f01a081997ffc6e2401f76e781d049bb4aa94d899313577a9cf -8d1057071051772f7c8bedce53a862af6fd530dd56ae6321eaf2b9fc6a68beff5ed745e1c429ad09d5a118650bfd420a -8aadc4f70ace4fcb8d93a78610779748dcffc36182d45b932c226dc90e48238ea5daa91f137c65ed532352c4c4d57416 -a2ea05ae37e673b4343232ae685ee14e6b88b867aef6dfac35db3589cbcd76f99540fed5c2641d5bb5a4a9f808e9bf0d -947f1abad982d65648ae4978e094332b4ecb90f482c9be5741d5d1cf5a28acf4680f1977bf6e49dd2174c37f11e01296 -a27b144f1565e4047ba0e3f4840ef19b5095d1e281eaa463c5358f932114cbd018aa6dcf97546465cf2946d014d8e6d6 -8574e1fc3acade47cd4539df578ce9205e745e161b91e59e4d088711a7ab5aa3b410d517d7304b92109924d9e2af8895 -a48ee6b86b88015d6f0d282c1ae01d2a5b9e8c7aa3d0c18b35943dceb1af580d08a65f54dc6903cde82fd0d73ce94722 -8875650cec543a7bf02ea4f2848a61d167a66c91ffaefe31a9e38dc8511c6a25bde431007eefe27a62af3655aca208dc -999b0a6e040372e61937bf0d68374e230346b654b5a0f591a59d33a4f95bdb2f3581db7c7ccb420cd7699ed709c50713 -878c9e56c7100c5e47bbe77dc8da5c5fe706cec94d37fa729633bca63cace7c40102eee780fcdabb655f5fa47a99600e -865006fb5b475ada5e935f27b96f9425fc2d5449a3c106aa366e55ebed3b4ee42adc3c3f0ac19fd129b40bc7d6bc4f63 -b7a7da847f1202e7bc1672553e68904715e84fd897d529243e3ecda59faa4e17ba99c649a802d53f6b8dfdd51f01fb74 -8b2fb4432c05653303d8c8436473682933a5cb604da10c118ecfcd2c8a0e3132e125afef562bdbcc3df936164e5ce4f2 -808d95762d33ddfa5d0ee3d7d9f327de21a994d681a5f372e2e3632963ea974da7f1f9e5bac8ccce24293509d1f54d27 -932946532e3c397990a1df0e94c90e1e45133e347a39b6714c695be21aeb2d309504cb6b1dde7228ff6f6353f73e1ca2 -9705e7c93f0cdfaa3fa96821f830fe53402ad0806036cd1b48adc2f022d8e781c1fbdab60215ce85c653203d98426da3 -aa180819531c3ec1feb829d789cb2092964c069974ae4faad60e04a6afcce5c3a59aec9f11291e6d110a788d22532bc6 -88f755097f7e25cb7dd3c449520c89b83ae9e119778efabb54fbd5c5714b6f37c5f9e0346c58c6ab09c1aef2483f895d -99fc03ab7810e94104c494f7e40b900f475fde65bdec853e60807ffd3f531d74de43335c3b2646b5b8c26804a7448898 -af2dea9683086bed1a179110efb227c9c00e76cd00a2015b089ccbcee46d1134aa18bda5d6cab6f82ae4c5cd2461ac21 -a500f87ba9744787fdbb8e750702a3fd229de6b8817594348dec9a723b3c4240ddfa066262d002844b9e38240ce55658 -924d0e45c780f5bc1c1f35d15dfc3da28036bdb59e4c5440606750ecc991b85be18bc9a240b6c983bc5430baa4c68287 -865b11e0157b8bf4c5f336024b016a0162fc093069d44ac494723f56648bc4ded13dfb3896e924959ea11c96321afefc -93672d8607d4143a8f7894f1dcca83fb84906dc8d6dd7dd063bb0049cfc20c1efd933e06ca7bd03ea4cb5a5037990bfe -826891efbdff0360446825a61cd1fa04326dd90dae8c33dfb1ed97b045e165766dd070bd7105560994d0b2044bdea418 -93c4a4a8bcbc8b190485cc3bc04175b7c0ed002c28c98a540919effd6ed908e540e6594f6db95cd65823017258fb3b1c -aeb2a0af2d2239fda9aa6b8234b019708e8f792834ff0dd9c487fa09d29800ddceddd6d7929faa9a3edcb9e1b3aa0d6b -87f11de7236d387863ec660d2b04db9ac08143a9a2c4dfff87727c95b4b1477e3bc473a91e5797313c58754905079643 -80dc1db20067a844fe8baceca77f80db171a5ca967acb24e2d480eae9ceb91a3343c31ad1c95b721f390829084f0eae6 -9825c31f1c18da0de3fa84399c8b40f8002c3cae211fb6a0623c76b097b4d39f5c50058f57a16362f7a575909d0a44a2 -a99fc8de0c38dbf7b9e946de83943a6b46a762167bafe2a603fb9b86f094da30d6de7ed55d639aafc91936923ee414b3 -ad594678b407db5d6ea2e90528121f84f2b96a4113a252a30d359a721429857c204c1c1c4ff71d8bb5768c833f82e80e -b33d985e847b54510b9b007e31053732c8a495e43be158bd2ffcea25c6765bcbc7ca815f7c60b36ad088b955dd6e9350 -815f8dfc6f90b3342ca3fbd968c67f324dae8f74245cbf8bc3bef10e9440c65d3a2151f951e8d18959ba01c1b50b0ec1 -94c608a362dd732a1abc56e338637c900d59013db8668e49398b3c7a0cae3f7e2f1d1bf94c0299eeafe6af7f76c88618 -8ebd8446b23e5adfcc393adc5c52fe172f030a73e63cd2d515245ca0dd02782ceed5bcdd9ccd9c1b4c5953dfac9c340c -820437f3f6f9ad0f5d7502815b221b83755eb8dc56cd92c29e9535eb0b48fb8d08c9e4fcc26945f9c8cca60d89c44710 -8910e4e8a56bf4be9cc3bbf0bf6b1182a2f48837a2ed3c2aaec7099bfd7f0c83e14e608876b17893a98021ff4ab2f20d -9633918fde348573eec15ce0ad53ac7e1823aac86429710a376ad661002ae6d049ded879383faaa139435122f64047c6 -a1f5e3fa558a9e89318ca87978492f0fb4f6e54a9735c1b8d2ecfb1d1c57194ded6e0dd82d077b2d54251f3bee1279e1 -b208e22d04896abfd515a95c429ff318e87ff81a5d534c8ac2c33c052d6ffb73ef1dccd39c0bbe0734b596c384014766 -986d5d7d2b5bde6d16336f378bd13d0e671ad23a8ec8a10b3fc09036faeeb069f60662138d7a6df3dfb8e0d36180f770 -a2d4e6c5f5569e9cef1cddb569515d4b6ace38c8aed594f06da7434ba6b24477392cc67ba867c2b079545ca0c625c457 -b5ac32b1d231957d91c8b7fc43115ce3c5c0d8c13ca633374402fa8000b6d9fb19499f9181844f0c10b47357f3f757ce -96b8bf2504b4d28fa34a4ec378e0e0b684890c5f44b7a6bb6e19d7b3db2ab27b1e2686389d1de9fbd981962833a313ea -953bfd7f6c3a0469ad432072b9679a25486f5f4828092401eff494cfb46656c958641a4e6d0d97d400bc59d92dba0030 -876ab3cea7484bbfd0db621ec085b9ac885d94ab55c4bb671168d82b92e609754b86aaf472c55df3d81421d768fd108a -885ff4e67d9ece646d02dd425aa5a087e485c3f280c3471b77532b0db6145b69b0fbefb18aa2e3fa5b64928b43a94e57 -b91931d93f806d0b0e6cc62a53c718c099526140f50f45d94b8bbb57d71e78647e06ee7b42aa5714aed9a5c05ac8533f -a0313eeadd39c720c9c27b3d671215331ab8d0a794e71e7e690f06bcd87722b531d6525060c358f35f5705dbb7109ccb -874c0944b7fedc6701e53344100612ddcb495351e29305c00ec40a7276ea5455465ffb7bded898886c1853139dfb1fc7 -8dc31701a01ee8137059ca1874a015130d3024823c0576aa9243e6942ec99d377e7715ed1444cd9b750a64b85dcaa3e5 -836d2a757405e922ec9a2dfdcf489a58bd48b5f9683dd46bf6047688f778c8dee9bc456de806f70464df0b25f3f3d238 -b30b0a1e454a503ea3e2efdec7483eaf20b0a5c3cefc42069e891952b35d4b2c955cf615f3066285ed8fafd9fcfbb8f6 -8e6d4044b55ab747e83ec8762ea86845f1785cc7be0279c075dadf08aca3ccc5a096c015bb3c3f738f647a4eadea3ba5 -ad7735d16ab03cbe09c029610aa625133a6daecfc990b297205b6da98eda8c136a7c50db90f426d35069708510d5ae9c -8d62d858bbb59ec3c8cc9acda002e08addab4d3ad143b3812098f3d9087a1b4a1bb255dcb1635da2402487d8d0249161 -805beec33238b832e8530645a3254aeef957e8f7ea24bcfc1054f8b9c69421145ebb8f9d893237e8a001c857fedfc77e -b1005644be4b085e3f5775aa9bd3e09a283e87ddada3082c04e7a62d303dcef3b8cf8f92944c200c7ae6bb6bdf63f832 -b4ba0e0790dc29063e577474ffe3b61f5ea2508169f5adc1e394934ebb473e356239413a17962bc3e5d3762d72cce8c2 -a157ba9169c9e3e6748d9f1dd67fbe08b9114ade4c5d8fc475f87a764fb7e6f1d21f66d7905cd730f28a1c2d8378682a -913e52b5c93989b5d15e0d91aa0f19f78d592bc28bcfdfddc885a9980c732b1f4debb8166a7c4083c42aeda93a702898 -90fbfc1567e7cd4e096a38433704d3f96a2de2f6ed3371515ccc30bc4dd0721a704487d25a97f3c3d7e4344472702d8d -89646043028ffee4b69d346907586fd12c2c0730f024acb1481abea478e61031966e72072ff1d5e65cb8c64a69ad4eb1 -b125a45e86117ee11d2fb42f680ab4a7894edd67ff927ae2c808920c66c3e55f6a9d4588eee906f33a05d592e5ec3c04 -aad47f5b41eae9be55fb4f67674ff1e4ae2482897676f964a4d2dcb6982252ee4ff56aac49578b23f72d1fced707525e -b9ddff8986145e33851b4de54d3e81faa3352e8385895f357734085a1616ef61c692d925fe62a5ed3be8ca49f5d66306 -b3cb0963387ed28c0c0adf7fe645f02606e6e1780a24d6cecef5b7c642499109974c81a7c2a198b19862eedcea2c2d8c -ac9c53c885457aaf5cb36c717a6f4077af701e0098eebd7aa600f5e4b14e6c1067255b3a0bc40e4a552025231be7de60 -8e1a8d823c4603f6648ec21d064101094f2a762a4ed37dd2f0a2d9aa97b2d850ce1e76f4a4b8cae58819b058180f7031 -b268b73bf7a179b6d22bd37e5e8cb514e9f5f8968c78e14e4f6d5700ca0d0ca5081d0344bb73b028970eebde3cb4124e -a7f57d71940f0edbd29ed8473d0149cae71d921dd15d1ff589774003e816b54b24de2620871108cec1ab9fa956ad6ce6 -8053e6416c8b120e2b999cc2fc420a6a55094c61ac7f2a6c6f0a2c108a320890e389af96cbe378936132363c0d551277 -b3823f4511125e5aa0f4269e991b435a0d6ceb523ebd91c04d7add5534e3df5fc951c504b4fd412a309fd3726b7f940b -ae6eb04674d04e982ca9a6add30370ab90e303c71486f43ed3efbe431af1b0e43e9d06c11c3412651f304c473e7dbf39 -96ab55e641ed2e677591f7379a3cd126449614181fce403e93e89b1645d82c4af524381ff986cae7f9cebe676878646d -b52423b4a8c37d3c3e2eca8f0ddbf7abe0938855f33a0af50f117fab26415fb0a3da5405908ec5fdc22a2c1f2ca64892 -82a69ce1ee92a09cc709d0e3cd22116c9f69d28ea507fe5901f5676000b5179b9abe4c1875d052b0dd42d39925e186bb -a84c8cb84b9d5cfb69a5414f0a5283a5f2e90739e9362a1e8c784b96381b59ac6c18723a4aa45988ee8ef5c1f45cc97d -afd7efce6b36813082eb98257aae22a4c1ae97d51cac7ea9c852d4a66d05ef2732116137d8432e3f117119725a817d24 -a0f5fe25af3ce021b706fcff05f3d825384a272284d04735574ce5fb256bf27100fad0b1f1ba0e54ae9dcbb9570ecad3 -8751786cb80e2e1ff819fc7fa31c2833d25086534eb12b373d31f826382430acfd87023d2a688c65b5e983927e146336 -8cf5c4b17fa4f3d35c78ce41e1dc86988fd1135cd5e6b2bb0c108ee13538d0d09ae7102609c6070f39f937b439b31e33 -a9108967a2fedd7c322711eca8159c533dd561bedcb181b646de98bf5c3079449478eab579731bee8d215ae8852c7e21 -b54c5171704f42a6f0f4e70767cdb3d96ffc4888c842eece343a01557da405961d53ffdc34d2f902ea25d3e1ed867cad -ae8d4b764a7a25330ba205bf77e9f46182cd60f94a336bbd96773cf8064e3d39caf04c310680943dc89ed1fbad2c6e0d -aa5150e911a8e1346868e1b71c5a01e2a4bb8632c195861fb6c3038a0e9b85f0e09b3822e9283654a4d7bb17db2fc5f4 -9685d3756ce9069bf8bb716cf7d5063ebfafe37e15b137fc8c3159633c4e006ff4887ddd0ae90360767a25c3f90cba7f -82155fd70f107ab3c8e414eadf226c797e07b65911508c76c554445422325e71af8c9a8e77fd52d94412a6fc29417cd3 -abfae52f53a4b6e00760468d973a267f29321997c3dbb5aee36dc1f20619551229c0c45b9d9749f410e7f531b73378e8 -81a76d921f8ef88e774fd985e786a4a330d779b93fad7def718c014685ca0247379e2e2a007ad63ee7f729cd9ed6ce1b -81947c84bc5e28e26e2e533af5ae8fe10407a7b77436dbf8f1d5b0bbe86fc659eae10f974659dc7c826c6dabd03e3a4b -92b8c07050d635b8dd4fd09df9054efe4edae6b86a63c292e73cc819a12a21dd7d104ce51fa56af6539dedf6dbe6f7b6 -b44c579e3881f32b32d20c82c207307eca08e44995dd2aac3b2692d2c8eb2a325626c80ac81c26eeb38c4137ff95add5 -97efab8941c90c30860926dea69a841f2dcd02980bf5413b9fd78d85904588bf0c1021798dbc16c8bbb32cce66c82621 -913363012528b50698e904de0588bf55c8ec5cf6f0367cfd42095c4468fcc64954fbf784508073e542fee242d0743867 -8ed203cf215148296454012bd10fddaf119203db1919a7b3d2cdc9f80e66729464fdfae42f1f2fc5af1ed53a42b40024 -ab84312db7b87d711e9a60824f4fe50e7a6190bf92e1628688dfcb38930fe87b2d53f9e14dd4de509b2216856d8d9188 -880726def069c160278b12d2258eac8fa63f729cd351a710d28b7e601c6712903c3ac1e7bbd0d21e4a15f13ca49db5aa -980699cd51bac6283959765f5174e543ed1e5f5584b5127980cbc2ef18d984ecabba45042c6773b447b8e694db066028 -aeb019cb80dc4cb4207430d0f2cd24c9888998b6f21d9bf286cc638449668d2eec0018a4cf3fe6448673cd6729335e2b -b29852f6aa6c60effdffe96ae88590c88abae732561d35cc19e82d3a51e26cb35ea00986193e07f90060756240f5346e -a0fa855adc5ba469f35800c48414b8921455950a5c0a49945d1ef6e8f2a1881f2e2dfae47de6417270a6bf49deeb091d -b6c7332e3b14813641e7272d4f69ecc7e09081df0037d6dab97ce13a9e58510f5c930d300633f208181d9205c5534001 -85a6c050f42fce560b5a8d54a11c3bbb8407abbadd859647a7b0c21c4b579ec65671098b74f10a16245dc779dff7838e -8f3eb34bb68759d53c6677de4de78a6c24dd32c8962a7fb355ed362572ef8253733e6b52bc21c9f92ecd875020a9b8de -a17dd44181e5dab4dbc128e1af93ec22624b57a448ca65d2d9e246797e4af7d079e09c6e0dfb62db3a9957ce92f098d5 -a56a1b854c3183082543a8685bb34cae1289f86cfa8123a579049dbd059e77982886bfeb61bf6e05b4b1fe4e620932e7 -aedae3033cb2fb7628cb4803435bdd7757370a86f808ae4cecb9a268ad0e875f308c048c80cbcac523de16b609683887 -9344905376aa3982b1179497fac5a1d74b14b7038fd15e3b002db4c11c8bfc7c39430db492cdaf58b9c47996c9901f28 -a3bfafdae011a19f030c749c3b071f83580dee97dd6f949e790366f95618ca9f828f1daaeabad6dcd664fcef81b6556d -81c03d8429129e7e04434dee2c529194ddb01b414feda3adee2271eb680f6c85ec872a55c9fa9d2096f517e13ed5abcc -98205ef3a72dff54c5a9c82d293c3e45d908946fa74bb749c3aabe1ab994ea93c269bcce1a266d2fe67a8f02133c5985 -85a70aeed09fda24412fadbafbbbf5ba1e00ac92885df329e147bfafa97b57629a3582115b780d8549d07d19b7867715 -b0fbe81c719f89a57d9ea3397705f898175808c5f75f8eb81c2193a0b555869ba7bd2e6bc54ee8a60cea11735e21c68c -b03a0bd160495ee626ff3a5c7d95bc79d7da7e5a96f6d10116600c8fa20bedd1132f5170f25a22371a34a2d763f2d6d0 -a90ab04091fbca9f433b885e6c1d60ab45f6f1daf4b35ec22b09909d493a6aab65ce41a6f30c98239cbca27022f61a8b -b66f92aa3bf2549f9b60b86f99a0bd19cbdd97036d4ae71ca4b83d669607f275260a497208f6476cde1931d9712c2402 -b08e1fdf20e6a9b0b4942f14fa339551c3175c1ffc5d0ab5b226b6e6a322e9eb0ba96adc5c8d59ca4259e2bdd04a7eb0 -a2812231e92c1ce74d4f5ac3ab6698520288db6a38398bb38a914ac9326519580af17ae3e27cde26607e698294022c81 -abfcbbcf1d3b9e84c02499003e490a1d5d9a2841a9e50c7babbef0b2dd20d7483371d4dc629ba07faf46db659459d296 -b0fe9f98c3da70927c23f2975a9dc4789194d81932d2ad0f3b00843dd9cbd7fb60747a1da8fe5a79f136a601becf279d -b130a6dba7645165348cb90f023713bed0eefbd90a976b313521c60a36d34f02032e69a2bdcf5361e343ed46911297ec -862f0cffe3020cea7a5fd4703353aa1eb1be335e3b712b29d079ff9f7090d1d8b12013011e1bdcbaa80c44641fd37c9f -8c6f11123b26633e1abb9ed857e0bce845b2b3df91cc7b013b2fc77b477eee445da0285fc6fc793e29d5912977f40916 -91381846126ea819d40f84d3005e9fb233dc80071d1f9bb07f102bf015f813f61e5884ffffb4f5cd333c1b1e38a05a58 -8add7d908de6e1775adbd39c29a391f06692b936518db1f8fde74eb4f533fc510673a59afb86e3a9b52ade96e3004c57 -8780e086a244a092206edcde625cafb87c9ab1f89cc3e0d378bc9ee776313836160960a82ec397bc3800c0a0ec3da283 -a6cb4cd9481e22870fdd757fae0785edf4635e7aacb18072fe8dc5876d0bab53fb99ce40964a7d3e8bcfff6f0ab1332f -af30ff47ecc5b543efba1ba4706921066ca8bb625f40e530fb668aea0551c7647a9d126e8aba282fbcce168c3e7e0130 -91b0bcf408ce3c11555dcb80c4410b5bc2386d3c05caec0b653352377efdcb6bab4827f2018671fc8e4a0e90d772acc1 -a9430b975ef138b6b2944c7baded8fe102d31da4cfe3bd3d8778bda79189c99d38176a19c848a19e2d1ee0bddd9a13c1 -aa5a4eef849d7c9d2f4b018bd01271c1dd83f771de860c4261f385d3bdcc130218495860a1de298f14b703ec32fa235f -b0ce79e7f9ae57abe4ff366146c3b9bfb38b0dee09c28c28f5981a5d234c6810ad4d582751948affb480d6ae1c8c31c4 -b75122748560f73d15c01a8907d36d06dc068e82ce22b84b322ac1f727034493572f7907dec34ebc3ddcc976f2f89ed7 -b0fc7836369a3e4411d34792d6bd5617c14f61d9bba023dda64e89dc5fb0f423244e9b48ee64869258931daa9753a56f -8956d7455ae9009d70c6e4a0bcd7610e55f37494cf9897a8f9e1b904cc8febc3fd2d642ebd09025cfff4609ad7e3bc52 -ad741efe9e472026aa49ae3d9914cb9c1a6f37a54f1a6fe6419bebd8c7d68dca105a751c7859f4389505ede40a0de786 -b52f418797d719f0d0d0ffb0846788b5cba5d0454a69a2925de4b0b80fa4dd7e8c445e5eac40afd92897ed28ca650566 -a0ab65fb9d42dd966cd93b1de01d7c822694669dd2b7a0c04d99cd0f3c3de795f387b9c92da11353412f33af5c950e9a -a0052f44a31e5741a331f7cac515a08b3325666d388880162d9a7b97598fde8b61f9ff35ff220df224eb5c4e40ef0567 -a0101cfdc94e42b2b976c0d89612a720e55d145a5ef6ef6f1f78cf6de084a49973d9b5d45915349c34ce712512191e3c -a0dd99fcf3f5cead5aaf08e82212df3a8bb543c407a4d6fab88dc5130c1769df3f147e934a46f291d6c1a55d92b86917 -a5939153f0d1931bbda5cf6bdf20562519ea55fbfa978d6dbc6828d298260c0da7a50c37c34f386e59431301a96c2232 -9568269f3f5257200f9ca44afe1174a5d3cf92950a7f553e50e279c239e156a9faaa2a67f288e3d5100b4142efe64856 -b746b0832866c23288e07f24991bbf687cad794e7b794d3d3b79367566ca617d38af586cdc8d6f4a85a34835be41d54f -a871ce28e39ab467706e32fec1669fda5a4abba2f8c209c6745df9f7a0fa36bbf1919cf14cb89ea26fa214c4c907ae03 -a08dacdd758e523cb8484f6bd070642c0c20e184abdf8e2a601f61507e93952d5b8b0c723c34fcbdd70a8485eec29db2 -85bdb78d501382bb95f1166b8d032941005661aefd17a5ac32df9a3a18e9df2fc5dc2c1f07075f9641af10353cecc0c9 -98d730c28f6fa692a389e97e368b58f4d95382fad8f0baa58e71a3d7baaea1988ead47b13742ce587456f083636fa98e -a557198c6f3d5382be9fb363feb02e2e243b0c3c61337b3f1801c4a0943f18e38ce1a1c36b5c289c8fa2aa9d58742bab -89174f79201742220ac689c403fc7b243eed4f8e3f2f8aba0bf183e6f5d4907cb55ade3e238e3623d9885f03155c4d2b -b891d600132a86709e06f3381158db300975f73ea4c1f7c100358e14e98c5fbe792a9af666b85c4e402707c3f2db321e -b9e5b2529ef1043278c939373fc0dbafe446def52ddd0a8edecd3e4b736de87e63e187df853c54c28d865de18a358bb6 -8589b2e9770340c64679062c5badb7bbef68f55476289b19511a158a9a721f197da03ece3309e059fc4468b15ac33aa3 -aad8c6cd01d785a881b446f06f1e9cd71bca74ba98674c2dcddc8af01c40aa7a6d469037498b5602e76e9c91a58d3dbd -abaccb1bd918a8465f1bf8dbe2c9ad4775c620b055550b949a399f30cf0d9eb909f3851f5b55e38f9e461e762f88f499 -ae62339d26db46e85f157c0151bd29916d5cc619bd4b832814b3fd2f00af8f38e7f0f09932ffe5bba692005dab2d9a74 -93a6ff30a5c0edf8058c89aba8c3259e0f1b1be1b80e67682de651e5346f7e1b4b4ac3d87cbaebf198cf779524aff6bf -8980a2b1d8f574af45b459193c952400b10a86122b71fca2acb75ee0dbd492e7e1ef5b959baf609a5172115e371f3177 -8c2f49f3666faee6940c75e8c7f6f8edc3f704cca7a858bbb7ee5e96bba3b0cf0993996f781ba6be3b0821ef4cb75039 -b14b9e348215b278696018330f63c38db100b0542cfc5be11dc33046e3bca6a13034c4ae40d9cef9ea8b34fef0910c4e -b59bc3d0a30d66c16e6a411cb641f348cb1135186d5f69fda8b0a0934a5a2e7f6199095ba319ec87d3fe8f1ec4a06368 -8874aca2a3767aa198e4c3fec2d9c62d496bc41ff71ce242e9e082b7f38cdf356089295f80a301a3cf1182bde5308c97 -b1820ebd61376d91232423fc20bf008b2ba37e761199f4ef0648ea2bd70282766799b4de814846d2f4d516d525c8daa7 -a6b202e5dedc16a4073e04a11af3a8509b23dfe5a1952f899adeb240e75c3f5bde0c424f811a81ea48d343591faffe46 -a69becee9c93734805523b92150a59a62eed4934f66056b645728740d42223f2925a1ad38359ba644da24d9414f4cdda -ad72f0f1305e37c7e6b48c272323ee883320994cb2e0d850905d6655fafc9f361389bcb9c66b3ff8d2051dbb58c8aa96 -b563600bd56fad7c8853af21c6a02a16ed9d8a8bbeea2c31731d63b976d83cb05b9779372d898233e8fd597a75424797 -b0abb78ce465bf7051f563c62e8be9c57a2cc997f47c82819300f36e301fefd908894bb2053a9d27ce2d0f8c46d88b5b -a071a85fb8274bac2202e0cb8e0e2028a5e138a82d6e0374d39ca1884a549c7c401312f00071b91f455c3a2afcfe0cda -b931c271513a0f267b9f41444a5650b1918100b8f1a64959c552aff4e2193cc1b9927906c6fa7b8a8c68ef13d79aaa52 -a6a1bb9c7d32cb0ca44d8b75af7e40479fbce67d216b48a2bb680d3f3a772003a49d3cd675fc64e9e0f8fabeb86d6d61 -b98d609858671543e1c3b8564162ad828808bb50ded261a9f8690ded5b665ed8368c58f947365ed6e84e5a12e27b423d -b3dca58cd69ec855e2701a1d66cad86717ff103ef862c490399c771ad28f675680f9500cb97be48de34bcdc1e4503ffd -b34867c6735d3c49865e246ddf6c3b33baf8e6f164db3406a64ebce4768cb46b0309635e11be985fee09ab7a31d81402 -acb966c554188c5b266624208f31fab250b3aa197adbdd14aee5ab27d7fb886eb4350985c553b20fdf66d5d332bfd3fe -943c36a18223d6c870d54c3b051ef08d802b85e9dd6de37a51c932f90191890656c06adfa883c87b906557ae32d09da0 -81bca7954d0b9b6c3d4528aadf83e4bc2ef9ea143d6209bc45ae9e7ae9787dbcd8333c41f12c0b6deee8dcb6805e826a -aba176b92256efb68f574e543479e5cf0376889fb48e3db4ebfb7cba91e4d9bcf19dcfec444c6622d9398f06de29e2b9 -b9f743691448053216f6ece7cd699871fff4217a1409ceb8ab7bdf3312d11696d62c74b0664ba0a631b1e0237a8a0361 -a383c2b6276fa9af346b21609326b53fb14fdf6f61676683076e80f375b603645f2051985706d0401e6fbed7eb0666b6 -a9ef2f63ec6d9beb8f3d04e36807d84bda87bdd6b351a3e4a9bf7edcb5618c46c1f58cfbf89e64b40f550915c6988447 -a141b2d7a82f5005eaea7ae7d112c6788b9b95121e5b70b7168d971812f3381de8b0082ac1f0a82c7d365922ebd2d26a -b1b76ef8120e66e1535c17038b75255a07849935d3128e3e99e56567b842fb1e8d56ef932d508d2fb18b82f7868fe1a9 -8e2e234684c81f21099f5c54f6bbe2dd01e3b172623836c77668a0c49ce1fe218786c3827e4d9ae2ea25c50a8924fb3c -a5caf5ff948bfd3c4ca3ffbdfcd91eec83214a6c6017235f309a0bbf7061d3b0b466307c00b44a1009cf575163898b43 -986415a82ca16ebb107b4c50b0c023c28714281db0bcdab589f6cb13d80e473a3034b7081b3c358e725833f6d845cb14 -b94836bf406ac2cbacb10e6df5bcdfcc9d9124ae1062767ca4e322d287fd5e353fdcebd0e52407cb3cd68571258a8900 -83c6d70a640b33087454a4788dfd9ef3ed00272da084a8d36be817296f71c086b23b576f98178ab8ca6a74f04524b46b -ad4115182ad784cfe11bcfc5ce21fd56229cc2ce77ac82746e91a2f0aa53ca6593a22efd2dc4ed8d00f84542643d9c58 -ab1434c5e5065da826d10c2a2dba0facccab0e52b506ce0ce42fbe47ced5a741797151d9ecc99dc7d6373cfa1779bbf6 -8a8b591d82358d55e6938f67ea87a89097ab5f5496f7260adb9f649abb289da12b498c5b2539c2f9614fb4e21b1f66b0 -964f355d603264bc1f44c64d6d64debca66f37dff39c971d9fc924f2bc68e6c187b48564a6dc82660a98b035f8addb5d -b66235eaaf47456bc1dc4bde454a028e2ce494ece6b713a94cd6bf27cf18c717fd0c57a5681caaa2ad73a473593cdd7a -9103e3bb74304186fa4e3e355a02da77da4aca9b7e702982fc2082af67127ebb23a455098313c88465bc9b7d26820dd5 -b6a42ff407c9dd132670cdb83cbad4b20871716e44133b59a932cd1c3f97c7ac8ff7f61acfaf8628372508d8dc8cad7c -883a9c21c16a167a4171b0f084565c13b6f28ba7c4977a0de69f0a25911f64099e7bbb4da8858f2e93068f4155d04e18 -8dbb3220abc6a43220adf0331e3903d3bfd1d5213aadfbd8dfcdf4b2864ce2e96a71f35ecfb7a07c3bbabf0372b50271 -b4ad08aee48e176bda390b7d9acf2f8d5eb008f30d20994707b757dc6a3974b2902d29cd9b4d85e032810ad25ac49e97 -865bb0f33f7636ec501bb634e5b65751c8a230ae1fa807a961a8289bbf9c7fe8c59e01fbc4c04f8d59b7f539cf79ddd5 -86a54d4c12ad1e3605b9f93d4a37082fd26e888d2329847d89afa7802e815f33f38185c5b7292293d788ad7d7da1df97 -b26c8615c5e47691c9ff3deca3021714662d236c4d8401c5d27b50152ce7e566266b9d512d14eb63e65bc1d38a16f914 -827639d5ce7db43ba40152c8a0eaad443af21dc92636cc8cc2b35f10647da7d475a1e408901cd220552fddad79db74df -a2b79a582191a85dbe22dc384c9ca3de345e69f6aa370aa6d3ff1e1c3de513e30b72df9555b15a46586bd27ea2854d9d -ae0d74644aba9a49521d3e9553813bcb9e18f0b43515e4c74366e503c52f47236be92dfbd99c7285b3248c267b1de5a0 -80fb0c116e0fd6822a04b9c25f456bdca704e2be7bdc5d141dbf5d1c5eeb0a2c4f5d80db583b03ef3e47517e4f9a1b10 -ac3a1fa3b4a2f30ea7e0a114cdc479eb51773573804c2a158d603ad9902ae8e39ffe95df09c0d871725a5d7f9ba71a57 -b56b2b0d601cba7f817fa76102c68c2e518c6f20ff693aad3ff2e07d6c4c76203753f7f91686b1801e8c4659e4d45c48 -89d50c1fc56e656fb9d3915964ebce703cb723fe411ab3c9eaa88ccc5d2b155a9b2e515363d9c600d3c0cee782c43f41 -b24207e61462f6230f3cd8ccf6828357d03e725769f7d1de35099ef9ee4dca57dbce699bb49ed994462bee17059d25ce -b886f17fcbcbfcd08ac07f04bb9543ef58510189decaccea4b4158c9174a067cb67d14b6be3c934e6e2a18c77efa9c9c -b9c050ad9cafd41c6e2e192b70d080076eed59ed38ea19a12bd92fa17b5d8947d58d5546aaf5e8e27e1d3b5481a6ce51 -aaf7a34d3267e3b1ddbc54c641e3922e89303f7c86ebebc7347ebca4cffad5b76117dac0cbae1a133053492799cd936f -a9ee604ada50adef82e29e893070649d2d4b7136cc24fa20e281ce1a07bd736bf0de7c420369676bcbcecff26fb6e900 -9855315a12a4b4cf80ab90b8bd13003223ba25206e52fd4fe6a409232fbed938f30120a3db23eab9c53f308bd8b9db81 -8cd488dd7a24f548a3cf03c54dec7ff61d0685cb0f6e5c46c2d728e3500d8c7bd6bba0156f4bf600466fda53e5b20444 -890ad4942ebac8f5b16c777701ab80c68f56fa542002b0786f8fea0fb073154369920ac3dbfc07ea598b82f4985b8ced -8de0cf9ddc84c9b92c59b9b044387597799246b30b9f4d7626fc12c51f6e423e08ee4cbfe9289984983c1f9521c3e19d -b474dfb5b5f4231d7775b3c3a8744956b3f0c7a871d835d7e4fd9cc895222c7b868d6c6ce250de568a65851151fac860 -86433b6135d9ed9b5ee8cb7a6c40e5c9d30a68774cec04988117302b8a02a11a71a1e03fd8e0264ef6611d219f103007 -80b9ed4adbe9538fb1ef69dd44ec0ec5b57cbfea820054d8d445b4261962624b4c70ac330480594bc5168184378379c3 -8b2e83562ccd23b7ad2d17f55b1ab7ef5fbef64b3a284e6725b800f3222b8bdf49937f4a873917ada9c4ddfb090938c2 -abe78cebc0f5a45d754140d1f685e387489acbfa46d297a8592aaa0d676a470654f417a4f7d666fc0b2508fab37d908e -a9c5f8ff1f8568e252b06d10e1558326db9901840e6b3c26bbd0cd5e850cb5fb3af3f117dbb0f282740276f6fd84126f -975f8dc4fb55032a5df3b42b96c8c0ffecb75456f01d4aef66f973cb7270d4eff32c71520ceefc1adcf38d77b6b80c67 -b043306ed2c3d8a5b9a056565afd8b5e354c8c4569fda66b0d797a50a3ce2c08cffbae9bbe292da69f39e89d5dc7911e -8d2afc36b1e44386ba350c14a6c1bb31ff6ea77128a0c5287584ac3584282d18516901ce402b4644a53db1ed8e7fa581 -8c294058bed53d7290325c363fe243f6ec4f4ea2343692f4bac8f0cb86f115c069ccb8334b53d2e42c067691ad110dba -b92157b926751aaf7ef82c1aa8c654907dccab6376187ee8b3e8c0c82811eae01242832de953faa13ebaff7da8698b3e -a780c4bdd9e4ba57254b09d745075cecab87feda78c88ffee489625c5a3cf96aa6b3c9503a374a37927d9b78de9bd22b -811f548ef3a2e6a654f7dcb28ac9378de9515ed61e5a428515d9594a83e80b35c60f96a5cf743e6fab0d3cb526149f49 -85a4dccf6d90ee8e094731eec53bd00b3887aec6bd81a0740efddf812fd35e3e4fe4f983afb49a8588691c202dabf942 -b152c2da6f2e01c8913079ae2b40a09b1f361a80f5408a0237a8131b429677c3157295e11b365b1b1841924b9efb922e -849b9efee8742502ffd981c4517c88ed33e4dd518a330802caff168abae3cd09956a5ee5eda15900243bc2e829016b74 -955a933f3c18ec0f1c0e38fa931e4427a5372c46a3906ebe95082bcf878c35246523c23f0266644ace1fa590ffa6d119 -911989e9f43e580c886656377c6f856cdd4ff1bd001b6db3bbd86e590a821d34a5c6688a29b8d90f28680e9fdf03ba69 -b73b8b4f1fd6049fb68d47cd96a18fcba3f716e0a1061aa5a2596302795354e0c39dea04d91d232aec86b0bf2ba10522 -90f87456d9156e6a1f029a833bf3c7dbed98ca2f2f147a8564922c25ae197a55f7ea9b2ee1f81bf7383197c4bad2e20c -903cba8b1e088574cb04a05ca1899ab00d8960580c884bd3c8a4c98d680c2ad11410f2b75739d6050f91d7208cac33a5 -9329987d42529c261bd15ecedd360be0ea8966e7838f32896522c965adfc4febf187db392bd441fb43bbd10c38fdf68b -8178ee93acf5353baa349285067b20e9bb41aa32d77b5aeb7384fe5220c1fe64a2461bd7a83142694fe673e8bbf61b7c -a06a8e53abcff271b1394bcc647440f81fb1c1a5f29c27a226e08f961c3353f4891620f2d59b9d1902bf2f5cc07a4553 -aaf5fe493b337810889e777980e6bbea6cac39ac66bc0875c680c4208807ac866e9fda9b5952aa1d04539b9f4a4bec57 -aa058abb1953eceac14ccfa7c0cc482a146e1232905dcecc86dd27f75575285f06bbae16a8c9fe8e35d8713717f5f19f -8f15dd732799c879ca46d2763453b359ff483ca33adb1d0e0a57262352e0476c235987dc3a8a243c74bc768f93d3014c -a61cc8263e9bc03cce985f1663b8a72928a607121005a301b28a278e9654727fd1b22bc8a949af73929c56d9d3d4a273 -98d6dc78502d19eb9f921225475a6ebcc7b44f01a2df6f55ccf6908d65b27af1891be2a37735f0315b6e0f1576c1f8d8 -8bd258b883f3b3793ec5be9472ad1ff3dc4b51bc5a58e9f944acfb927349ead8231a523cc2175c1f98e7e1e2b9f363b8 -aeacc2ecb6e807ad09bedd99654b097a6f39840e932873ace02eabd64ccfbb475abdcb62939a698abf17572d2034c51e -b8ccf78c08ccd8df59fd6eda2e01de328bc6d8a65824d6f1fc0537654e9bc6bf6f89c422dd3a295cce628749da85c864 -8f91fd8cb253ba2e71cc6f13da5e05f62c2c3b485c24f5d68397d04665673167fce1fc1aec6085c69e87e66ec555d3fd -a254baa10cb26d04136886073bb4c159af8a8532e3fd36b1e9c3a2e41b5b2b6a86c4ebc14dbe624ee07b7ccdaf59f9ab -94e3286fe5cd68c4c7b9a7d33ae3d714a7f265cf77cd0e9bc19fc51015b1d1c34ad7e3a5221c459e89f5a043ee84e3a9 -a279da8878af8d449a9539bec4b17cea94f0242911f66fab275b5143ab040825f78c89cb32a793930609415cfa3a1078 -ac846ceb89c9e5d43a2991c8443079dc32298cd63e370e64149cec98cf48a6351c09c856f2632fd2f2b3d685a18bbf8b -a847b27995c8a2e2454aaeb983879fb5d3a23105c33175839f7300b7e1e8ec3efd6450e9fa3f10323609dee7b98c6fd5 -a2f432d147d904d185ff4b2de8c6b82fbea278a2956bc406855b44c18041854c4f0ecccd472d1d0dff1d8aa8e281cb1d -94a48ad40326f95bd63dff4755f863a1b79e1df771a1173b17937f9baba57b39e651e7695be9f66a472f098b339364fc -a12a0ccd8f96e96e1bc6494341f7ebce959899341b3a084aa1aa87d1c0d489ac908552b7770b887bb47e7b8cbc3d8e66 -81a1f1681bda923bd274bfe0fbb9181d6d164fe738e54e25e8d4849193d311e2c4253614ed673c98af2c798f19a93468 -abf71106a05d501e84cc54610d349d7d5eae21a70bd0250f1bebbf412a130414d1c8dbe673ffdb80208fd72f1defa4d4 -96266dc2e0df18d8136d79f5b59e489978eee0e6b04926687fe389d4293c14f36f055c550657a8e27be4118b64254901 -8df5dcbefbfb4810ae3a413ca6b4bf08619ca53cd50eb1dde2a1c035efffc7b7ac7dff18d403253fd80104bd83dc029e -9610b87ff02e391a43324a7122736876d5b3af2a137d749c52f75d07b17f19900b151b7f439d564f4529e77aa057ad12 -a90a5572198b40fe2fcf47c422274ff36c9624df7db7a89c0eb47eb48a73a03c985f4ac5016161c76ca317f64339bce1 -98e5e61a6ab6462ba692124dba7794b6c6bde4249ab4fcc98c9edd631592d5bc2fb5e38466691a0970a38e48d87c2e43 -918cefb8f292f78d4db81462c633daf73b395e772f47b3a7d2cea598025b1d8c3ec0cbff46cdb23597e74929981cde40 -a98918a5dc7cf610fe55f725e4fd24ce581d594cb957bb9b4e888672e9c0137003e1041f83e3f1d7b9caab06462c87d4 -b92b74ac015262ca66c33f2d950221e19d940ba3bf4cf17845f961dc1729ae227aa9e1f2017829f2135b489064565c29 -a053ee339f359665feb178b4e7ee30a85df37debd17cacc5a27d6b3369d170b0114e67ad1712ed26d828f1df641bcd99 -8c3c8bad510b35da5ce5bd84b35c958797fbea024ad1c97091d2ff71d9b962e9222f65a9b776e5b3cc29c36e1063d2ee -af99dc7330fe7c37e850283eb47cc3257888e7c197cb0d102edf94439e1e02267b6a56306d246c326c4c79f9dc8c6986 -afecb2dc34d57a725efbd7eb93d61eb29dbe8409b668ab9ea040791f5b796d9be6d4fc10d7f627bf693452f330cf0435 -93334fedf19a3727a81a6b6f2459db859186227b96fe7a391263f69f1a0884e4235de64d29edebc7b99c44d19e7c7d7a -89579c51ac405ad7e9df13c904061670ce4b38372492764170e4d3d667ed52e5d15c7cd5c5991bbfa3a5e4e3fa16363e -9778f3e8639030f7ef1c344014f124e375acb8045bd13d8e97a92c5265c52de9d1ffebaa5bc3e1ad2719da0083222991 -88f77f34ee92b3d36791bdf3326532524a67d544297dcf1a47ff00b47c1b8219ff11e34034eab7d23b507caa2fd3c6b9 -a699c1e654e7c484431d81d90657892efeb4adcf72c43618e71ca7bd7c7a7ebbb1db7e06e75b75dc4c74efd306b5df3f -81d13153baebb2ef672b5bdb069d3cd669ce0be96b742c94e04038f689ff92a61376341366b286eee6bf3ae85156f694 -81efb17de94400fdacc1deec2550cbe3eecb27c7af99d8207e2f9be397e26be24a40446d2a09536bb5172c28959318d9 -989b21ebe9ceab02488992673dc071d4d5edec24bff0e17a4306c8cb4b3c83df53a2063d1827edd8ed16d6e837f0d222 -8d6005d6536825661b13c5fdce177cb37c04e8b109b7eb2b6d82ea1cb70efecf6a0022b64f84d753d165edc2bba784a3 -a32607360a71d5e34af2271211652d73d7756d393161f4cf0da000c2d66a84c6826e09e759bd787d4fd0305e2439d342 -aaad8d6f6e260db45d51b2da723be6fa832e76f5fbcb77a9a31e7f090dd38446d3b631b96230d78208cae408c288ac4e -abcfe425255fd3c5cffd3a818af7650190c957b6b07b632443f9e33e970a8a4c3bf79ac9b71f4d45f238a04d1c049857 -aeabf026d4c783adc4414b5923dbd0be4b039cc7201219f7260d321f55e9a5b166d7b5875af6129c034d0108fdc5d666 -af49e740c752d7b6f17048014851f437ffd17413c59797e5078eaaa36f73f0017c3e7da020310cfe7d3c85f94a99f203 -8854ca600d842566e3090040cd66bb0b3c46dae6962a13946f0024c4a8aca447e2ccf6f240045f1ceee799a88cb9210c -b6c03b93b1ab1b88ded8edfa1b487a1ed8bdce8535244dddb558ffb78f89b1c74058f80f4db2320ad060d0c2a9c351cc -b5bd7d17372faff4898a7517009b61a7c8f6f0e7ed4192c555db264618e3f6e57fb30a472d169fea01bf2bf0362a19a8 -96eb1d38319dc74afe7e7eb076fcd230d19983f645abd14a71e6103545c01301b31c47ae931e025f3ecc01fb3d2f31fa -b55a8d30d4403067def9b65e16f867299f8f64c9b391d0846d4780bc196569622e7e5b64ce799b5aefac8f965b2a7a7b -8356d199a991e5cbbff608752b6291731b6b6771aed292f8948b1f41c6543e4ab1bedc82dd26d10206c907c03508df06 -97f4137445c2d98b0d1d478049de952610ad698c91c9d0f0e7227d2aae690e9935e914ec4a2ea1fbf3fc1dddfeeacebb -af5621707e0938320b15ddfc87584ab325fbdfd85c30efea36f8f9bd0707d7ec12c344eff3ec21761189518d192df035 -8ac7817e71ea0825b292687928e349da7140285d035e1e1abff0c3704fa8453faaae343a441b7143a74ec56539687cc4 -8a5e0a9e4758449489df10f3386029ada828d1762e4fb0a8ffe6b79e5b6d5d713cb64ed95960e126398b0cdb89002bc9 -81324be4a71208bbb9bca74b77177f8f1abb9d3d5d9db195d1854651f2cf333cd618d35400da0f060f3e1b025124e4b2 -849971d9d095ae067525b3cbc4a7dfae81f739537ade6d6cec1b42fb692d923176197a8770907c58069754b8882822d6 -89f830825416802477cc81fdf11084885865ee6607aa15aa4eb28e351c569c49b8a1b9b5e95ddc04fa0ebafe20071313 -9240aeeaff37a91af55f860b9badd466e8243af9e8c96a7aa8cf348cd270685ab6301bc135b246dca9eda696f8b0e350 -acf74db78cc33138273127599eba35b0fb4e7b9a69fe02dae18fc6692d748ca332bd00b22afa8e654ed587aab11833f3 -b091e6d37b157b50d76bd297ad752220cd5c9390fac16dc838f8557aed6d9833fc920b61519df21265406216315e883f -a6446c429ebf1c7793c622250e23594c836b2fbcaf6c5b3d0995e1595a37f50ea643f3e549b0be8bbdadd69044d72ab9 -93e675353bd60e996bf1c914d5267eeaa8a52fc3077987ccc796710ef9becc6b7a00e3d82671a6bdfb8145ee3c80245a -a2f731e43251d04ed3364aa2f072d05355f299626f2d71a8a38b6f76cf08c544133f7d72dd0ab4162814b674b9fc7fa6 -97a8b791a5a8f6e1d0de192d78615d73d0c38f1e557e4e15d15adc663d649e655bc8da3bcc499ef70112eafe7fb45c7a -98cd624cbbd6c53a94469be4643c13130916b91143425bcb7d7028adbbfede38eff7a21092af43b12d4fab703c116359 -995783ce38fd5f6f9433027f122d4cf1e1ff3caf2d196ce591877f4a544ce9113ead60de2de1827eaff4dd31a20d79a8 -8cf251d6f5229183b7f3fe2f607a90b4e4b6f020fb4ba2459d28eb8872426e7be8761a93d5413640a661d73e34a5b81f -b9232d99620652a3aa7880cad0876f153ff881c4ed4c0c2e7b4ea81d5d42b70daf1a56b869d752c3743c6d4c947e6641 -849716f938f9d37250cccb1bf77f5f9fde53096cdfc6f2a25536a6187029a8f1331cdbed08909184b201f8d9f04b792f -80c7c4de098cbf9c6d17b14eba1805e433b5bc905f6096f8f63d34b94734f2e4ebf4bce8a177efd1186842a61204a062 -b790f410cf06b9b8daadceeb4fd5ff40a2deda820c8df2537e0a7554613ae3948e149504e3e79aa84889df50c8678eeb -813aab8bd000299cd37485b73cd7cba06e205f8efb87f1efc0bae8b70f6db2bc7702eb39510ad734854fb65515fe9d0f -94f0ab7388ac71cdb67f6b85dfd5945748afb2e5abb622f0b5ad104be1d4d0062b651f134ba22385c9e32c2dfdcccce1 -ab6223dca8bd6a4f969e21ccd9f8106fc5251d321f9e90cc42cea2424b3a9c4e5060a47eeef6b23c7976109b548498e8 -859c56b71343fce4d5c5b87814c47bf55d581c50fd1871a17e77b5e1742f5af639d0e94d19d909ec7dfe27919e954e0c -aae0d632b6191b8ad71b027791735f1578e1b89890b6c22e37de0e4a6074886126988fe8319ae228ac9ef3b3bcccb730 -8ca9f32a27a024c3d595ecfaf96b0461de57befa3b331ab71dc110ec3be5824fed783d9516597537683e77a11d334338 -a061df379fb3f4b24816c9f6cd8a94ecb89b4c6dc6cd81e4b8096fa9784b7f97ab3540259d1de9c02eb91d9945af4823 -998603102ac63001d63eb7347a4bb2bf4cf33b28079bb48a169076a65c20d511ccd3ef696d159e54cc8e772fb5d65d50 -94444d96d39450872ac69e44088c252c71f46be8333a608a475147752dbb99db0e36acfc5198f158509401959c12b709 -ac1b51b6c09fe055c1d7c9176eea9adc33f710818c83a1fbfa073c8dc3a7eb3513cbdd3f5960b7845e31e3e83181e6ba -803d530523fc9e1e0f11040d2412d02baef3f07eeb9b177fa9bfa396af42eea898a4276d56e1db998dc96ae47b644cb2 -85a3c9fc7638f5bf2c3e15ba8c2fa1ae87eb1ceb44c6598c67a2948667a9dfa41e61f66d535b4e7fda62f013a5a8b885 -a961cf5654c46a1a22c29baf7a4e77837a26b7f138f410e9d1883480ed5fa42411d522aba32040b577046c11f007388e -ad1154142344f494e3061ef45a34fab1aaacf5fdf7d1b26adbb5fbc3d795655fa743444e39d9a4119b4a4f82a6f30441 -b1d6c30771130c77806e7ab893b73d4deb590b2ff8f2f8b5e54c2040c1f3e060e2bd99afc668cf706a2df666a508bbf6 -a00361fd440f9decabd98d96c575cd251dc94c60611025095d1201ef2dedde51cb4de7c2ece47732e5ed9b3526c2012c -a85c5ab4d17d328bda5e6d839a9a6adcc92ff844ec25f84981e4f44a0e8419247c081530f8d9aa629c7eb4ca21affba6 -a4ddd3eab4527a2672cf9463db38bc29f61460e2a162f426b7852b7a7645fbd62084fd39a8e4d60e1958cce436dd8f57 -811648140080fe55b8618f4cf17f3c5a250adb0cd53d885f2ddba835d2b4433188e41fc0661faac88e4ff910b16278c0 -b85c7f1cfb0ed29addccf7546023a79249e8f15ac2d14a20accbfef4dd9dc11355d599815fa09d2b6b4e966e6ea8cff1 -a10b5d8c260b159043b020d5dd62b3467df2671afea6d480ca9087b7e60ed170c82b121819d088315902842d66c8fb45 -917e191df1bcf3f5715419c1e2191da6b8680543b1ba41fe84ed07ef570376e072c081beb67b375fca3565a2565bcabb -881fd967407390bfd7badc9ab494e8a287559a01eb07861f527207c127eadea626e9bcc5aa9cca2c5112fbac3b3f0e9c -959fd71149af82cc733619e0e5bf71760ca2650448c82984b3db74030d0e10f8ab1ce1609a6de6f470fe8b5bd90df5b3 -a3370898a1c5f33d15adb4238df9a6c945f18b9ada4ce2624fc32a844f9ece4c916a64e9442225b6592afa06d2e015f2 -817efb8a791435e4236f7d7b278181a5fa34587578c629dbc14fbf9a5c26772290611395eecd20222a4c58649fc256d8 -a04c9876acf2cfdc8ef96de4879742709270fa1d03fe4c8511fbef2d59eb0aaf0336fa2c7dfe41a651157377fa217813 -81e15875d7ea7f123e418edf14099f2e109d4f3a6ce0eb65f67fe9fb10d2f809a864a29f60ad3fc949f89e2596b21783 -b49f529975c09e436e6bc202fdc16e3fdcbe056db45178016ad6fdece9faad4446343e83aed096209690b21a6910724f -879e8eda589e1a279f7f49f6dd0580788c040d973748ec4942dbe51ea8fbd05983cc919b78f0c6b92ef3292ae29db875 -81a2b74b2118923f34139a102f3d95e7eee11c4c2929c2576dee200a5abfd364606158535a6c9e4178a6a83dbb65f3c4 -8913f281d8927f2b45fc815d0f7104631cb7f5f7278a316f1327d670d15868daadd2a64e3eb98e1f53fe7e300338cc80 -a6f815fba7ef9af7fbf45f93bc952e8b351f5de6568a27c7c47a00cb39a254c6b31753794f67940fc7d2e9cc581529f4 -b3722a15c66a0014ce4d082de118def8d39190c15678a472b846225585f3a83756ae1b255b2e3f86a26168878e4773b2 -817ae61ab3d0dd5b6e24846b5a5364b1a7dc2e77432d9fed587727520ae2f307264ea0948c91ad29f0aea3a11ff38624 -b3db467464415fcad36dc1de2d6ba7686772a577cc2619242ac040d6734881a45d3b40ed4588db124e4289cfeec4bbf6 -ad66a14f5a54ac69603b16e5f1529851183da77d3cc60867f10aea41339dd5e06a5257982e9e90a352cdd32750f42ee4 -adafa3681ef45d685555601a25a55cf23358319a17f61e2179e704f63df83a73bdd298d12cf6cef86db89bd17119e11d -a379dc44cb6dd3b9d378c07b2ec654fec7ca2f272de6ba895e3d00d20c9e4c5550498a843c8ac67e4221db2115bedc1c -b7bf81c267a78efc6b9e5a904574445a6487678d7ef70054e3e93ea6a23f966c2b68787f9164918e3b16d2175459ed92 -b41d66a13a4afafd5760062b77f79de7e6ab8ccacde9c6c5116a6d886912fb491dc027af435b1b44aacc6af7b3c887f2 -9904d23a7c1c1d2e4bab85d69f283eb0a8e26d46e8b7b30224438015c936729b2f0af7c7c54c03509bb0500acb42d8a4 -ae30d65e9e20c3bfd603994ae2b175ff691d51f3e24b2d058b3b8556d12ca4c75087809062dddd4aaac81c94d15d8a17 -9245162fab42ac01527424f6013310c3eb462982518debef6c127f46ba8a06c705d7dc9f0a41e796ba8d35d60ae6cc64 -87fab853638d7a29a20f3ba2b1a7919d023e9415bfa78ebb27973d8cbc7626f584dc5665d2e7ad71f1d760eba9700d88 -85aac46ecd330608e5272430970e6081ff02a571e8ea444f1e11785ea798769634a22a142d0237f67b75369d3c484a8a -938c85ab14894cc5dfce3d80456f189a2e98eddbc8828f4ff6b1df1dcb7b42b17ca2ff40226a8a1390a95d63dca698dd -a18ce1f846e3e3c4d846822f60271eecf0f5d7d9f986385ac53c5ace9589dc7c0188910448c19b91341a1ef556652fa9 -8611608a9d844f0e9d7584ad6ccf62a5087a64f764caf108db648a776b5390feb51e5120f0ef0e9e11301af3987dd7dc -8106333ba4b4de8d1ae43bc9735d3fea047392e88efd6a2fa6f7b924a18a7a265ca6123c3edc0f36307dd7fb7fe89257 -a91426fa500951ff1b051a248c050b7139ca30dde8768690432d597d2b3c4357b11a577be6b455a1c5d145264dcf81fc -b7f9f90e0e450f37b081297f7f651bad0496a8b9afd2a4cf4120a2671aaaa8536dce1af301258bfbfdb122afa44c5048 -84126da6435699b0c09fa4032dec73d1fca21d2d19f5214e8b0bea43267e9a8dd1fc44f8132d8315e734c8e2e04d7291 -aff064708103884cb4f1a3c1718b3fc40a238d35cf0a7dc24bdf9823693b407c70da50df585bf5bc4e9c07d1c2d203e8 -a8b40fc6533752983a5329c31d376c7a5c13ce6879cc7faee648200075d9cd273537001fb4c86e8576350eaac6ba60c2 -a02db682bdc117a84dcb9312eb28fcbde12d49f4ce915cc92c610bb6965ec3cc38290f8c5b5ec70afe153956692cda95 -86decd22b25d300508472c9ce75d3e465b737e7ce13bc0fcce32835e54646fe12322ba5bc457be18bfd926a1a6ca4a38 -a18666ef65b8c2904fd598791f5627207165315a85ee01d5fb0e6b2e10bdd9b00babc447da5bd63445e3337de33b9b89 -89bb0c06effadefdaf34ffe4b123e1678a90d4451ee856c863df1e752eef41fd984689ded8f0f878bf8916d5dd8e8024 -97cfcba08ebec05d0073992a66b1d7d6fb9d95871f2cdc36db301f78bf8069294d1c259efef5c93d20dc937eedae3a1a -ac2643b14ece79dcb2e289c96776a47e2bebd40dd6dc74fd035df5bb727b5596f40e3dd2d2202141e69b0993717ede09 -a5e6fd88a2f9174d9bd4c6a55d9c30974be414992f22aa852f552c7648f722ed8077acf5aba030abd47939bb451b2c60 -8ad40a612824a7994487731a40b311b7349038c841145865539c6ada75c56de6ac547a1c23df190e0caaafecddd80ccc -953a7cea1d857e09202c438c6108060961f195f88c32f0e012236d7a4b39d840c61b162ec86436e8c38567328bea0246 -80d8b47a46dae1868a7b8ccfe7029445bbe1009dad4a6c31f9ef081be32e8e1ac1178c3c8fb68d3e536c84990cc035b1 -81ecd99f22b3766ce0aca08a0a9191793f68c754fdec78b82a4c3bdc2db122bbb9ebfd02fc2dcc6e1567a7d42d0cc16a -b1dd0446bccc25846fb95d08c1c9cc52fb51c72c4c5d169ffde56ecfe800f108dc1106d65d5c5bd1087c656de3940b63 -b87547f0931e164e96de5c550ca5aa81273648fe34f6e193cd9d69cf729cb432e17aa02e25b1c27a8a0d20a3b795e94e -820a94e69a927e077082aae66f6b292cfbe4589d932edf9e68e268c9bd3d71ef76cf7d169dd445b93967c25db11f58f1 -b0d07ddf2595270c39adfa0c8cf2ab1322979b0546aa4d918f641be53cd97f36c879bb75d205e457c011aca3bbd9f731 -8700b876b35b4b10a8a9372c5230acecd39539c1bb87515640293ad4464a9e02929d7d6a6a11112e8a29564815ac0de4 -a61a601c5bb27dcb97e37c8e2b9ce479c6b192a5e04d9ed5e065833c5a1017ee5f237b77d1a17be5d48f8e7cc0bcacf6 -92fb88fe774c1ba1d4a08cae3c0e05467ad610e7a3f1d2423fd47751759235fe0a3036db4095bd6404716aa03820f484 -b274f140d77a3ce0796f5e09094b516537ccaf27ae1907099bff172e6368ba85e7c3ef8ea2a07457cac48ae334da95b3 -b2292d9181f16581a9a9142490b2bdcdfb218ca6315d1effc8592100d792eb89d5356996c890441f04f2b4a95763503e -8897e73f576d86bc354baa3bd96e553107c48cf5889dcc23c5ba68ab8bcd4e81f27767be2233fdfa13d39f885087e668 -a29eac6f0829791c728d71abc49569df95a4446ecbfc534b39f24f56c88fe70301838dfc1c19751e7f3c5c1b8c6af6a0 -9346dc3720adc5df500a8df27fd9c75ef38dc5c8f4e8ed66983304750e66d502c3c59b8e955be781b670a0afc70a2167 -9566d534e0e30a5c5f1428665590617e95fd05d45f573715f58157854ad596ece3a3cfec61356aee342308d623e029d5 -a464fb8bffe6bd65f71938c1715c6e296cc6d0311a83858e4e7eb5873b7f2cf0c584d2101e3407b85b64ca78b2ac93ce -b54088f7217987c87e9498a747569ac5b2f8afd5348f9c45bf3fd9fbf713a20f495f49c8572d087efe778ac7313ad6d3 -91fa9f5f8000fe050f5b224d90b59fcce13c77e903cbf98ded752e5b3db16adb2bc1f8c94be48b69f65f1f1ad81d6264 -92d04a5b0ac5d8c8e313709b432c9434ecd3e73231f01e9b4e7952b87df60cbfa97b5dedd2200bd033b4b9ea8ba45cc1 -a94b90ad3c3d6c4bbe169f8661a790c40645b40f0a9d1c7220f01cf7fc176e04d80bab0ced9323fcafb93643f12b2760 -94d86149b9c8443b46196f7e5a3738206dd6f3be7762df488bcbb9f9ee285a64c997ed875b7b16b26604fa59020a8199 -82efe4ae2c50a2d7645240c173a047f238536598c04a2c0b69c96e96bd18e075a99110f1206bc213f39edca42ba00cc1 -ab8667685f831bc14d4610f84a5da27b4ea5b133b4d991741a9e64dceb22cb64a3ce8f1b6e101d52af6296df7127c9ad -83ba433661c05dcc5d562f4a9a261c8110dac44b8d833ae1514b1fc60d8b4ee395b18804baea04cb10adb428faf713c3 -b5748f6f660cc5277f1211d2b8649493ed8a11085b871cd33a5aea630abd960a740f08c08be5f9c21574600ac9bf5737 -a5c8dd12af48fb710642ad65ebb97ca489e8206741807f7acfc334f8035d3c80593b1ff2090c9bb7bd138f0c48714ca8 -a2b382fd5744e3babf454b1d806cc8783efeb4761bc42b6914ea48a46a2eae835efbe0a18262b6bc034379e03cf1262b -b3145ffaf603f69f15a64936d32e3219eea5ed49fdfd2f5bf40ea0dfd974b36fb6ff12164d4c2282d892db4cf3ff3ce1 -87a316fb213f4c5e30c5e3face049db66be4f28821bd96034714ec23d3e97849d7b301930f90a4323c7ccf53de23050c -b9de09a919455070fed6220fc179c8b7a4c753062bcd27acf28f5b9947a659c0b364298daf7c85c4ca6fca7f945add1f -806fbd98d411b76979464c40ad88bc07a151628a27fcc1012ba1dfbaf5b5cc9d962fb9b3386008978a12515edce934bc -a15268877fae0d21610ae6a31061ed7c20814723385955fac09fdc9693a94c33dea11db98bb89fdfe68f933490f5c381 -8d633fb0c4da86b2e0b37d8fad5972d62bff2ac663c5ec815d095cd4b7e1fe66ebef2a2590995b57eaf941983c7ad7a4 -8139e5dd9cf405e8ef65f11164f0440827d98389ce1b418b0c9628be983a9ddd6cf4863036ccb1483b40b8a527acd9ed -88b15fa94a08eac291d2b94a2b30eb851ff24addf2cc30b678e72e32cfcb3424cf4b33aa395d741803f3e578ddf524de -b5eaf0c8506e101f1646bcf049ee38d99ea1c60169730da893fd6020fd00a289eb2f415947e44677af49e43454a7b1be -8489822ad0647a7e06aa2aa5595960811858ddd4542acca419dd2308a8c5477648f4dd969a6740bb78aa26db9bfcc555 -b1e9a7b9f3423c220330d45f69e45fa03d7671897cf077f913c252e3e99c7b1b1cf6d30caad65e4228d5d7b80eb86e5e -b28fe9629592b9e6a55a1406903be76250b1c50c65296c10c5e48c64b539fb08fe11f68cf462a6edcbba71b0cee3feb2 -a41acf96a02c96cd8744ff6577c244fc923810d17ade133587e4c223beb7b4d99fa56eae311a500d7151979267d0895c -880798938fe4ba70721be90e666dfb62fcab4f3556fdb7b0dc8ec5bc34f6b4513df965eae78527136eb391889fe2caf9 -98d4d89d358e0fb7e212498c73447d94a83c1b66e98fc81427ab13acddb17a20f52308983f3a5a8e0aaacec432359604 -81430b6d2998fc78ba937a1639c6020199c52da499f68109da227882dc26d005b73d54c5bdcac1a04e8356a8ca0f7017 -a8d906a4786455eb74613aba4ce1c963c60095ffb8658d368df9266fdd01e30269ce10bf984e7465f34b4fd83beba26a -af54167ac1f954d10131d44a8e0045df00d581dd9e93596a28d157543fbe5fb25d213806ed7fb3cba6b8f5b5423562db -8511e373a978a12d81266b9afbd55035d7bc736835cfa921903a92969eeba3624437d1346b55382e61415726ab84a448 -8cf43eea93508ae586fa9a0f1354a1e16af659782479c2040874a46317f9e8d572a23238efa318fdfb87cc63932602b7 -b0bdd3bacff077173d302e3a9678d1d37936188c7ecc34950185af6b462b7c679815176f3cce5db19aac8b282f2d60ad -a355e9b87f2f2672052f5d4d65b8c1c827d24d89b0d8594641fccfb69aef1b94009105f3242058bb31c8bf51caae5a41 -b8baa9e4b950b72ff6b88a6509e8ed1304bc6fd955748b2e59a523a1e0c5e99f52aec3da7fa9ff407a7adf259652466c -840bc3dbb300ea6f27d1d6dd861f15680bd098be5174f45d6b75b094d0635aced539fa03ddbccb453879de77fb5d1fe9 -b4bc7e7e30686303856472bae07e581a0c0bfc815657c479f9f5931cff208d5c12930d2fd1ff413ebd8424bcd7a9b571 -89b5d514155d7999408334a50822508b9d689add55d44a240ff2bdde2eee419d117031f85e924e2a2c1ca77db9b91eea -a8604b6196f87a04e1350302e8aa745bba8dc162115d22657b37a1d1a98cb14876ddf7f65840b5dbd77e80cd22b4256c -83cb7acdb9e03247515bb2ce0227486ccf803426717a14510f0d59d45e998b245797d356f10abca94f7a14e1a2f0d552 -aeb3266a9f16649210ab2df0e1908ac259f34ce1f01162c22b56cf1019096ee4ea5854c36e30bb2feb06c21a71e8a45c -89e72e86edf2aa032a0fc9acf4d876a40865fbb2c8f87cb7e4d88856295c4ac14583e874142fd0c314a49aba68c0aa3c -8c3576eba0583c2a7884976b4ed11fe1fda4f6c32f6385d96c47b0e776afa287503b397fa516a455b4b8c3afeedc76db -a31e5b633bda9ffa174654fee98b5d5930a691c3c42fcf55673d927dbc8d91c58c4e42e615353145431baa646e8bbb30 -89f2f3f7a8da1544f24682f41c68114a8f78c86bd36b066e27da13acb70f18d9f548773a16bd8e24789420e17183f137 -ada27fa4e90a086240c9164544d2528621a415a5497badb79f8019dc3dce4d12eb6b599597e47ec6ac39c81efda43520 -90dc1eb21bf21c0187f359566fc4bf5386abea52799306a0e5a1151c0817c5f5bc60c86e76b1929c092c0f3ff48cedd2 -b702a53ebcc17ae35d2e735a347d2c700e9cbef8eadbece33cac83df483b2054c126593e1f462cfc00a3ce9d737e2af5 -9891b06455ec925a6f8eafffba05af6a38cc5e193acaaf74ffbf199df912c5197106c5e06d72942bbb032ce277b6417f -8c0ee71eb01197b019275bcf96cae94e81d2cdc3115dbf2d8e3080074260318bc9303597e8f72b18f965ad601d31ec43 -8aaf580aaf75c1b7a5f99ccf60503506e62058ef43b28b02f79b8536a96be3f019c9f71caf327b4e6730134730d1bef5 -ae6f9fc21dd7dfa672b25a87eb0a41644f7609fab5026d5cedb6e43a06dbbfd6d6e30322a2598c8dedde88c52eaed626 -8159b953ffece5693edadb2e906ebf76ff080ee1ad22698950d2d3bfc36ac5ea78f58284b2ca180664452d55bd54716c -ab7647c32ca5e9856ac283a2f86768d68de75ceeba9e58b74c5324f8298319e52183739aba4340be901699d66ac9eb3f -a4d85a5701d89bcfaf1572db83258d86a1a0717603d6f24ac2963ffcf80f1265e5ab376a4529ca504f4396498791253c -816080c0cdbfe61b4d726c305747a9eb58ac26d9a35f501dd32ba43c098082d20faf3ccd41aad24600aa73bfa453dfac -84f3afac024f576b0fd9acc6f2349c2fcefc3f77dbe5a2d4964d14b861b88e9b1810334b908cf3427d9b67a8aee74b18 -94b390655557b1a09110018e9b5a14490681ade275bdc83510b6465a1218465260d9a7e2a6e4ec700f58c31dc3659962 -a8c66826b1c04a2dd4c682543242e7a57acae37278bd09888a3d17747c5b5fec43548101e6f46d703638337e2fd3277b -86e6f4608a00007fa533c36a5b054c5768ccafe41ad52521d772dcae4c8a4bcaff8f7609be30d8fab62c5988cbbb6830 -837da4cf09ae8aa0bceb16f8b3bfcc3b3367aecac9eed6b4b56d7b65f55981ef066490764fb4c108792623ecf8cad383 -941ff3011462f9b5bf97d8cbdb0b6f5d37a1b1295b622f5485b7d69f2cb2bcabc83630dae427f0259d0d9539a77d8424 -b99e5d6d82aa9cf7d5970e7f710f4039ac32c2077530e4c2779250c6b9b373bc380adb0a03b892b652f649720672fc8c -a791c78464b2d65a15440b699e1e30ebd08501d6f2720adbc8255d989a82fcded2f79819b5f8f201bed84a255211b141 -84af7ad4a0e31fcbb3276ab1ad6171429cf39adcf78dc03750dc5deaa46536d15591e26d53e953dfb31e1622bc0743ab -a833e62fe97e1086fae1d4917fbaf09c345feb6bf1975b5cb863d8b66e8d621c7989ab3dbecda36bc9eaffc5eaa6fa66 -b4ef79a46a2126f53e2ebe62770feb57fd94600be29459d70a77c5e9cc260fa892be06cd60f886bf48459e48eb50d063 -b43b8f61919ea380bf151c294e54d3a3ff98e20d1ee5efbfe38aa2b66fafbc6a49739793bd5cb1c809f8b30466277c3a -ab37735af2412d2550e62df9d8b3b5e6f467f20de3890bf56faf1abf2bf3bd1d98dc3fa0ad5e7ab3fce0fa20409eb392 -82416b74b1551d484250d85bb151fabb67e29cce93d516125533df585bc80779ab057ea6992801a3d7d5c6dcff87a018 -8145d0787f0e3b5325190ae10c1d6bee713e6765fb6a0e9214132c6f78f4582bb2771aaeae40d3dad4bafb56bf7e36d8 -b6935886349ecbdd5774e12196f4275c97ec8279fdf28ccf940f6a022ebb6de8e97d6d2173c3fe402cbe9643bed3883b -87ef9b4d3dc71ac86369f8ed17e0dd3b91d16d14ae694bc21a35b5ae37211b043d0e36d8ff07dcc513fb9e6481a1f37f -ae1d0ded32f7e6f1dc8fef495879c1d9e01826f449f903c1e5034aeeabc5479a9e323b162b688317d46d35a42d570d86 -a40d16497004db4104c6794e2f4428d75bdf70352685944f3fbe17526df333e46a4ca6de55a4a48c02ecf0bde8ba03c0 -8d45121efba8cc308a498e8ee39ea6fa5cae9fb2e4aab1c2ff9d448aa8494ccbec9a078f978a86fcd97b5d5e7be7522a -a8173865c64634ba4ac2fa432740f5c05056a9deaf6427cb9b4b8da94ca5ddbc8c0c5d3185a89b8b28878194de9cdfcd -b6ec06a74d690f6545f0f0efba236e63d1fdfba54639ca2617408e185177ece28901c457d02b849fd00f1a53ae319d0a -b69a12df293c014a40070e3e760169b6f3c627caf9e50b35a93f11ecf8df98b2bc481b410eecb7ab210bf213bbe944de -97e7dc121795a533d4224803e591eef3e9008bab16f12472210b73aaf77890cf6e3877e0139403a0d3003c12c8f45636 -acdfa6fdd4a5acb7738cc8768f7cba84dbb95c639399b291ae8e4e63df37d2d4096900a84d2f0606bf534a9ccaa4993f -86ee253f3a9446a33e4d1169719b7d513c6b50730988415382faaf751988c10a421020609f7bcdef91be136704b906e2 -aac9438382a856caf84c5a8a234282f71b5fc5f65219103b147e7e6cf565522285fbfd7417b513bdad8277a00f652ca1 -83f3799d8e5772527930f5dc071a2e0a65471618993ec8990a96ccdeee65270e490bda9d26bb877612475268711ffd80 -93f28a81ac8c0ec9450b9d762fae9c7f8feaace87a6ee6bd141ef1d2d0697ef1bbd159fe6e1de640dbdab2b0361fca8a -a0825c95ba69999b90eac3a31a3fd830ea4f4b2b7409bde5f202b61d741d6326852ce790f41de5cb0eccec7af4db30c1 -83924b0e66233edd603c3b813d698daa05751fc34367120e3cf384ea7432e256ccee4d4daf13858950549d75a377107d -956fd9fa58345277e06ba2ec72f49ed230b8d3d4ff658555c52d6cddeb84dd4e36f1a614f5242d5ca0192e8daf0543c2 -944869912476baae0b114cced4ff65c0e4c90136f73ece5656460626599051b78802df67d7201c55d52725a97f5f29fe -865cb25b64b4531fb6fe4814d7c8cd26b017a6c6b72232ff53defc18a80fe3b39511b23f9e4c6c7249d06e03b2282ed2 -81e09ff55214960775e1e7f2758b9a6c4e4cd39edf7ec1adfaad51c52141182b79fe2176b23ddc7df9fd153e5f82d668 -b31006896f02bc90641121083f43c3172b1039334501fbaf1672f7bf5d174ddd185f945adf1a9c6cf77be34c5501483d -88b92f6f42ae45e9f05b16e52852826e933efd0c68b0f2418ac90957fd018df661bc47c8d43c2a7d7bfcf669dab98c3c -92fc68f595853ee8683930751789b799f397135d002eda244fe63ecef2754e15849edde3ba2f0cc8b865c9777230b712 -99ca06a49c5cd0bb097c447793fcdd809869b216a34c66c78c7e41e8c22f05d09168d46b8b1f3390db9452d91bc96dea -b48b9490a5d65296802431852d548d81047bbefc74fa7dc1d4e2a2878faacdfcb365ae59209cb0ade01901a283cbd15d -aff0fdbef7c188b120a02bc9085d7b808e88f73973773fef54707bf2cd772cd066740b1b6f4127b5c349f657bd97e738 -966fd4463b4f43dd8ccba7ad50baa42292f9f8b2e70da23bb6780e14155d9346e275ef03ddaf79e47020dcf43f3738bd -9330c3e1fadd9e08ac85f4839121ae20bbeb0a5103d84fa5aadbd1213805bdcda67bf2fb75fc301349cbc851b5559d20 -993bb99867bd9041a71a55ad5d397755cfa7ab6a4618fc526179bfc10b7dc8b26e4372fe9a9b4a15d64f2b63c1052dda -a29b59bcfab51f9b3c490a3b96f0bf1934265c315349b236012adbd64a56d7f6941b2c8cc272b412044bc7731f71e1dc -a65c9cefe1fc35d089fe8580c2e7671ebefdb43014ac291528ff4deefd4883fd4df274af83711dad610dad0d615f9d65 -944c78c56fb227ae632805d448ca3884cd3d2a89181cead3d2b7835e63297e6d740aa79a112edb1d4727824991636df5 -a73d782da1db7e4e65d7b26717a76e16dd9fab4df65063310b8e917dc0bc24e0d6755df5546c58504d04d9e68c3b474a -af80f0b87811ae3124f68108b4ca1937009403f87928bbc53480e7c5408d072053ace5eeaf5a5aba814dab8a45502085 -88aaf1acfc6e2e19b8387c97da707cb171c69812fefdd4650468e9b2c627bd5ccfb459f4d8e56bdfd84b09ddf87e128f -92c97276ff6f72bab6e9423d02ad6dc127962dbce15a0dd1e4a393b4510c555df6aa27be0f697c0d847033a9ca8b8dfd -a0e07d43d96e2d85b6276b3c60aadb48f0aedf2de8c415756dc597249ea64d2093731d8735231dadc961e5682ac59479 -adc9e6718a8f9298957d1da3842a7751c5399bbdf56f8de6c1c4bc39428f4aee6f1ba6613d37bf46b9403345e9d6fc81 -951da434da4b20d949b509ceeba02e24da7ed2da964c2fcdf426ec787779c696b385822c7dbea4df3e4a35921f1e912c -a04cbce0d2b2e87bbf038c798a12ec828423ca6aca08dc8d481cf6466e3c9c73d4d4a7fa47df9a7e2e15aae9e9f67208 -8f855cca2e440d248121c0469de1f94c2a71b8ee2682bbad3a78243a9e03da31d1925e6760dbc48a1957e040fae9abe8 -b642e5b17c1df4a4e101772d73851180b3a92e9e8b26c918050f51e6dd3592f102d20b0a1e96f0e25752c292f4c903ff -a92454c300781f8ae1766dbbb50a96192da7d48ef4cbdd72dd8cbb44c6eb5913c112cc38e9144615fdc03684deb99420 -8b74f7e6c2304f8e780df4649ef8221795dfe85fdbdaa477a1542d135b75c8be45bf89adbbb6f3ddf54ca40f02e733e9 -85cf66292cbb30cec5fd835ab10c9fcb3aea95e093aebf123e9a83c26f322d76ebc89c4e914524f6c5f6ee7d74fc917d -ae0bfe0cdc97c09542a7431820015f2d16067b30dca56288013876025e81daa8c519e5e347268e19aa1a85fa1dc28793 -921322fc6a47dc091afa0ad6df18ed14cde38e48c6e71550aa513918b056044983aee402de21051235eecf4ce8040fbe -96c030381e97050a45a318d307dcb3c8377b79b4dd5daf6337cded114de26eb725c14171b9b8e1b3c08fe1f5ea6b49e0 -90c23b86b6111818c8baaf53a13eaee1c89203b50e7f9a994bf0edf851919b48edbac7ceef14ac9414cf70c486174a77 -8bf6c301240d2d1c8d84c71d33a6dfc6d9e8f1cfae66d4d0f7a256d98ae12b0bcebfa94a667735ee89f810bcd7170cff -a41a4ffbbea0e36874d65c009ee4c3feffff322f6fc0e30d26ee4dbc1f46040d05e25d9d0ecb378cef0d24a7c2c4b850 -a8d4cdd423986bb392a0a92c12a8bd4da3437eec6ef6af34cf5310944899287452a2eb92eb5386086d5063381189d10e -a81dd26ec057c4032a4ed7ad54d926165273ed51d09a1267b2e477535cf6966835a257c209e4e92d165d74fa75695fa3 -8d7f708c3ee8449515d94fc26b547303b53d8dd55f177bc3b25d3da2768accd9bc8e9f09546090ebb7f15c66e6c9c723 -839ba65cffcd24cfffa7ab3b21faabe3c66d4c06324f07b2729c92f15cad34e474b0f0ddb16cd652870b26a756b731d3 -87f1a3968afec354d92d77e2726b702847c6afcabb8438634f9c6f7766de4c1504317dc4fa9a4a735acdbf985e119564 -91a8a7fd6542f3e0673f07f510d850864b34ac087eb7eef8845a1d14b2b1b651cbdc27fa4049bdbf3fea54221c5c8549 -aef3cf5f5e3a2385ead115728d7059e622146c3457d266c612e778324b6e06fbfb8f98e076624d2f3ce1035d65389a07 -819915d6232e95ccd7693fdd78d00492299b1983bc8f96a08dcb50f9c0a813ed93ae53c0238345d5bea0beda2855a913 -8e9ba68ded0e94935131b392b28218315a185f63bf5e3c1a9a9dd470944509ca0ba8f6122265f8da851b5cc2abce68f1 -b28468e9b04ee9d69003399a3cf4457c9bf9d59f36ab6ceeb8e964672433d06b58beeea198fedc7edbaa1948577e9fa2 -a633005e2c9f2fd94c8bce2dd5bb708fe946b25f1ec561ae65e54e15cdd88dc339f1a083e01f0d39610c8fe24151aaf0 -841d0031e22723f9328dd993805abd13e0c99b0f59435d2426246996b08d00ce73ab906f66c4eab423473b409e972ce0 -85758d1b084263992070ec8943f33073a2d9b86a8606672550c17545507a5b3c88d87382b41916a87ee96ff55a7aa535 -8581b06b0fc41466ef94a76a1d9fb8ae0edca6d018063acf6a8ca5f4b02d76021902feba58972415691b4bdbc33ae3b4 -83539597ff5e327357ee62bc6bf8c0bcaec2f227c55c7c385a4806f0d37fb461f1690bad5066b8a5370950af32fafbef -aee3557290d2dc10827e4791d00e0259006911f3f3fce4179ed3c514b779160613eca70f720bff7804752715a1266ffa -b48d2f0c4e90fc307d5995464e3f611a9b0ef5fe426a289071f4168ed5cc4f8770c9332960c2ca5c8c427f40e6bb389f -847af8973b4e300bb06be69b71b96183fd1a0b9d51b91701bef6fcfde465068f1eb2b1503b07afda380f18d69de5c9e1 -a70a6a80ce407f07804c0051ac21dc24d794b387be94eb24e1db94b58a78e1bcfb48cd0006db8fc1f9bedaece7a44fbe -b40e942b8fa5336910ff0098347df716bff9d1fa236a1950c16eeb966b3bc1a50b8f7b0980469d42e75ae13ced53cead -b208fabaa742d7db3148515330eb7a3577487845abdb7bd9ed169d0e081db0a5816595c33d375e56aeac5b51e60e49d3 -b7c8194b30d3d6ef5ab66ec88ad7ebbc732a3b8a41731b153e6f63759a93f3f4a537eab9ad369705bd730184bdbbdc34 -9280096445fe7394d04aa1bc4620c8f9296e991cc4d6c131bd703cb1cc317510e6e5855ac763f4d958c5edfe7eebeed7 -abc2aa4616a521400af1a12440dc544e3c821313d0ab936c86af28468ef8bbe534837e364598396a81cf8d06274ed5a6 -b18ca8a3325adb0c8c18a666d4859535397a1c3fe08f95eebfac916a7a99bbd40b3c37b919e8a8ae91da38bc00fa56c0 -8a40c33109ecea2a8b3558565877082f79121a432c45ec2c5a5e0ec4d1c203a6788e6b69cb37f1fd5b8c9a661bc5476d -88c47301dd30998e903c84e0b0f2c9af2e1ce6b9f187dab03528d44f834dc991e4c86d0c474a2c63468cf4020a1e24a0 -920c832853e6ab4c851eecfa9c11d3acc7da37c823be7aa1ab15e14dfd8beb5d0b91d62a30cec94763bd8e4594b66600 -98e1addbe2a6b8edc7f12ecb9be81c3250aeeca54a1c6a7225772ca66549827c15f3950d01b8eb44aecb56fe0fff901a -8cfb0fa1068be0ec088402f5950c4679a2eb9218c729da67050b0d1b2d7079f3ddf4bf0f57d95fe2a8db04bc6bcdb20c -b70f381aafe336b024120453813aeab70baac85b9c4c0f86918797b6aee206e6ed93244a49950f3d8ec9f81f4ac15808 -a4c8edf4aa33b709a91e1062939512419711c1757084e46f8f4b7ed64f8e682f4e78b7135920c12f0eb0422fe9f87a6a -b4817e85fd0752d7ebb662d3a51a03367a84bac74ebddfba0e5af5e636a979500f72b148052d333b3dedf9edd2b4031b -a87430169c6195f5d3e314ff2d1c2f050e766fd5d2de88f5207d72dba4a7745bb86d0baca6e9ae156582d0d89e5838c7 -991b00f8b104566b63a12af4826b61ce7aa40f4e5b8fff3085e7a99815bdb4471b6214da1e480214fac83f86a0b93cc5 -b39966e3076482079de0678477df98578377a094054960ee518ef99504d6851f8bcd3203e8da5e1d4f6f96776e1fe6eb -a448846d9dc2ab7a0995fa44b8527e27f6b3b74c6e03e95edb64e6baa4f1b866103f0addb97c84bef1d72487b2e21796 -894bec21a453ae84b592286e696c35bc30e820e9c2fd3e63dd4fbe629e07df16439c891056070faa490155f255bf7187 -a9ec652a491b11f6a692064e955f3f3287e7d2764527e58938571469a1e29b5225b9415bd602a45074dfbfe9c131d6ca -b39d37822e6cbe28244b5f42ce467c65a23765bd16eb6447c5b3e942278069793763483dafd8c4dd864f8917aad357fe -88dba51133f2019cb266641c56101e3e5987d3b77647a2e608b5ff9113dfc5f85e2b7c365118723131fbc0c9ca833c9c -b566579d904b54ecf798018efcb824dccbebfc6753a0fd2128ac3b4bd3b038c2284a7c782b5ca6f310eb7ea4d26a3f0a -a97a55c0a492e53c047e7d6f9d5f3e86fb96f3dddc68389c0561515343b66b4bc02a9c0d5722dff1e3445308240b27f7 -a044028ab4bcb9e1a2b9b4ca4efbf04c5da9e4bf2fff0e8bd57aa1fc12a71e897999c25d9117413faf2f45395dee0f13 -a78dc461decbeaeed8ebd0909369b491a5e764d6a5645a7dac61d3140d7dc0062526f777b0eb866bff27608429ebbdde -b2c2a8991f94c39ca35fea59f01a92cb3393e0eccb2476dfbf57261d406a68bd34a6cff33ed80209991688c183609ef4 -84189eefb521aff730a4fd3fd5b10ddfd29f0d365664caef63bb015d07e689989e54c33c2141dd64427805d37a7e546e -85ac80bd734a52235da288ff042dea9a62e085928954e8eacd2c751013f61904ed110e5b3afe1ab770a7e6485efb7b5e -9183a560393dcb22d0d5063e71182020d0fbabb39e32493eeffeb808df084aa243eb397027f150b55a247d1ed0c8513e -81c940944df7ecc58d3c43c34996852c3c7915ed185d7654627f7af62abae7e0048dd444a6c09961756455000bd96d09 -aa8c34e164019743fd8284b84f06c3b449aae7996e892f419ee55d82ad548cb300fd651de329da0384243954c0ef6a60 -89a7b7bdfc7e300d06a14d463e573d6296d8e66197491900cc9ae49504c4809ff6e61b758579e9091c61085ba1237b83 -878d21809ba540f50bd11f4c4d9590fb6f3ab9de5692606e6e2ef4ed9d18520119e385be5e1f4b3f2e2b09c319f0e8fc -8eb248390193189cf0355365e630b782cd15751e672dc478b39d75dc681234dcd9309df0d11f4610dbb249c1e6be7ef9 -a1d7fb3aecb896df3a52d6bd0943838b13f1bd039c936d76d03de2044c371d48865694b6f532393b27fd10a4cf642061 -a34bca58a24979be442238cbb5ece5bee51ae8c0794dd3efb3983d4db713bc6f28a96e976ac3bd9a551d3ed9ba6b3e22 -817c608fc8cacdd178665320b5a7587ca21df8bdd761833c3018b967575d25e3951cf3d498a63619a3cd2ad4406f5f28 -86c95707db0495689afd0c2e39e97f445f7ca0edffad5c8b4cacd1421f2f3cc55049dfd504f728f91534e20383955582 -99c3b0bb15942c301137765d4e19502f65806f3b126dc01a5b7820c87e8979bce6a37289a8f6a4c1e4637227ad5bf3bf -8aa1518a80ea8b074505a9b3f96829f5d4afa55a30efe7b4de4e5dbf666897fdd2cf31728ca45921e21a78a80f0e0f10 -8d74f46361c79e15128ac399e958a91067ef4cec8983408775a87eca1eed5b7dcbf0ddf30e66f51780457413496c7f07 -a41cde4a786b55387458a1db95171aca4fd146507b81c4da1e6d6e495527c3ec83fc42fad1dfe3d92744084a664fd431 -8c352852c906fae99413a84ad11701f93f292fbf7bd14738814f4c4ceab32db02feb5eb70bc73898b0bc724a39d5d017 -a5993046e8f23b71ba87b7caa7ace2d9023fb48ce4c51838813174880d918e9b4d2b0dc21a2b9c6f612338c31a289df8 -83576d3324bf2d8afbfb6eaecdc5d767c8e22e7d25160414924f0645491df60541948a05e1f4202e612368e78675de8a -b43749b8df4b15bc9a3697e0f1c518e6b04114171739ef1a0c9c65185d8ec18e40e6954d125cbc14ebc652cf41ad3109 -b4eebd5d80a7327a040cafb9ccdb12b2dfe1aa86e6bc6d3ac8a57fadfb95a5b1a7332c66318ff72ba459f525668af056 -9198be7f1d413c5029b0e1c617bcbc082d21abe2c60ec8ce9b54ca1a85d3dba637b72fda39dae0c0ae40d047eab9f55a -8d96a0232832e24d45092653e781e7a9c9520766c3989e67bbe86b3a820c4bf621ea911e7cd5270a4bfea78b618411f6 -8d7160d0ea98161a2d14d46ef01dff72d566c330cd4fabd27654d300e1bc7644c68dc8eabf2a20a59bfe7ba276545f9b -abb60fce29dec7ba37e3056e412e0ec3e05538a1fc0e2c68877378c867605966108bc5742585ab6a405ce0c962b285b6 -8fabffa3ed792f05e414f5839386f6449fd9f7b41a47595c5d71074bd1bb3784cc7a1a7e1ad6b041b455035957e5b2dc -90ff017b4804c2d0533b72461436b10603ab13a55f86fd4ec11b06a70ef8166f958c110519ca1b4cc7beba440729fe2d -b340cfd120f6a4623e3a74cf8c32bfd7cd61a280b59dfd17b15ca8fae4d82f64a6f15fbde4c02f424debc72b7db5fe67 -871311c9c7220c932e738d59f0ecc67a34356d1429fe570ca503d340c9996cb5ee2cd188fad0e3bd16e4c468ec1dbebd -a772470262186e7b94239ba921b29f2412c148d6f97c4412e96d21e55f3be73f992f1ad53c71008f0558ec3f84e2b5a7 -b2a897dcb7ffd6257f3f2947ec966f2077d57d5191a88840b1d4f67effebe8c436641be85524d0a21be734c63ab5965d -a044f6eacc48a4a061fa149500d96b48cbf14853469aa4d045faf3dca973be1bd4b4ce01646d83e2f24f7c486d03205d -981af5dc2daa73f7fa9eae35a93d81eb6edba4a7f673b55d41f6ecd87a37685d31bb40ef4f1c469b3d72f2f18b925a17 -912d2597a07864de9020ac77083eff2f15ceb07600f15755aba61251e8ce3c905a758453b417f04d9c38db040954eb65 -9642b7f6f09394ba5e0805734ef6702c3eddf9eea187ba98c676d5bbaec0e360e3e51dc58433aaa1e2da6060c8659cb7 -8ab3836e0a8ac492d5e707d056310c4c8e0489ca85eb771bff35ba1d658360084e836a6f51bb990f9e3d2d9aeb18fbb5 -879e058e72b73bb1f4642c21ffdb90544b846868139c6511f299aafe59c2d0f0b944dffc7990491b7c4edcd6a9889250 -b9e60b737023f61479a4a8fd253ed0d2a944ea6ba0439bbc0a0d3abf09b0ad1f18d75555e4a50405470ae4990626f390 -b9c2535d362796dcd673640a9fa2ebdaec274e6f8b850b023153b0a7a30fffc87f96e0b72696f647ebe7ab63099a6963 -94aeff145386a087b0e91e68a84a5ede01f978f9dd9fe7bebca78941938469495dc30a96bba9508c0d017873aeea9610 -98b179f8a3d9f0d0a983c30682dd425a2ddc7803be59bd626c623c8951a5179117d1d2a68254c95c9952989877d0ee55 -889ecf5f0ee56938273f74eb3e9ecfb5617f04fb58e83fe4c0e4aef51615cf345bc56f3f61b17f6eed3249d4afd54451 -a0f2b2c39bcea4b50883e2587d16559e246248a66ecb4a4b7d9ab3b51fb39fe98d83765e087eee37a0f86b0ba4144c02 -b2a61e247ed595e8a3830f7973b07079cbda510f28ad8c78c220b26cb6acde4fbb5ee90c14a665f329168ee951b08cf0 -95bd0fcfb42f0d6d8a8e73d7458498a85bcddd2fb132fd7989265648d82ac2707d6d203fac045504977af4f0a2aca4b7 -843e5a537c298666e6cf50fcc044f13506499ef83c802e719ff2c90e85003c132024e04711be7234c04d4b0125512d5d -a46d1797c5959dcd3a5cfc857488f4d96f74277c3d13b98b133620192f79944abcb3a361d939a100187f1b0856eae875 -a1c7786736d6707a48515c38660615fcec67eb8a2598f46657855215f804fd72ab122d17f94fcffad8893f3be658dca7 -b23dc9e610abc7d8bd21d147e22509a0fa49db5be6ea7057b51aae38e31654b3aa044df05b94b718153361371ba2f622 -b00cc8f257d659c22d30e6d641f79166b1e752ea8606f558e4cad6fc01532e8319ea4ee12265ba4140ac45aa4613c004 -ac7019af65221b0cc736287b32d7f1a3561405715ba9a6a122342e04e51637ba911c41573de53e4781f2230fdcb2475f -81a630bc41b3da8b3eb4bf56cba10cd9f93153c3667f009dc332287baeb707d505fb537e6233c8e53d299ec0f013290c -a6b7aea5c545bb76df0f230548539db92bc26642572cb7dd3d5a30edca2b4c386f44fc8466f056b42de2a452b81aff5b -8271624ff736b7b238e43943c81de80a1612207d32036d820c11fc830c737972ccc9c60d3c2359922b06652311e3c994 -8a684106458cb6f4db478170b9ad595d4b54c18bf63b9058f095a2fa1b928c15101472c70c648873d5887880059ed402 -a5cc3c35228122f410184e4326cf61a37637206e589fcd245cb5d0cec91031f8f7586b80503070840fdfd8ce75d3c88b -9443fc631aed8866a7ed220890911057a1f56b0afe0ba15f0a0e295ab97f604b134b1ed9a4245e46ee5f9a93aa74f731 -984b6f7d79835dffde9558c6bb912d992ca1180a2361757bdba4a7b69dc74b056e303adc69fe67414495dd9c2dd91e64 -b15a5c8cba5de080224c274d31c68ed72d2a7126d347796569aef0c4e97ed084afe3da4d4b590b9dda1a07f0c2ff3dfb -991708fe9650a1f9a4e43938b91d45dc68c230e05ee999c95dbff3bf79b1c1b2bb0e7977de454237c355a73b8438b1d9 -b4f7edc7468b176a4a7c0273700c444fa95c726af6697028bed4f77eee887e3400f9c42ee15b782c0ca861c4c3b8c98a -8c60dcc16c51087eb477c13e837031d6c6a3dc2b8bf8cb43c23f48006bc7173151807e866ead2234b460c2de93b31956 -83ad63e9c910d1fc44bc114accfb0d4d333b7ebe032f73f62d25d3e172c029d5e34a1c9d547273bf6c0fead5c8801007 -85de73213cc236f00777560756bdbf2b16841ba4b55902cf2cad9742ecaf5d28209b012ceb41f337456dfeca93010cd7 -a7561f8827ccd75b6686ba5398bb8fc3083351c55a589b18984e186820af7e275af04bcd4c28e1dc11be1e8617a0610b -88c0a4febd4068850557f497ea888035c7fc9f404f6cc7794e7cc8722f048ad2f249e7dc62743e7a339eb7473ad3b0cd -932b22b1d3e6d5a6409c34980d176feb85ada1bf94332ef5c9fc4d42b907dabea608ceef9b5595ef3feee195151f18d8 -a2867bb3f5ab88fbdae3a16c9143ab8a8f4f476a2643c505bb9f37e5b1fd34d216cab2204c9a017a5a67b7ad2dda10e8 -b573d5f38e4e9e8a3a6fd82f0880dc049efa492a946d00283019bf1d5e5516464cf87039e80aef667cb86fdea5075904 -b948f1b5ab755f3f5f36af27d94f503b070696d793b1240c1bdfd2e8e56890d69e6904688b5f8ff5a4bdf5a6abfe195f -917eae95ebc4109a2e99ddd8fec7881d2f7aaa0e25fda44dec7ce37458c2ee832f1829db7d2dcfa4ca0f06381c7fe91d -95751d17ed00a3030bce909333799bb7f4ab641acf585807f355b51d6976dceee410798026a1a004ef4dcdff7ec0f5b8 -b9b7bd266f449a79bbfe075e429613e76c5a42ac61f01c8f0bbbd34669650682efe01ff9dbbc400a1e995616af6aa278 -ac1722d097ce9cd7617161f8ec8c23d68f1fb1c9ca533e2a8b4f78516c2fd8fb38f23f834e2b9a03bb06a9d655693ca9 -a7ad9e96ffd98db2ecdb6340c5d592614f3c159abfd832fe27ee9293519d213a578e6246aae51672ee353e3296858873 -989b8814d5de7937c4acafd000eec2b4cd58ba395d7b25f98cafd021e8efa37029b29ad8303a1f6867923f5852a220eb -a5bfe6282c771bc9e453e964042d44eff4098decacb89aecd3be662ea5b74506e1357ab26f3527110ba377711f3c9f41 -8900a7470b656639721d2abbb7b06af0ac4222ab85a1976386e2a62eb4b88bfb5b72cf7921ddb3cf3a395d7eeb192a2e -95a71b55cd1f35a438cf5e75f8ff11c5ec6a2ebf2e4dba172f50bfad7d6d5dca5de1b1afc541662c81c858f7604c1163 -82b5d62fea8db8d85c5bc3a76d68dedd25794cf14d4a7bc368938ffca9e09f7e598fdad2a5aac614e0e52f8112ae62b9 -997173f07c729202afcde3028fa7f52cefc90fda2d0c8ac2b58154a5073140683e54c49ed1f254481070d119ce0ce02a -aeffb91ccc7a72bbd6ffe0f9b99c9e66e67d59cec2e02440465e9636a613ab3017278cfa72ea8bc4aba9a8dc728cb367 -952743b06e8645894aeb6440fc7a5f62dd3acf96dab70a51e20176762c9751ea5f2ba0b9497ccf0114dc4892dc606031 -874c63baeddc56fbbca2ff6031f8634b745f6e34ea6791d7c439201aee8f08ef5ee75f7778700a647f3b21068513fce6 -85128fec9c750c1071edfb15586435cc2f317e3e9a175bb8a9697bcda1eb9375478cf25d01e7fed113483b28f625122d -85522c9576fd9763e32af8495ae3928ed7116fb70d4378448926bc9790e8a8d08f98cf47648d7da1b6e40d6a210c7924 -97d0f37a13cfb723b848099ca1c14d83e9aaf2f7aeb71829180e664b7968632a08f6a85f557d74b55afe6242f2a36e7c -abaa472d6ad61a5fccd1a57c01aa1bc081253f95abbcba7f73923f1f11c4e79b904263890eeb66926de3e2652f5d1c70 -b3c04945ba727a141e5e8aec2bf9aa3772b64d8fd0e2a2b07f3a91106a95cbcb249adcd074cbe498caf76fffac20d4ef -82c46781a3d730d9931bcabd7434a9171372dde57171b6180e5516d4e68db8b23495c8ac3ab96994c17ddb1cf249b9fb -a202d8b65613c42d01738ccd68ed8c2dbc021631f602d53f751966e04182743ebc8e0747d600b8a8676b1da9ae7f11ab -ae73e7256e9459db04667a899e0d3ea5255211fb486d084e6550b6dd64ca44af6c6b2d59d7aa152de9f96ce9b58d940d -b67d87b176a9722945ec7593777ee461809861c6cfd1b945dde9ee4ff009ca4f19cf88f4bbb5c80c9cbab2fe25b23ac8 -8f0b7a317a076758b0dac79959ee4a06c08b07d0f10538a4b53d3da2eda16e2af26922feb32c090330dc4d969cf69bd3 -90b36bf56adbd8c4b6cb32febc3a8d5f714370c2ac3305c10fa6d168dffb2a026804517215f9a2d4ec8310cdb6bb459b -aa80c19b0682ead69934bf18cf476291a0beddd8ef4ed75975d0a472e2ab5c70f119722a8574ae4973aceb733d312e57 -a3fc9abb12574e5c28dcb51750b4339b794b8e558675eef7d26126edf1de920c35e992333bcbffcbf6a5f5c0d383ce62 -a1573ff23ab972acdcd08818853b111fc757fdd35aa070186d3e11e56b172fb49d840bf297ac0dd222e072fc09f26a81 -98306f2be4caa92c2b4392212d0cbf430b409b19ff7d5b899986613bd0e762c909fc01999aa94be3bd529d67f0113d7f -8c1fc42482a0819074241746d17dc89c0304a2acdae8ed91b5009e9e3e70ff725ba063b4a3e68fdce05b74f5180c545e -a6c6113ebf72d8cf3163b2b8d7f3fa24303b13f55752522c660a98cd834d85d8c79214d900fa649499365e2e7641f77a -ab95eea424f8a2cfd9fb1c78bb724e5b1d71a0d0d1e4217c5d0f98b0d8bbd3f8400a2002abc0a0e4576d1f93f46fefad -823c5a4fd8cf4a75fdc71d5f2dd511b6c0f189b82affeacd2b7cfcad8ad1a5551227dcc9bfdb2e34b2097eaa00efbb51 -b97314dfff36d80c46b53d87a61b0e124dc94018a0bb680c32765b9a2d457f833a7c42bbc90b3b1520c33a182580398d -b17566ee3dcc6bb3b004afe4c0136dfe7dd27df9045ae896dca49fb36987501ae069eb745af81ba3fc19ff037e7b1406 -b0bdc0f55cfd98d331e3a0c4fbb776a131936c3c47c6bffdc3aaf7d8c9fa6803fbc122c2fefbb532e634228687d52174 -aa5d9e60cc9f0598559c28bb9bdd52aa46605ab4ffe3d192ba982398e72cec9a2a44c0d0d938ce69935693cabc0887ea -802b6459d2354fa1d56c592ac1346c428dadea6b6c0a87bf7d309bab55c94e1cf31dd98a7a86bd92a840dd51f218b91b -a526914efdc190381bf1a73dd33f392ecf01350b9d3f4ae96b1b1c3d1d064721c7d6eec5788162c933245a3943f5ee51 -b3b8fcf637d8d6628620a1a99dbe619eabb3e5c7ce930d6efd2197e261bf394b74d4e5c26b96c4b8009c7e523ccfd082 -8f7510c732502a93e095aba744535f3928f893f188adc5b16008385fb9e80f695d0435bfc5b91cdad4537e87e9d2551c -97b90beaa56aa936c3ca45698f79273a68dd3ccd0076eab48d2a4db01782665e63f33c25751c1f2e070f4d1a8525bf96 -b9fb798324b1d1283fdc3e48288e3861a5449b2ab5e884b34ebb8f740225324af86e4711da6b5cc8361c1db15466602f -b6d52b53cea98f1d1d4c9a759c25bf9d8a50b604b144e4912acbdbdc32aab8b9dbb10d64a29aa33a4f502121a6fb481c -9174ffff0f2930fc228f0e539f5cfd82c9368d26b074467f39c07a774367ff6cccb5039ac63f107677d77706cd431680 -a33b6250d4ac9e66ec51c063d1a6a31f253eb29bbaed12a0d67e2eccfffb0f3a52750fbf52a1c2aaba8c7692346426e7 -a97025fd5cbcebe8ef865afc39cd3ea707b89d4e765ec817fd021d6438e02fa51e3544b1fd45470c58007a08efac6edd -b32a78480edd9ff6ba2f1eec4088db5d6ceb2d62d7e59e904ecaef7bb4a2e983a4588e51692b3be76e6ffbc0b5f911a5 -b5ab590ef0bb77191f00495b33d11c53c65a819f7d0c1f9dc4a2caa147a69c77a4fff7366a602d743ee1f395ce934c1e -b3fb0842f9441fb1d0ee0293b6efbc70a8f58d12d6f769b12872db726b19e16f0f65efbc891cf27a28a248b0ef9c7e75 -9372ad12856fefb928ccb0d34e198df99e2f8973b07e9d417a3134d5f69e12e79ff572c4e03ccd65415d70639bc7c73e -aa8d6e83d09ce216bfe2009a6b07d0110d98cf305364d5529c170a23e693aabb768b2016befb5ada8dabdd92b4d012bb -a954a75791eeb0ce41c85200c3763a508ed8214b5945a42c79bfdcfb1ec4f86ad1dd7b2862474a368d4ac31911a2b718 -8e2081cfd1d062fe3ab4dab01f68062bac802795545fede9a188f6c9f802cb5f884e60dbe866710baadbf55dc77c11a4 -a2f06003b9713e7dd5929501ed485436b49d43de80ea5b15170763fd6346badf8da6de8261828913ee0dacd8ff23c0e1 -98eecc34b838e6ffd1931ca65eec27bcdb2fdcb61f33e7e5673a93028c5865e0d1bf6d3bec040c5e96f9bd08089a53a4 -88cc16019741b341060b95498747db4377100d2a5bf0a5f516f7dec71b62bcb6e779de2c269c946d39040e03b3ae12b7 -ad1135ccbc3019d5b2faf59a688eef2500697642be8cfbdf211a1ab59abcc1f24483e50d653b55ff1834675ac7b4978f -a946f05ed9972f71dfde0020bbb086020fa35b482cce8a4cc36dd94355b2d10497d7f2580541bb3e81b71ac8bba3c49f -a83aeed488f9a19d8cfd743aa9aa1982ab3723560b1cd337fc2f91ad82f07afa412b3993afb845f68d47e91ba4869840 -95eebe006bfc316810cb71da919e5d62c2cebb4ac99d8e8ef67be420302320465f8b69873470982de13a7c2e23516be9 -a55f8961295a11e91d1e5deadc0c06c15dacbfc67f04ccba1d069cba89d72aa3b3d64045579c3ea8991b150ac29366ae -b321991d12f6ac07a5de3c492841d1a27b0d3446082fbce93e7e1f9e8d8fe3b45d41253556261c21b70f5e189e1a7a6f -a0b0822f15f652ce7962a4f130104b97bf9529797c13d6bd8e24701c213cc37f18157bd07f3d0f3eae6b7cd1cb40401f -96e2fa4da378aa782cc2d5e6e465fc9e49b5c805ed01d560e9b98abb5c0de8b74a2e7bec3aa5e2887d25cccb12c66f0c -97e4ab610d414f9210ed6f35300285eb3ccff5b0b6a95ed33425100d7725e159708ea78704497624ca0a2dcabce3a2f9 -960a375b17bdb325761e01e88a3ea57026b2393e1d887b34b8fa5d2532928079ce88dc9fd06a728b26d2bb41b12b9032 -8328a1647398e832aadc05bd717487a2b6fcdaa0d4850d2c4da230c6a2ed44c3e78ec4837b6094f3813f1ee99414713f -aa283834ebd18e6c99229ce4b401eda83f01d904f250fedd4e24f1006f8fa0712a6a89a7296a9bf2ce8de30e28d1408e -b29e097f2caadae3e0f0ae3473c072b0cd0206cf6d2e9b22c1a5ad3e07d433e32bd09ed1f4e4276a2da4268633357b7f -9539c5cbba14538b2fe077ecf67694ef240da5249950baaabea0340718b882a966f66d97f08556b08a4320ceb2cc2629 -b4529f25e9b42ae8cf8338d2eface6ba5cd4b4d8da73af502d081388135c654c0b3afb3aa779ffc80b8c4c8f4425dd2b -95be0739c4330619fbe7ee2249c133c91d6c07eab846c18c5d6c85fc21ac5528c5d56dcb0145af68ed0c6a79f68f2ccd -ac0c83ea802227bfc23814a24655c9ff13f729619bcffdb487ccbbf029b8eaee709f8bddb98232ef33cd70e30e45ca47 -b503becb90acc93b1901e939059f93e671900ca52c6f64ae701d11ac891d3a050b505d89324ce267bc43ab8275da6ffe -98e3811b55b1bacb70aa409100abb1b870f67e6d059475d9f278c751b6e1e2e2d6f2e586c81a9fb6597fda06e7923274 -b0b0f61a44053fa6c715dbb0731e35d48dba257d134f851ee1b81fd49a5c51a90ebf5459ec6e489fce25da4f184fbdb1 -b1d2117fe811720bb997c7c93fe9e4260dc50fca8881b245b5e34f724aaf37ed970cdad4e8fcb68e05ac8cf55a274a53 -a10f502051968f14b02895393271776dee7a06db9de14effa0b3471825ba94c3f805302bdddac4d397d08456f620999d -a3dbad2ef060ae0bb7b02eaa4a13594f3f900450faa1854fc09620b01ac94ab896321dfb1157cf2374c27e5718e8026a -b550fdec503195ecb9e079dcdf0cad559d64d3c30818ef369b4907e813e689da316a74ad2422e391b4a8c2a2bef25fc0 -a25ba865e2ac8f28186cea497294c8649a201732ecb4620c4e77b8e887403119910423df061117e5f03fc5ba39042db1 -b3f88174e03fdb443dd6addd01303cf88a4369352520187c739fc5ae6b22fa99629c63c985b4383219dab6acc5f6f532 -97a7503248e31e81b10eb621ba8f5210c537ad11b539c96dfb7cf72b846c7fe81bd7532c5136095652a9618000b7f8d3 -a8bcdc1ce5aa8bfa683a2fc65c1e79de8ff5446695dcb8620f7350c26d2972a23da22889f9e2b1cacb3f688c6a2953dc -8458c111df2a37f5dd91a9bee6c6f4b79f4f161c93fe78075b24a35f9817da8dde71763218d627917a9f1f0c4709c1ed -ac5f061a0541152b876cbc10640f26f1cc923c9d4ae1b6621e4bb3bf2cec59bbf87363a4eb72fb0e5b6d4e1c269b52d5 -a9a25ca87006e8a9203cbb78a93f50a36694aa4aad468b8d80d3feff9194455ca559fcc63838128a0ab75ad78c07c13a -a450b85f5dfffa8b34dfd8bc985f921318efacf8857cf7948f93884ba09fb831482ee90a44224b1a41e859e19b74962f -8ed91e7f92f5c6d7a71708b6132f157ac226ecaf8662af7d7468a4fa25627302efe31e4620ad28719318923e3a59bf82 -ab524165fd4c71b1fd395467a14272bd2b568592deafa039d8492e9ef36c6d3f96927c95c72d410a768dc0b6d1fbbc9b -b662144505aa8432c75ffb8d10318526b6d5777ac7af9ebfad87d9b0866c364f7905a6352743bd8fd79ffd9d5dd4f3e6 -a48f1677550a5cd40663bb3ba8f84caaf8454f332d0ceb1d94dbea52d0412fe69c94997f7749929712fd3995298572f7 -8391cd6e2f6b0c242de1117a612be99776c3dc95cb800b187685ea5bf7e2722275eddb79fd7dfc8be8e389c4524cdf70 -875d3acb9af47833b72900bc0a2448999d638f153c5e97e8a14ec02d0c76f6264353a7e275e1f1a5855daced523d243b -91f1823657d30b59b2f627880a9a9cb530f5aca28a9fd217fe6f2f5133690dfe7ad5a897872e400512db2e788b3f7628 -ad3564332aa56cea84123fc7ca79ea70bb4fef2009fa131cb44e4b15e8613bd11ca1d83b9d9bf456e4b7fee9f2e8b017 -8c530b84001936d5ab366c84c0b105241a26d1fb163669f17c8f2e94776895c2870edf3e1bc8ccd04d5e65531471f695 -932d01fa174fdb0c366f1230cffde2571cc47485f37f23ba5a1825532190cc3b722aeb1f15aed62cf83ccae9403ba713 -88b28c20585aca50d10752e84b901b5c2d58efef5131479fbbe53de7bce2029e1423a494c0298e1497669bd55be97a5d -b914148ca717721144ebb3d3bf3fcea2cd44c30c5f7051b89d8001502f3856fef30ec167174d5b76265b55d70f8716b5 -81d0173821c6ddd2a068d70766d9103d1ee961c475156e0cbd67d54e668a796310474ef698c7ab55abe6f2cf76c14679 -8f28e8d78e2fe7fa66340c53718e0db4b84823c8cfb159c76eac032a62fb53da0a5d7e24ca656cf9d2a890cb2a216542 -8a26360335c73d1ab51cec3166c3cf23b9ea51e44a0ad631b0b0329ef55aaae555420348a544e18d5760969281759b61 -94f326a32ed287545b0515be9e08149eb0a565025074796d72387cc3a237e87979776410d78339e23ef3172ca43b2544 -a785d2961a2fa5e70bffa137858a92c48fe749fee91b02599a252b0cd50d311991a08efd7fa5e96b78d07e6e66ffe746 -94af9030b5ac792dd1ce517eaadcec1482206848bea4e09e55cc7f40fd64d4c2b3e9197027c5636b70d6122c51d2235d -9722869f7d1a3992850fe7be405ec93aa17dc4d35e9e257d2e469f46d2c5a59dbd504056c85ab83d541ad8c13e8bcd54 -b13c4088b61a06e2c03ac9813a75ff1f68ffdfee9df6a8f65095179a475e29cc49119cad2ce05862c3b1ac217f3aace9 -8c64d51774753623666b10ca1b0fe63ae42f82ed6aa26b81dc1d48c86937c5772eb1402624c52a154b86031854e1fb9f -b47e4df18002b7dac3fee945bf9c0503159e1b8aafcce2138818e140753011b6d09ef1b20894e08ba3006b093559061b -93cb5970076522c5a0483693f6a35ffd4ea2aa7aaf3730c4eccd6af6d1bebfc1122fc4c67d53898ae13eb6db647be7e2 -a68873ef80986795ea5ed1a597d1cd99ed978ec25e0abb57fdcc96e89ef0f50aeb779ff46e3dce21dc83ada3157a8498 -8cab67f50949cc8eee6710e27358aea373aae3c92849f8f0b5531c080a6300cdf2c2094fe6fecfef6148de0d28446919 -993e932bcb616dbaa7ad18a4439e0565211d31071ef1b85a0627db74a05d978c60d507695eaeea5c7bd9868a21d06923 -acdadff26e3132d9478a818ef770e9fa0d2b56c6f5f48bd3bd674436ccce9bdfc34db884a73a30c04c5f5e9764cb2218 -a0d3e64c9c71f84c0eef9d7a9cb4fa184224b969db5514d678e93e00f98b41595588ca802643ea225512a4a272f5f534 -91c9140c9e1ba6e330cb08f6b2ce4809cd0d5a0f0516f70032bf30e912b0ed684d07b413b326ab531ee7e5b4668c799b -87bc2ee7a0c21ba8334cd098e35cb703f9af57f35e091b8151b9b63c3a5b0f89bd7701dbd44f644ea475901fa6d9ef08 -9325ccbf64bf5d71b303e31ee85d486298f9802c5e55b2c3d75427097bf8f60fa2ab4fcaffa9b60bf922c3e24fbd4b19 -95d0506e898318f3dc8d28d16dfd9f0038b54798838b3c9be2a2ae3c2bf204eb496166353fc042220b0bd4f6673b9285 -811de529416331fe9c416726d45df9434c29dcd7e949045eb15740f47e97dde8f31489242200e19922cac2a8b7c6fd1f -ade632d04a4c8bbab6ca7df370b2213cb9225023e7973f0e29f4f5e52e8aeaabc65171306bbdd12a67b195dfbb96d48f -88b7f029e079b6ae956042c0ea75d53088c5d0efd750dd018adaeacf46be21bf990897c58578c491f41afd3978d08073 -91f477802de507ffd2be3f4319903119225b277ad24f74eb50f28b66c14d32fae53c7edb8c7590704741af7f7f3e3654 -809838b32bb4f4d0237e98108320d4b079ee16ed80c567e7548bd37e4d7915b1192880f4812ac0e00476d246aec1dbc8 -84183b5fc4a7997a8ae5afedb4d21dce69c480d5966b5cbdafd6dd10d29a9a6377f3b90ce44da0eb8b176ac3af0253bb -8508abbf6d3739a16b9165caf0f95afb3b3ac1b8c38d6d374cf0c91296e2c1809a99772492b539cda184510bce8a0271 -8722054e59bab2062e6419a6e45fc803af77fde912ef2cd23055ad0484963de65a816a2debe1693d93c18218d2b8e81a -8e895f80e485a7c4f56827bf53d34b956281cdc74856c21eb3b51f6288c01cc3d08565a11cc6f3e2604775885490e8c5 -afc92714771b7aa6e60f3aee12efd9c2595e9659797452f0c1e99519f67c8bc3ac567119c1ddfe82a3e961ee9defea9a -818ff0fd9cefd32db87b259e5fa32967201016fc02ef44116cdca3c63ce5e637756f60477a408709928444a8ad69c471 -8251e29af4c61ae806fc5d032347fb332a94d472038149225298389495139ce5678fae739d02dfe53a231598a992e728 -a0ea39574b26643f6f1f48f99f276a8a64b5481989cfb2936f9432a3f8ef5075abfe5c067dc5512143ce8bf933984097 -af67a73911b372bf04e57e21f289fc6c3dfac366c6a01409b6e76fea4769bdb07a6940e52e8d7d3078f235c6d2f632c6 -b5291484ef336024dd2b9b4cf4d3a6b751133a40656d0a0825bcc6d41c21b1c79cb50b0e8f4693f90c29c8f4358641f9 -8bc0d9754d70f2cb9c63f991902165a87c6535a763d5eece43143b5064ae0bcdce7c7a8f398f2c1c29167b2d5a3e6867 -8d7faff53579ec8f6c92f661c399614cc35276971752ce0623270f88be937c414eddcb0997e14724a783905a026c8883 -9310b5f6e675fdf60796f814dbaa5a6e7e9029a61c395761e330d9348a7efab992e4e115c8be3a43d08e90d21290c892 -b5eb4f3eb646038ad2a020f0a42202532d4932e766da82b2c1002bf9c9c2e5336b54c8c0ffcc0e02d19dde2e6a35b6cc -91dabfd30a66710f1f37a891136c9be1e23af4abf8cb751f512a40c022a35f8e0a4fb05b17ec36d4208de02d56f0d53a -b3ded14e82d62ac7a5a036122a62f00ff8308498f3feae57d861babaff5a6628d43f0a0c5fc903f10936bcf4e2758ceb -a88e8348fed2b26acca6784d19ef27c75963450d99651d11a950ea81d4b93acd2c43e0ecce100eaf7e78508263d5baf3 -b1f5bbf7c4756877b87bb42163ac570e08c6667c4528bf68b5976680e19beeff7c5effd17009b0718797077e2955457a -ad2e7b516243f915d4d1415326e98b1a7390ae88897d0b03b66c2d9bd8c3fba283d7e8fe44ed3333296a736454cef6d8 -8f82eae096d5b11f995de6724a9af895f5e1c58d593845ad16ce8fcae8507e0d8e2b2348a0f50a1f66a17fd6fac51a5c -890e4404d0657c6c1ee14e1aac132ecf7a568bb3e04137b85ac0f84f1d333bd94993e8750f88eee033a33fb00f85dcc7 -82ac7d3385e035115f1d39a99fc73e5919de44f5e6424579776d118d711c8120b8e5916372c6f27bed4cc64cac170b6c -85ee16d8901c272cfbbe966e724b7a891c1bd5e68efd5d863043ad8520fc409080af61fd726adc680b3f1186fe0ac8b8 -86dc564c9b545567483b43a38f24c41c6551a49cabeebb58ce86404662a12dbfafd0778d30d26e1c93ce222e547e3898 -a29f5b4522db26d88f5f95f18d459f8feefab02e380c2edb65aa0617a82a3c1a89474727a951cef5f15050bcf7b380fb -a1ce039c8f6cac53352899edb0e3a72c76da143564ad1a44858bd7ee88552e2fe6858d1593bbd74aeee5a6f8034b9b9d -97f10d77983f088286bd7ef3e7fdd8fa275a56bec19919adf33cf939a90c8f2967d2b1b6fc51195cb45ad561202a3ed7 -a25e2772e8c911aaf8712bdac1dd40ee061c84d3d224c466cfaae8e5c99604053f940cde259bd1c3b8b69595781dbfec -b31bb95a0388595149409c48781174c340960d59032ab2b47689911d03c68f77a2273576fbe0c2bf4553e330656058c7 -b8b2e9287ad803fb185a13f0d7456b397d4e3c8ad5078f57f49e8beb2e85f661356a3392dbd7bcf6a900baa5582b86a1 -a3d0893923455eb6e96cc414341cac33d2dbc88fba821ac672708cce131761d85a0e08286663a32828244febfcae6451 -82310cb42f647d99a136014a9f881eb0b9791efd2e01fc1841907ad3fc8a9654d3d1dab6689c3607214b4dc2aca01cee -874022d99c16f60c22de1b094532a0bc6d4de700ad01a31798fac1d5088b9a42ad02bef8a7339af7ed9c0d4f16b186ee -94981369e120265aed40910eebc37eded481e90f4596b8d57c3bec790ab7f929784bd33ddd05b7870aad6c02e869603b -a4f1f50e1e2a73f07095e0dd31cb45154f24968dae967e38962341c1241bcd473102fff1ff668b20c6547e9732d11701 -ae2328f3b0ad79fcda807e69a1b5278145225083f150f67511dafc97e079f860c3392675f1752ae7e864c056e592205b -875d8c971e593ca79552c43d55c8c73b17cd20c81ff2c2fed1eb19b1b91e4a3a83d32df150dbfd5db1092d0aebde1e1f -add2e80aa46aae95da73a11f130f4bda339db028e24c9b11e5316e75ba5e63bc991d2a1da172c7c8e8fee038baae3433 -b46dbe1cb3424002aa7de51e82f600852248e251465c440695d52538d3f36828ff46c90ed77fc1d11534fe3c487df8ef -a5e5045d28b4e83d0055863c30c056628c58d4657e6176fd0536f5933f723d60e851bb726d5bf3c546b8ce4ac4a57ef8 -91fec01e86dd1537e498fff7536ea3ca012058b145f29d9ada49370cd7b7193ac380e116989515df1b94b74a55c45df3 -a7428176d6918cd916a310bdc75483c72de660df48cac4e6e7478eef03205f1827ea55afc0df5d5fa7567d14bbea7fc9 -851d89bef45d9761fe5fdb62972209335193610015e16a675149519f9911373bac0919add226ef118d9f3669cfdf4734 -b74acf5c149d0042021cb2422ea022be4c4f72a77855f42393e71ffd12ebb3eec16bdf16f812159b67b79a9706e7156d -99f35dce64ec99aa595e7894b55ce7b5a435851b396e79036ffb249c28206087db4c85379df666c4d95857db02e21ff9 -b6b9a384f70db9e298415b8ab394ee625dafff04be2886476e59df8d052ca832d11ac68a9b93fba7ab055b7bc36948a4 -898ee4aefa923ffec9e79f2219c7389663eb11eb5b49014e04ed4a336399f6ea1691051d86991f4c46ca65bcd4fdf359 -b0f948217b0d65df7599a0ba4654a5e43c84db477936276e6f11c8981efc6eaf14c90d3650107ed4c09af4cc8ec11137 -aa6286e27ac54f73e63dbf6f41865dd94d24bc0cf732262fcaff67319d162bb43af909f6f8ee27b1971939cfbba08141 -8bca7cdf730cf56c7b2c8a2c4879d61361a6e1dba5a3681a1a16c17a56e168ace0e99cf0d15826a1f5e67e6b8a8a049a -a746d876e8b1ce225fcafca603b099b36504846961526589af977a88c60d31ba2cc56e66a3dec8a77b3f3531bf7524c9 -a11e2e1927e6704cdb8874c75e4f1842cef84d7d43d7a38e339e61dc8ba90e61bbb20dd3c12e0b11d2471d58eed245be -a36395e22bc1d1ba8b0459a235203177737397da5643ce54ded3459d0869ff6d8d89f50c73cb62394bf66a959cde9b90 -8b49f12ba2fdf9aca7e5f81d45c07d47f9302a2655610e7634d1e4bd16048381a45ef2c95a8dd5b0715e4b7cf42273af -91cffa2a17e64eb7f76bccbe4e87280ee1dd244e04a3c9eac12e15d2d04845d876eb24fe2ec6d6d266cce9efb281077f -a6b8afabf65f2dee01788114e33a2f3ce25376fb47a50b74da7c3c25ff1fdc8aa9f41307534abbf48acb6f7466068f69 -8d13db896ccfea403bd6441191995c1a65365cab7d0b97fbe9526da3f45a877bd1f4ef2edef160e8a56838cd1586330e -98c717de9e01bef8842c162a5e757fe8552d53269c84862f4d451e7c656ae6f2ae473767b04290b134773f63be6fdb9d -8c2036ace1920bd13cf018e82848c49eb511fad65fd0ff51f4e4b50cf3bfc294afb63cba682c16f52fb595a98fa84970 -a3520fdff05dbad9e12551b0896922e375f9e5589368bcb2cc303bde252743b74460cb5caf99629325d3620f13adc796 -8d4f83a5bfec05caf5910e0ce538ee9816ee18d0bd44c1d0da2a87715a23cd2733ad4d47552c6dc0eb397687d611dd19 -a7b39a0a6a02823452d376533f39d35029867b3c9a6ad6bca181f18c54132d675613a700f9db2440fb1b4fa13c8bf18a -80bcb114b2544b80f404a200fc36860ed5e1ad31fe551acd4661d09730c452831751baa9b19d7d311600d267086a70bc -90dcce03c6f88fc2b08f2b42771eedde90cc5330fe0336e46c1a7d1b5a6c1641e5fcc4e7b3d5db00bd8afca9ec66ed81 -aec15f40805065c98e2965b1ae12a6c9020cfdb094c2d0549acfc7ea2401a5fb48d3ea7d41133cf37c4e096e7ff53eb9 -80e129b735dba49fa627a615d6c273119acec8e219b2f2c4373a332b5f98d66cbbdd688dfbe72a8f8bfefaccc02c50c1 -a9b596da3bdfe23e6799ece5f7975bf7a1979a75f4f546deeaf8b34dfe3e0d623217cb4cf4ccd504cfa3625b88cd53f1 -abcbbb70b16f6e517c0ab4363ab76b46e4ff58576b5f8340e5c0e8cc0e02621b6e23d742d73b015822a238b17cfd7665 -a046937cc6ea6a2e1adae543353a9fe929c1ae4ad655be1cc051378482cf88b041e28b1e9a577e6ccff2d3570f55e200 -831279437282f315e65a60184ef158f0a3dddc15a648dc552bdc88b3e6fe8288d3cfe9f0031846d81350f5e7874b4b33 -993d7916fa213c6d66e7c4cafafc1eaec9a2a86981f91c31eb8a69c5df076c789cbf498a24c84e0ee77af95b42145026 -823907a3b6719f8d49b3a4b7c181bd9bb29fcf842d7c70660c4f351852a1e197ca46cf5e879b47fa55f616fa2b87ce5e -8d228244e26132b234930ee14c75d88df0943cdb9c276a8faf167d259b7efc1beec2a87c112a6c608ad1600a239e9aae -ab6e55766e5bfb0cf0764ed909a8473ab5047d3388b4f46faeba2d1425c4754c55c6daf6ad4751e634c618b53e549529 -ab0cab6860e55a84c5ad2948a7e0989e2b4b1fd637605634b118361497332df32d9549cb854b2327ca54f2bcb85eed8f -b086b349ae03ef34f4b25a57bcaa5d1b29bd94f9ebf87e22be475adfe475c51a1230c1ebe13506cb72c4186192451658 -8a0b49d8a254ca6d91500f449cbbfbb69bb516c6948ac06808c65595e46773e346f97a5ce0ef7e5a5e0de278af22709c -ac49de11edaaf04302c73c578cc0824bdd165c0d6321be1c421c1950e68e4f3589aa3995448c9699e93c6ebae8803e27 -884f02d841cb5d8f4c60d1402469216b114ab4e93550b5bc1431756e365c4f870a9853449285384a6fa49e12ce6dc654 -b75f3a28fa2cc8d36b49130cb7448a23d73a7311d0185ba803ad55c8219741d451c110f48b786e96c728bc525903a54f -80ae04dbd41f4a35e33f9de413b6ad518af0919e5a30cb0fa1b061b260420780bb674f828d37fd3b52b5a31673cbd803 -b9a8011eb5fcea766907029bf743b45262db3e49d24f84503687e838651ed11cb64c66281e20a0ae9f6aa51acc552263 -90bfdd75e2dc9cf013e22a5d55d2d2b8a754c96103a17524488e01206e67f8b6d52b1be8c4e3d5307d4fe06d0e51f54c -b4af353a19b06203a815ec43e79a88578cc678c46f5a954b85bc5c53b84059dddba731f3d463c23bfd5273885c7c56a4 -aa125e96d4553b64f7140e5453ff5d2330318b69d74d37d283e84c26ad672fa00e3f71e530eb7e28be1e94afb9c4612e -a18e060aee3d49cde2389b10888696436bb7949a79ca7d728be6456a356ea5541b55492b2138da90108bd1ce0e6f5524 -93e55f92bdbccc2de655d14b1526836ea2e52dba65eb3f87823dd458a4cb5079bf22ce6ef625cb6d6bfdd0995ab9a874 -89f5a683526b90c1c3ceebbb8dc824b21cff851ce3531b164f6626e326d98b27d3e1d50982e507d84a99b1e04e86a915 -83d1c38800361633a3f742b1cb2bfc528129496e80232611682ddbe403e92c2ac5373aea0bca93ecb5128b0b2b7a719e -8ecba560ac94905e19ce8d9c7af217bf0a145d8c8bd38e2db82f5e94cc3f2f26f55819176376b51f154b4aab22056059 -a7e2a4a002b60291924850642e703232994acb4cfb90f07c94d1e0ecd2257bb583443283c20fc6017c37e6bfe85b7366 -93ed7316fa50b528f1636fc6507683a672f4f4403e55e94663f91221cc198199595bd02eef43d609f451acc9d9b36a24 -a1220a8ebc5c50ceed76a74bc3b7e0aa77f6884c71b64b67c4310ac29ce5526cb8992d6abc13ef6c8413ce62486a6795 -b2f6eac5c869ad7f4a25161d3347093e2f70e66cd925032747e901189355022fab3038bca4d610d2f68feb7e719c110b -b703fa11a4d511ca01c7462979a94acb40b5d933759199af42670eb48f83df202fa0c943f6ab3b4e1cc54673ea3aab1e -b5422912afbfcb901f84791b04f1ddb3c3fbdc76d961ee2a00c5c320e06d3cc5b5909c3bb805df66c5f10c47a292b13d -ad0934368da823302e1ac08e3ede74b05dfdbfffca203e97ffb0282c226814b65c142e6e15ec1e754518f221f01b30f7 -a1dd302a02e37df15bf2f1147efe0e3c06933a5a767d2d030e1132f5c3ce6b98e216b6145eb39e1e2f74e76a83165b8d -a346aab07564432f802ae44738049a36f7ca4056df2d8f110dbe7fef4a3e047684dea609b2d03dc6bf917c9c2a47608f -b96c5f682a5f5d02123568e50f5d0d186e4b2c4c9b956ec7aabac1b3e4a766d78d19bd111adb5176b898e916e49be2aa -8a96676d56876fc85538db2e806e1cba20fd01aeb9fa3cb43ca6ca94a2c102639f65660db330e5d74a029bb72d6a0b39 -ab0048336bd5c3def1a4064eadd49e66480c1f2abb4df46e03afbd8a3342c2c9d74ee35d79f08f4768c1646681440984 -888427bdf76caec90814c57ee1c3210a97d107dd88f7256f14f883ad0f392334b82be11e36dd8bfec2b37935177c7831 -b622b282becf0094a1916fa658429a5292ba30fb48a4c8066ce1ddcefb71037948262a01c95bab6929ed3a76ba5db9fe -b5b9e005c1f456b6a368a3097634fb455723abe95433a186e8278dceb79d4ca2fbe21f8002e80027b3c531e5bf494629 -a3c6707117a1e48697ed41062897f55d8119403eea6c2ee88f60180f6526f45172664bfee96bf61d6ec0b7fbae6aa058 -b02a9567386a4fbbdb772d8a27057b0be210447348efe6feb935ceec81f361ed2c0c211e54787dc617cdffed6b4a6652 -a9b8364e40ef15c3b5902e5534998997b8493064fa2bea99600def58279bb0f64574c09ba11e9f6f669a8354dd79dc85 -9998a2e553a9aa9a206518fae2bc8b90329ee59ab23005b10972712389f2ec0ee746033c733092ffe43d73d33abbb8ef -843a4b34d9039bf79df96d79f2d15e8d755affb4d83d61872daf540b68c0a3888cf8fc00d5b8b247b38524bcb3b5a856 -84f7128920c1b0bb40eee95701d30e6fc3a83b7bb3709f16d97e72acbb6057004ee7ac8e8f575936ca9dcb7866ab45f7 -918d3e2222e10e05edb34728162a899ad5ada0aaa491aeb7c81572a9c0d506e31d5390e1803a91ff3bd8e2bb15d47f31 -9442d18e2489613a7d47bb1cb803c8d6f3259d088cd079460976d87f7905ee07dea8f371b2537f6e1d792d36d7e42723 -b491976970fe091995b2ed86d629126523ccf3e9daf8145302faca71b5a71a5da92e0e05b62d7139d3efac5c4e367584 -aa628006235dc77c14cef4c04a308d66b07ac92d377df3de1a2e6ecfe3144f2219ad6d7795e671e1cb37a3641910b940 -99d386adaea5d4981d7306feecac9a555b74ffdc218c907c5aa7ac04abaead0ec2a8237300d42a3fbc464673e417ceed -8f78e8b1556f9d739648ea3cab9606f8328b52877fe72f9305545a73b74d49884044ba9c1f1c6db7d9b7c7b7c661caba -8fb357ae49932d0babdf74fc7aa7464a65d3b6a2b3acf4f550b99601d3c0215900cfd67f2b6651ef94cfc323bac79fae -9906f2fa25c0290775aa001fb6198113d53804262454ae8b83ef371b5271bde189c0460a645829cb6c59f9ee3a55ce4d -8f4379b3ebb50e052325b27655ca6a82e6f00b87bf0d2b680d205dd2c7afdc9ff32a9047ae71a1cdf0d0ce6b9474d878 -a85534e88c2bd43c043792eaa75e50914b21741a566635e0e107ae857aed0412035f7576cf04488ade16fd3f35fdbb87 -b4ce93199966d3c23251ca7f28ec5af7efea1763d376b0385352ffb2e0a462ef95c69940950278cf0e3dafd638b7bd36 -b10cb3d0317dd570aa73129f4acf63c256816f007607c19b423fb42f65133ce21f2f517e0afb41a5378cccf893ae14d0 -a9b231c9f739f7f914e5d943ed9bff7eba9e2c333fbd7c34eb1648a362ee01a01af6e2f7c35c9fe962b11152cddf35de -99ff6a899e156732937fb81c0cced80ae13d2d44c40ba99ac183aa246103b31ec084594b1b7feb96da58f4be2dd5c0ed -8748d15d18b75ff2596f50d6a9c4ce82f61ecbcee123a6ceae0e43cab3012a29b6f83cf67b48c22f6f9d757c6caf76b2 -b88ab05e4248b7fb634cf640a4e6a945d13e331237410f7217d3d17e3e384ddd48897e7a91e4516f1b9cbd30f35f238b -8d826deaeeb84a3b2d2c04c2300ca592501f992810582d6ae993e0d52f6283a839dba66c6c72278cff5871802b71173b -b36fed027c2f05a5ef625ca00b0364b930901e9e4420975b111858d0941f60e205546474bb25d6bfa6928d37305ae95f -af2fcfc6b87967567e8b8a13a4ed914478185705724e56ce68fb2df6d1576a0cf34a61e880997a0d35dc2c3276ff7501 -ac351b919cd1fbf106feb8af2c67692bfcddc84762d18cea681cfa7470a5644839caace27efee5f38c87d3df306f4211 -8d6665fb1d4d8d1fa23bd9b8a86e043b8555663519caac214d1e3e3effbc6bee7f2bcf21e645f77de0ced279d69a8a8b -a9fc1c2061756b2a1a169c1b149f212ff7f0d2488acd1c5a0197eba793cffa593fc6d1d1b40718aa75ca3ec77eff10e1 -aff64f0fa009c7a6cf0b8d7a22ddb2c8170c3cb3eec082e60d5aadb00b0040443be8936d728d99581e33c22178c41c87 -82e0b181adc5e3b1c87ff8598447260e839d53debfae941ebea38265575546c3a74a14b4325a030833a62ff6c52d9365 -b7ad43cbb22f6f892c2a1548a41dc120ab1f4e1b8dea0cb6272dd9cb02054c542ecabc582f7e16de709d48f5166cae86 -985e0c61094281532c4afb788ecb2dfcba998e974b5d4257a22040a161883908cdd068fe80f8eb49b8953cfd11acf43a -ae46895c6d67ea6d469b6c9c07b9e5d295d9ae73b22e30da4ba2c973ba83a130d7eef39717ec9d0f36e81d56bf742671 -8600177ea1f7e7ef90514b38b219a37dedfc39cb83297e4c7a5b479817ef56479d48cf6314820960c751183f6edf8b0e -b9208ec1c1d7a1e99b59c62d3e4e61dfb706b0e940d09d3abfc3454c19749083260614d89cfd7e822596c3cdbcc6bb95 -a1e94042c796c2b48bc724352d2e9f3a22291d9a34705993357ddb6adabd76da6fc25dac200a8cb0b5bbd99ecddb7af6 -b29c3adedd0bcad8a930625bc4dfdc3552a9afd5ca6dd9c0d758f978068c7982b50b711aa0eb5b97f2b84ee784637835 -af0632a238bb1f413c7ea8e9b4c3d68f2827bd2e38cd56024391fba6446ac5d19a780d0cfd4a78fe497d537b766a591a -aaf6e7f7d54f8ef5e2e45dd59774ecbeecf8683aa70483b2a75be6a6071b5981bbaf1627512a65d212817acdfab2e428 -8c751496065da2e927cf492aa5ca9013b24f861d5e6c24b30bbf52ec5aaf1905f40f9a28175faef283dd4ed4f2182a09 -8952377d8e80a85cf67d6b45499f3bad5fd452ea7bcd99efc1b066c4720d8e5bff1214cea90fd1f972a7f0baac3d29be -a1946ee543d1a6e21f380453be4d446e4130950c5fc3d075794eb8260f6f52d0a795c1ff91d028a648dc1ce7d9ab6b47 -89f3fefe37af31e0c17533d2ca1ce0884cc1dc97c15cbfab9c331b8debd94781c9396abef4bb2f163d09277a08d6adf0 -a2753f1e6e1a154fb117100a5bd9052137add85961f8158830ac20541ab12227d83887d10acf7fd36dcaf7c2596d8d23 -814955b4198933ee11c3883863b06ff98c7eceb21fc3e09df5f916107827ccf3323141983e74b025f46ae00284c9513b -8cc5c6bb429073bfef47cae7b3bfccb0ffa076514d91a1862c6bda4d581e0df87db53cc6c130bf8a7826304960f5a34e -909f22c1f1cdc87f7be7439c831a73484a49acbf8f23d47087d7cf867c64ef61da3bde85dc57d705682b4c3fc710d36e -8048fee7f276fcd504aed91284f28e73693615e0eb3858fa44bcf79d7285a9001c373b3ef71d9a3054817ba293ebe28c -94400e5cf5d2700ca608c5fe35ce14623f71cc24959f2bc27ca3684092850f76b67fb1f07ca9e5b2ca3062cf8ad17bd4 -81c2ae7d4d1b17f8b6de6a0430acc0d58260993980fe48dc2129c4948269cdc74f9dbfbf9c26b19360823fd913083d48 -8c41fe765128e63f6889d6a979f6a4342300327c8b245a8cfe3ecfbcac1e09c3da30e2a1045b24b78efc6d6d50c8c6ac -a5dd4ae51ae48c8be4b218c312ade226cffce671cf121cb77810f6c0990768d6dd767badecb5c69921d5574d5e8433d3 -b7642e325f4ba97ae2a39c1c9d97b35aafd49d53dba36aed3f3cb0ca816480b3394079f46a48252d46596559c90f4d58 -ae87375b40f35519e7bd4b1b2f73cd0b329b0c2cb9d616629342a71c6c304338445eda069b78ea0fbe44087f3de91e09 -b08918cb6f736855e11d3daca1ddfbdd61c9589b203b5493143227bf48e2c77c2e8c94b0d1aa2fab2226e0eae83f2681 -ac36b84a4ac2ebd4d6591923a449c564e3be8a664c46092c09e875c2998eba16b5d32bfd0882fd3851762868e669f0b1 -a44800a3bb192066fa17a3f29029a23697240467053b5aa49b9839fb9b9b8b12bcdcbfc557f024b61f4f51a9aacdefcb -9064c688fec23441a274cdf2075e5a449caf5c7363cc5e8a5dc9747183d2e00a0c69f2e6b3f6a7057079c46014c93b3b -aa367b021469af9f5b764a79bb3afbe2d87fe1e51862221672d1a66f954b165778b7c27a705e0f93841fab4c8468344d -a1a8bfc593d4ab71f91640bc824de5c1380ab2591cfdafcbc78a14b32de3c0e15f9d1b461d85c504baa3d4232c16bb53 -97df48da1799430f528184d30b6baa90c2a2f88f34cdfb342d715339c5ebd6d019aa693cea7c4993daafc9849063a3aa -abd923831fbb427e06e0dd335253178a9e5791395c84d0ab1433c07c53c1209161097e9582fb8736f8a60bde62d8693e -84cd1a43f1a438b43dc60ffc775f646937c4f6871438163905a3cebf1115f814ccd38a6ccb134130bff226306e412f32 -91426065996b0743c5f689eb3ca68a9f7b9e4d01f6c5a2652b57fa9a03d8dc7cd4bdbdab0ca5a891fee1e97a7f00cf02 -a4bee50249db3df7fd75162b28f04e57c678ba142ce4d3def2bc17bcb29e4670284a45f218dad3969af466c62a903757 -83141ebcc94d4681404e8b67a12a46374fded6df92b506aff3490d875919631408b369823a08b271d006d5b93136f317 -a0ea1c8883d58d5a784da3d8c8a880061adea796d7505c1f903d07c287c5467f71e4563fc0faafbc15b5a5538b0a7559 -89d9d480574f201a87269d26fb114278ed2c446328df431dc3556e3500e80e4cd01fcac196a2459d8646361ebda840df -8bf302978973632dd464bec819bdb91304712a3ec859be071e662040620422c6e75eba6f864f764cffa2799272efec39 -922f666bc0fd58b6d7d815c0ae4f66d193d32fc8382c631037f59eeaeae9a8ca6c72d08e72944cf9e800b8d639094e77 -81ad8714f491cdff7fe4399f2eb20e32650cff2999dd45b9b3d996d54a4aba24cc6c451212e78c9e5550368a1a38fb3f -b58fcf4659d73edb73175bd9139d18254e94c3e32031b5d4b026f2ed37aa19dca17ec2eb54c14340231615277a9d347e -b365ac9c2bfe409b710928c646ea2fb15b28557e0f089d39878e365589b9d1c34baf5566d20bb28b33bb60fa133f6eff -8fcae1d75b53ab470be805f39630d204853ca1629a14158bac2f52632277d77458dec204ff84b7b2d77e641c2045be65 -a03efa6bebe84f4f958a56e2d76b5ba4f95dd9ed7eb479edc7cc5e646c8d4792e5b0dfc66cc86aa4b4afe2f7a4850760 -af1c823930a3638975fb0cc5c59651771b2719119c3cd08404fbd4ce77a74d708cefbe3c56ea08c48f5f10e6907f338f -8260c8299b17898032c761c325ac9cabb4c5b7e735de81eacf244f647a45fb385012f4f8df743128888c29aefcaaad16 -ab2f37a573c82e96a8d46198691cd694dfa860615625f477e41f91b879bc58a745784fccd8ffa13065834ffd150d881d -986c746c9b4249352d8e5c629e8d7d05e716b3c7aab5e529ca969dd1e984a14b5be41528baef4c85d2369a42d7209216 -b25e32da1a8adddf2a6080725818b75bc67240728ad1853d90738485d8924ea1e202df0a3034a60ffae6f965ec55cf63 -a266e627afcebcefea6b6b44cbc50f5c508f7187e87d047b0450871c2a030042c9e376f3ede0afcf9d1952f089582f71 -86c3bbca4c0300606071c0a80dbdec21ce1dd4d8d4309648151c420854032dff1241a1677d1cd5de4e4de4385efda986 -b9a21a1fe2d1f3273a8e4a9185abf2ff86448cc98bfa435e3d68306a2b8b4a6a3ea33a155be3cb62a2170a86f77679a5 -b117b1ea381adce87d8b342cba3a15d492ff2d644afa28f22424cb9cbc820d4f7693dfc1a4d1b3697046c300e1c9b4c8 -9004c425a2e68870d6c69b658c344e3aa3a86a8914ee08d72b2f95c2e2d8a4c7bb0c6e7e271460c0e637cec11117bf8e -86a18aa4783b9ebd9131580c8b17994825f27f4ac427b0929a1e0236907732a1c8139e98112c605488ee95f48bbefbfc -84042243b955286482ab6f0b5df4c2d73571ada00716d2f737ca05a0d2e88c6349e8ee9e67934cfee4a1775dbf7f4800 -92c2153a4733a62e4e1d5b60369f3c26777c7d01cd3c8679212660d572bd3bac9b8a8a64e1f10f7dbf5eaa7579c4e423 -918454b6bb8e44a2afa144695ba8d48ae08d0cdfef4ad078f67709eddf3bb31191e8b006f04e82ea45a54715ef4d5817 -acf0b54f6bf34cf6ed6c2b39cf43194a40d68de6bcf1e4b82c34c15a1343e9ac3737885e1a30b78d01fa3a5125463db8 -a7d60dbe4b6a7b054f7afe9ee5cbbfeca0d05dc619e6041fa2296b549322529faddb8a11e949562309aecefb842ac380 -91ffb53e6d7e5f11159eaf13e783d6dbdfdb1698ed1e6dbf3413c6ea23492bbb9e0932230a9e2caac8fe899a17682795 -b6e8d7be5076ee3565d5765a710c5ecf17921dd3cf555c375d01e958a365ae087d4a88da492a5fb81838b7b92bf01143 -a8c6b763de2d4b2ed42102ef64eccfef31e2fb2a8a2776241c82912fa50fc9f77f175b6d109a97ede331307c016a4b1a -99839f86cb700c297c58bc33e28d46b92931961548deac29ba8df91d3e11721b10ea956c8e16984f9e4acf1298a79b37 -8c2e2c338f25ea5c25756b7131cde0d9a2b35abf5d90781180a00fe4b8e64e62590dc63fe10a57fba3a31c76d784eb01 -9687d7df2f41319ca5469d91978fed0565a5f11f829ebadaa83db92b221755f76c6eacd7700735e75c91e257087512e3 -8795fdfb7ff8439c58b9bf58ed53873d2780d3939b902b9ddaaa4c99447224ced9206c3039a23c2c44bcc461e2bb637f -a803697b744d2d087f4e2307218d48fa88620cf25529db9ce71e2e3bbcc65bac5e8bb9be04777ef7bfb5ed1a5b8e6170 -80f3d3efbbb9346ddd413f0a8e36b269eb5d7ff6809d5525ff9a47c4bcab2c01b70018b117f6fe05253775612ff70c6b -9050e0e45bcc83930d4c505af35e5e4d7ca01cd8681cba92eb55821aececcebe32bb692ebe1a4daac4e7472975671067 -8d206812aac42742dbaf233e0c080b3d1b30943b54b60283515da005de05ea5caa90f91fedcfcba72e922f64d7040189 -a2d44faaeb2eff7915c83f32b13ca6f31a6847b1c1ce114ea240bac3595eded89f09b2313b7915ad882292e2b586d5b4 -961776c8576030c39f214ea6e0a3e8b3d32f023d2600958c098c95c8a4e374deeb2b9dc522adfbd6bda5949bdc09e2a2 -993fa7d8447407af0fbcd9e6d77f815fa5233ab00674efbcf74a1f51c37481445ae291cc7b76db7c178f9cb0e570e0fc -abd5b1c78e05f9d7c8cc99bdaef8b0b6a57f2daf0f02bf492bec48ea4a27a8f1e38b5854da96efff11973326ff980f92 -8f15af4764bc275e6ccb892b3a4362cacb4e175b1526a9a99944e692fe6ccb1b4fc19abf312bb2a089cb1f344d91a779 -a09b27ccd71855512aba1d0c30a79ffbe7f6707a55978f3ced50e674b511a79a446dbc6d7946add421ce111135a460af -94b2f98ce86a9271fbd4153e1fc37de48421fe3490fb3840c00f2d5a4d0ba8810c6a32880b002f6374b59e0a7952518b -8650ac644f93bbcb88a6a0f49fee2663297fd4bc6fd47b6a89b9d8038d32370438ab3a4775ec9b58cb10aea8a95ef7b6 -95e5c2f2e84eed88c6980bbba5a1c0bb375d5a628bff006f7516d45bb7d723da676add4fdd45956f312e7bab0f052644 -b3278a3fa377ac93af7cfc9453f8cb594aae04269bbc99d2e0e45472ff4b6a2f97a26c4c57bf675b9d86f5e77a5d55d1 -b4bcbe6eb666a206e2ea2f877912c1d3b5bdbd08a989fc4490eb06013e1a69ad1ba08bcdac048bf29192312be399077b -a76d70b78c99fffcbf9bb9886eab40f1ea4f99a309710b660b64cbf86057cbcb644d243f6e341711bb7ef0fedf0435a7 -b2093c1ee945dca7ac76ad5aed08eae23af31dd5a77c903fd7b6f051f4ab84425d33a03c3d45bf2907bc93c02d1f3ad8 -904b1f7534e053a265b22d20be859912b9c9ccb303af9a8d6f1d8f6ccdc5c53eb4a45a1762b880d8444d9be0cd55e7f9 -8f664a965d65bc730c9ef1ec7467be984d4b8eb46bd9b0d64e38e48f94e6e55dda19aeac82cbcf4e1473440e64c4ca18 -8bcee65c4cc7a7799353d07b114c718a2aae0cd10a3f22b7eead5185d159dafd64852cb63924bf87627d176228878bce -8c78f2e3675096fef7ebaa898d2615cd50d39ca3d8f02b9bdfb07e67da648ae4be3da64838dffc5935fd72962c4b96c7 -8c40afd3701629421fec1df1aac4e849384ef2e80472c0e28d36cb1327acdf2826f99b357f3d7afdbc58a6347fc40b3c -a197813b1c65a8ea5754ef782522a57d63433ef752215ecda1e7da76b0412ee619f58d904abd2e07e0c097048b6ae1dd -a670542629e4333884ad7410f9ea3bd6f988df4a8f8a424ca74b9add2312586900cf9ae8bd50411f9146e82626b4af56 -a19875cc07ab84e569d98b8b67fb1dbbdfb59093c7b748fae008c8904a6fd931a63ca8d03ab5fea9bc8d263568125a9b -b57e7f68e4eb1bd04aafa917b1db1bdab759a02aa8a9cdb1cba34ba8852b5890f655645c9b4e15d5f19bf37e9f2ffe9f -8abe4e2a4f6462b6c64b3f10e45db2a53c2b0d3c5d5443d3f00a453e193df771eda635b098b6c8604ace3557514027af -8459e4fb378189b22b870a6ef20183deb816cefbf66eca1dc7e86d36a2e011537db893729f500dc154f14ce24633ba47 -930851df4bc7913c0d8c0f7bd3b071a83668987ed7c397d3d042fdc0d9765945a39a3bae83da9c88cb6b686ed8aeeb26 -8078c9e5cd05e1a8c932f8a1d835f61a248b6e7133fcbb3de406bf4ffc0e584f6f9f95062740ba6008d98348886cf76b -addff62bb29430983fe578e3709b0949cdc0d47a13a29bc3f50371a2cb5c822ce53e2448cfaa01bcb6e0aa850d5a380e -9433add687b5a1e12066721789b1db2edf9b6558c3bdc0f452ba33b1da67426abe326e9a34d207bfb1c491c18811bde1 -822beda3389963428cccc4a2918fa9a8a51cf0919640350293af70821967108cded5997adae86b33cb917780b097f1ca -a7a9f52bda45e4148ed56dd176df7bd672e9b5ed18888ccdb405f47920fdb0844355f8565cefb17010b38324edd8315f -b35c3a872e18e607b2555c51f9696a17fa18da1f924d503b163b4ec9fe22ed0c110925275cb6c93ce2d013e88f173d6a -adf34b002b2b26ab84fc1bf94e05bd8616a1d06664799ab149363c56a6e0c807fdc473327d25632416e952ea327fcd95 -ae4a6b9d22a4a3183fac29e2551e1124a8ce4a561a9a2afa9b23032b58d444e6155bb2b48f85c7b6d70393274e230db7 -a2ea3be4fc17e9b7ce3110284038d46a09e88a247b6971167a7878d9dcf36925d613c382b400cfa4f37a3ebea3699897 -8e5863786b641ce3140fbfe37124d7ad3925472e924f814ebfc45959aaf3f61dc554a597610b5defaecc85b59a99b50f -aefde3193d0f700d0f515ab2aaa43e2ef1d7831c4f7859f48e52693d57f97fa9e520090f3ed700e1c966f4b76048e57f -841a50f772956622798e5cd208dc7534d4e39eddee30d8ce133383d66e5f267e389254a0cdae01b770ecd0a9ca421929 -8fbc2bfd28238c7d47d4c03b1b910946c0d94274a199575e5b23242619b1de3497784e646a92aa03e3e24123ae4fcaba -926999579c8eec1cc47d7330112586bdca20b4149c8b2d066f527c8b9f609e61ce27feb69db67eea382649c6905efcf9 -b09f31f305efcc65589adf5d3690a76cf339efd67cd43a4e3ced7b839507466e4be72dd91f04e89e4bbef629d46e68c0 -b917361f6b95f759642638e0b1d2b3a29c3bdef0b94faa30de562e6078c7e2d25976159df3edbacbf43614635c2640b4 -8e7e8a1253bbda0e134d62bfe003a2669d471b47bd2b5cde0ff60d385d8e62279d54022f5ac12053b1e2d3aaa6910b4c -b69671a3c64e0a99d90b0ed108ce1912ff8ed983e4bddd75a370e9babde25ee1f5efb59ec707edddd46793207a8b1fe7 -910b2f4ebd37b7ae94108922b233d0920b4aba0bd94202c70f1314418b548d11d8e9caa91f2cd95aff51b9432d122b7f -82f645c90dfb52d195c1020346287c43a80233d3538954548604d09fbab7421241cde8593dbc4acc4986e0ea39a27dd9 -8fee895f0a140d88104ce442fed3966f58ff9d275e7373483f6b4249d64a25fb5374bbdc6bce6b5ab0270c2847066f83 -84f5bd7aab27b2509397aeb86510dd5ac0a53f2c8f73799bf720f2f87a52277f8d6b0f77f17bc80739c6a7119b7eb062 -9903ceced81099d7e146e661bcf01cbaccab5ba54366b85e2177f07e2d8621e19d9c9c3eee14b9266de6b3f9b6ea75ae -b9c16ea2a07afa32dd6c7c06df0dec39bca2067a9339e45475c98917f47e2320f6f235da353fd5e15b477de97ddc68dd -9820a9bbf8b826bec61ebf886de2c4f404c1ebdc8bab82ee1fea816d9de29127ce1852448ff717a3fe8bbfe9e92012e5 -817224d9359f5da6f2158c2c7bf9165501424f063e67ba9859a07ab72ee2ee62eb00ca6da821cfa19065c3282ca72c74 -94b95c465e6cb00da400558a3c60cfec4b79b27e602ca67cbc91aead08de4b6872d8ea096b0dc06dca4525c8992b8547 -a2b539a5bccd43fa347ba9c15f249b417997c6a38c63517ca38394976baa08e20be384a360969ff54e7e721db536b3e5 -96caf707e34f62811ee8d32ccf28d8d6ec579bc33e424d0473529af5315c456fd026aa910c1fed70c91982d51df7d3ca -8a77b73e890b644c6a142bdbac59b22d6a676f3b63ddafb52d914bb9d395b8bf5aedcbcc90429337df431ebd758a07a6 -8857830a7351025617a08bc44caec28d2fae07ebf5ffc9f01d979ce2a53839a670e61ae2783e138313929129790a51a1 -aa3e420321ed6f0aa326d28d1a10f13facec6f605b6218a6eb9cbc074801f3467bf013a456d1415a5536f12599efa3d3 -824aed0951957b00ea2f3d423e30328a3527bf6714cf9abbae84cf27e58e5c35452ba89ccc011de7c68c75d6e021d8f1 -a2e87cc06bf202e953fb1081933d8b4445527dde20e38ed1a4f440144fd8fa464a2b73e068b140562e9045e0f4bd3144 -ae3b8f06ad97d7ae3a5e5ca839efff3e4824dc238c0c03fc1a8d2fc8aa546cdfd165b784a31bb4dec7c77e9305b99a4b -b30c3e12395b1fb8b776f3ec9f87c70e35763a7b2ddc68f0f60a4982a84017f27c891a98561c830038deb033698ed7fc -874e507757cd1177d0dff0b0c62ce90130324442a33da3b2c8ee09dbca5d543e3ecfe707e9f1361e7c7db641c72794bb -b53012dd10b5e7460b57c092eaa06d6502720df9edbbe3e3f61a9998a272bf5baaac4a5a732ad4efe35d6fac6feca744 -85e6509d711515534d394e6cacbed6c81da710074d16ef3f4950bf2f578d662a494d835674f79c4d6315bced4defc5f0 -b6132b2a34b0905dcadc6119fd215419a7971fe545e52f48b768006944b4a9d7db1a74b149e2951ea48c083b752d0804 -989867da6415036d19b4bacc926ce6f4df7a556f50a1ba5f3c48eea9cefbb1c09da81481c8009331ee83f0859185e164 -960a6c36542876174d3fbc1505413e29f053ed87b8d38fef3af180491c7eff25200b45dd5fe5d4d8e63c7e8c9c00f4c8 -9040b59bd739d9cc2e8f6e894683429e4e876a8106238689ff4c22770ae5fdae1f32d962b30301fa0634ee163b524f35 -af3fcd0a45fe9e8fe256dc7eab242ef7f582dd832d147444483c62787ac820fafc6ca55d639a73f76bfa5e7f5462ab8f -b934c799d0736953a73d91e761767fdb78454355c4b15c680ce08accb57ccf941b13a1236980001f9e6195801cffd692 -8871e8e741157c2c326b22cf09551e78da3c1ec0fc0543136f581f1550f8bab03b0a7b80525c1e99812cdbf3a9698f96 -a8a977f51473a91d178ee8cfa45ffef8d6fd93ab1d6e428f96a3c79816d9c6a93cd70f94d4deda0125fd6816e30f3bea -a7688b3b0a4fc1dd16e8ba6dc758d3cfe1b7cf401c31739484c7fa253cce0967df1b290769bcefc9d23d3e0cb19e6218 -8ae84322662a57c6d729e6ff9d2737698cc2da2daeb1f39e506618750ed23442a6740955f299e4a15dda6db3e534d2c6 -a04a961cdccfa4b7ef83ced17ab221d6a043b2c718a0d6cc8e6f798507a31f10bf70361f70a049bc8058303fa7f96864 -b463e39732a7d9daec8a456fb58e54b30a6e160aa522a18b9a9e836488cce3342bcbb2e1deab0f5e6ec0a8796d77197d -b1434a11c6750f14018a2d3bcf94390e2948f4f187e93bb22070ca3e5393d339dc328cbfc3e48815f51929465ffe7d81 -84ff81d73f3828340623d7e3345553610aa22a5432217ef0ebd193cbf4a24234b190c65ca0873c22d10ea7b63bd1fbed -b6fe2723f0c47757932c2ddde7a4f8434f665612f7b87b4009c2635d56b6e16b200859a8ade49276de0ef27a2b6c970a -9742884ed7cd52b4a4a068a43d3faa02551a424136c85a9313f7cb58ea54c04aa83b0728fd741d1fe39621e931e88f8f -b7d2d65ea4d1ad07a5dee39e40d6c03a61264a56b1585b4d76fc5b2a68d80a93a42a0181d432528582bf08d144c2d6a9 -88c0f66bada89f8a43e5a6ead2915088173d106c76f724f4a97b0f6758aed6ae5c37c373c6b92cdd4aea8f6261f3a374 -81f9c43582cb42db3900747eb49ec94edb2284999a499d1527f03315fd330e5a509afa3bff659853570e9886aab5b28b -821f9d27d6beb416abf9aa5c79afb65a50ed276dbda6060103bc808bcd34426b82da5f23e38e88a55e172f5c294b4d40 -8ba307b9e7cb63a6c4f3851b321aebfdb6af34a5a4c3bd949ff7d96603e59b27ff4dc4970715d35f7758260ff942c9e9 -b142eb6c5f846de33227d0bda61d445a7c33c98f0a8365fe6ab4c1fabdc130849be597ef734305894a424ea715372d08 -a732730ae4512e86a741c8e4c87fee8a05ee840fec0e23b2e037d58dba8dde8d10a9bc5191d34d00598941becbbe467f -adce6f7c30fd221f6b10a0413cc76435c4bb36c2d60bca821e5c67409fe9dbb2f4c36ef85eb3d734695e4be4827e9fd3 -a74f00e0f9b23aff7b2527ce69852f8906dab9d6abe62ecd497498ab21e57542e12af9918d4fd610bb09e10b0929c510 -a593b6b0ef26448ce4eb3ab07e84238fc020b3cb10d542ff4b16d4e2be1bcde3797e45c9cf753b8dc3b0ffdb63984232 -aed3913afccf1aa1ac0eb4980eb8426d0baccebd836d44651fd72af00d09fac488a870223c42aca3ceb39752070405ae -b2c44c66a5ea7fde626548ba4cef8c8710191343d3dadfd3bb653ce715c0e03056a5303a581d47dde66e70ea5a2d2779 -8e5029b2ccf5128a12327b5103f7532db599846e422531869560ceaff392236434d87159f597937dbf4054f810c114f4 -82beed1a2c4477e5eb39fc5b0e773b30cfec77ef2b1bf17eadaf60eb35b6d0dd9d8cf06315c48d3546badb3f21cd0cca -90077bd6cc0e4be5fff08e5d07a5a158d36cebd1d1363125bc4fae0866ffe825b26f933d4ee5427ba5cd0c33c19a7b06 -a7ec0d8f079970e8e34f0ef3a53d3e0e45428ddcef9cc776ead5e542ef06f3c86981644f61c5a637e4faf001fb8c6b3e -ae6d4add6d1a6f90b22792bc9d40723ee6850c27d0b97eefafd5b7fd98e424aa97868b5287cc41b4fbd7023bca6a322c -831aa917533d077da07c01417feaa1408846363ba2b8d22c6116bb858a95801547dd88b7d7fa1d2e3f0a02bdeb2e103d -96511b860b07c8a5ed773f36d4aa9d02fb5e7882753bf56303595bcb57e37ccc60288887eb83bef08c657ec261a021a2 -921d2a3e7e9790f74068623de327443666b634c8443aba80120a45bba450df920b2374d96df1ce3fb1b06dd06f8cf6e3 -aa74451d51fe82b4581ead8e506ec6cd881010f7e7dd51fc388eb9a557db5d3c6721f81c151d08ebd9c2591689fbc13e -a972bfbcf4033d5742d08716c927c442119bdae336bf5dff914523b285ccf31953da2733759aacaa246a9af9f698342c -ad1fcd0cae0e76840194ce4150cb8a56ebed728ec9272035f52a799d480dfc85840a4d52d994a18b6edb31e79be6e8ad -a2c69fe1d36f235215432dad48d75887a44c99dfa0d78149acc74087da215a44bdb5f04e6eef88ff7eff80a5a7decc77 -a94ab2af2b6ee1bc6e0d4e689ca45380d9fbd3c5a65b9bd249d266a4d4c07bf5d5f7ef2ae6000623aee64027892bf8fe -881ec1fc514e926cdc66480ac59e139148ff8a2a7895a49f0dff45910c90cdda97b66441a25f357d6dd2471cddd99bb3 -884e6d3b894a914c8cef946a76d5a0c8351843b2bffa2d1e56c6b5b99c84104381dd1320c451d551c0b966f4086e60f9 -817c6c10ce2677b9fc5223500322e2b880583254d0bb0d247d728f8716f5e05c9ff39f135854342a1afecd9fbdcf7c46 -aaf4a9cb686a14619aa1fc1ac285dd3843ac3dd99f2b2331c711ec87b03491c02f49101046f3c5c538dc9f8dba2a0ac2 -97ecea5ce53ca720b5d845227ae61d70269a2f53540089305c86af35f0898bfd57356e74a8a5e083fa6e1ea70080bd31 -a22d811e1a20a75feac0157c418a4bfe745ccb5d29466ffa854dca03e395b6c3504a734341746b2846d76583a780b32e -940cbaa0d2b2db94ae96b6b9cf2deefbfd059e3e5745de9aec4a25f0991b9721e5cd37ef71c631575d1a0c280b01cd5b -ae33cb4951191258a11044682de861bf8d92d90ce751b354932dd9f3913f542b6a0f8a4dc228b3cd9244ac32c4582832 -a580df5e58c4274fe0f52ac2da1837e32f5c9db92be16c170187db4c358f43e5cfdda7c5911dcc79d77a5764e32325f5 -81798178cb9d8affa424f8d3be67576ba94d108a28ccc01d330c51d5a63ca45bb8ca63a2f569b5c5fe1303cecd2d777f -89975b91b94c25c9c3660e4af4047a8bacf964783010820dbc91ff8281509379cb3b24c25080d5a01174dd9a049118d5 -a7327fcb3710ed3273b048650bde40a32732ef40a7e58cf7f2f400979c177944c8bc54117ba6c80d5d4260801dddab79 -92b475dc8cb5be4b90c482f122a51bcb3b6c70593817e7e2459c28ea54a7845c50272af38119406eaadb9bcb993368d0 -9645173e9ecefc4f2eae8363504f7c0b81d85f8949a9f8a6c01f2d49e0a0764f4eacecf3e94016dd407fc14494fce9f9 -9215fd8983d7de6ae94d35e6698226fc1454977ae58d42d294be9aad13ac821562ad37d5e7ee5cdfe6e87031d45cd197 -810360a1c9b88a9e36f520ab5a1eb8bed93f52deefbe1312a69225c0a08edb10f87cc43b794aced9c74220cefcc57e7d -ad7e810efd61ed4684aeda9ed8bb02fb9ae4b4b63fda8217d37012b94ff1b91c0087043bfa4e376f961fff030c729f3b -8b07c95c6a06db8738d10bb03ec11b89375c08e77f0cab7e672ce70b2685667ca19c7e1c8b092821d31108ea18dfd4c7 -968825d025ded899ff7c57245250535c732836f7565eab1ae23ee7e513201d413c16e1ba3f5166e7ac6cf74de8ceef4f -908243370c5788200703ade8164943ad5f8c458219186432e74dbc9904a701ea307fd9b94976c866e6c58595fd891c4b -959969d16680bc535cdc6339e6186355d0d6c0d53d7bbfb411641b9bf4b770fd5f575beef5deec5c4fa4d192d455c350 -ad177f4f826a961adeac76da40e2d930748effff731756c797eddc4e5aa23c91f070fb69b19221748130b0961e68a6bb -82f8462bcc25448ef7e0739425378e9bb8a05e283ce54aae9dbebaf7a3469f57833c9171672ad43a79778366c72a5e37 -a28fb275b1845706c2814d9638573e9bc32ff552ebaed761fe96fdbce70395891ca41c400ae438369264e31a2713b15f -8a9c613996b5e51dadb587a787253d6081ea446bf5c71096980bf6bd3c4b69905062a8e8a3792de2d2ece3b177a71089 -8d5aefef9f60cb27c1db2c649221204dda48bb9bf8bf48f965741da051340e8e4cab88b9d15c69f3f84f4c854709f48a -93ebf2ca6ad85ab6deace6de1a458706285b31877b1b4d7dcb9d126b63047efaf8c06d580115ec9acee30c8a7212fa55 -b3ee46ce189956ca298057fa8223b7fd1128cf52f39159a58bca03c71dd25161ac13f1472301f72aef3e1993fe1ab269 -a24d7a8d066504fc3f5027ccb13120e2f22896860e02c45b5eba1dbd512d6a17c28f39155ea581619f9d33db43a96f92 -ae9ceacbfe12137db2c1a271e1b34b8f92e4816bad1b3b9b6feecc34df0f8b3b0f7ed0133acdf59c537d43d33fc8d429 -83967e69bf2b361f86361bd705dce0e1ad26df06da6c52b48176fe8dfcbeb03c462c1a4c9e649eff8c654b18c876fdef -9148e6b814a7d779c19c31e33a068e97b597de1f8100513db3c581190513edc4d544801ce3dd2cf6b19e0cd6daedd28a -94ccdafc84920d320ed22de1e754adea072935d3c5f8c2d1378ebe53d140ea29853f056fb3fb1e375846061a038cc9bc -afb43348498c38b0fa5f971b8cdd3a62c844f0eb52bc33daf2f67850af0880fce84ecfb96201b308d9e6168a0d443ae3 -86d5736520a83538d4cd058cc4b4e84213ed00ebd6e7af79ae787adc17a92ba5359e28ba6c91936d967b4b28d24c3070 -b5210c1ff212c5b1e9ef9126e08fe120a41e386bb12c22266f7538c6d69c7fd8774f11c02b81fd4e88f9137b020801fe -b78cfd19f94d24e529d0f52e18ce6185cb238edc6bd43086270fd51dd99f664f43dd4c7d2fe506762fbd859028e13fcf -a6e7220598c554abdcc3fdc587b988617b32c7bb0f82c06205467dbedb58276cc07cae317a190f19d19078773f4c2bbb -b88862809487ee430368dccd85a5d72fa4d163ca4aad15c78800e19c1a95be2192719801e315d86cff7795e0544a77e4 -87ecb13a03921296f8c42ceb252d04716f10e09c93962239fcaa0a7fef93f19ab3f2680bc406170108bc583e9ff2e721 -a810cd473832b6581c36ec4cb403f2849357ba2d0b54df98ef3004b8a530c078032922a81d40158f5fb0043d56477f6e -a247b45dd85ca7fbb718b328f30a03f03c84aef2c583fbdc9fcc9eb8b52b34529e8c8f535505c10598b1b4dac3d7c647 -96ee0b91313c68bac4aa9e065ce9e1d77e51ca4cff31d6a438718c58264dee87674bd97fc5c6b8008be709521e4fd008 -837567ad073e42266951a9a54750919280a2ac835a73c158407c3a2b1904cf0d17b7195a393c71a18ad029cbd9cf79ee -a6a469c44b67ebf02196213e7a63ad0423aab9a6e54acc6fcbdbb915bc043586993454dc3cd9e4be8f27d67c1050879b -8712d380a843b08b7b294f1f06e2f11f4ad6bcc655fdde86a4d8bc739c23916f6fad2b902fe47d6212f03607907e9f0e -920adfb644b534789943cdae1bdd6e42828dda1696a440af2f54e6b97f4f97470a1c6ea9fa6a2705d8f04911d055acd1 -a161c73adf584a0061e963b062f59d90faac65c9b3a936b837a10d817f02fcabfa748824607be45a183dd40f991fe83f -874f4ecd408c76e625ea50bc59c53c2d930ee25baf4b4eca2440bfbffb3b8bc294db579caa7c68629f4d9ec24187c1ba -8bff18087f112be7f4aa654e85c71fef70eee8ae480f61d0383ff6f5ab1a0508f966183bb3fc4d6f29cb7ca234aa50d3 -b03b46a3ca3bc743a173cbc008f92ab1aedd7466b35a6d1ca11e894b9482ea9dc75f8d6db2ddd1add99bfbe7657518b7 -8b4f3691403c3a8ad9e097f02d130769628feddfa8c2b3dfe8cff64e2bed7d6e5d192c1e2ba0ac348b8585e94acd5fa1 -a0d9ca4a212301f97591bf65d5ef2b2664766b427c9dd342e23cb468426e6a56be66b1cb41fea1889ac5d11a8e3c50a5 -8c93ed74188ca23b3df29e5396974b9cc135c91fdefdea6c0df694c8116410e93509559af55533a3776ac11b228d69b1 -82dd331fb3f9e344ebdeeb557769b86a2cc8cc38f6c298d7572a33aea87c261afa9dbd898989139b9fc16bc1e880a099 -a65faedf326bcfd8ef98a51410c78b021d39206704e8291cd1f09e096a66b9b0486be65ff185ca224c45918ac337ddeb -a188b37d363ac072a766fd5d6fa27df07363feff1342217b19e3c37385e42ffde55e4be8355aceaa2f267b6d66b4ac41 -810fa3ba3e96d843e3bafd3f2995727f223d3567c8ba77d684c993ba1773c66551eb5009897c51b3fe9b37196984f5ec -87631537541852da323b4353af45a164f68b304d24c01183bf271782e11687f3fcf528394e1566c2a26cb527b3148e64 -b721cb2b37b3c477a48e3cc0044167d51ff568a5fd2fb606e5aec7a267000f1ddc07d3db919926ae12761a8e017c767c -904dfad4ba2cc1f6e60d1b708438a70b1743b400164cd981f13c064b8328d5973987d4fb9cf894068f29d3deaf624dfb -a70491538893552c20939fae6be2f07bfa84d97e2534a6bbcc0f1729246b831103505e9f60e97a8fa7d2e6c1c2384579 -8726cf1b26b41f443ff7485adcfddc39ace2e62f4d65dd0bb927d933e262b66f1a9b367ded5fbdd6f3b0932553ac1735 -ae8a11cfdf7aa54c08f80cb645e3339187ab3886babe9fae5239ba507bb3dd1c0d161ca474a2df081dcd3d63e8fe445e -92328719e97ce60e56110f30a00ac5d9c7a2baaf5f8d22355d53c1c77941e3a1fec7d1405e6fbf8959665fe2ba7a8cad -8d9d6255b65798d0018a8cccb0b6343efd41dc14ff2058d3eed9451ceaad681e4a0fa6af67b0a04318aa628024e5553d -b70209090055459296006742d946a513f0cba6d83a05249ee8e7a51052b29c0ca9722dc4af5f9816a1b7938a5dac7f79 -aab7b766b9bf91786dfa801fcef6d575dc6f12b77ecc662eb4498f0312e54d0de9ea820e61508fc8aeee5ab5db529349 -a8104b462337748b7f086a135d0c3f87f8e51b7165ca6611264b8fb639d9a2f519926cb311fa2055b5fadf03da70c678 -b0d2460747d5d8b30fc6c6bd0a87cb343ddb05d90a51b465e8f67d499cfc5e3a9e365da05ae233bbee792cdf90ec67d5 -aa55f5bf3815266b4a149f85ed18e451c93de9163575e3ec75dd610381cc0805bb0a4d7c4af5b1f94d10231255436d2c -8d4c6a1944ff94426151909eb5b99cfd92167b967dabe2bf3aa66bb3c26c449c13097de881b2cfc1bf052862c1ef7b03 -8862296162451b9b6b77f03bf32e6df71325e8d7485cf3335d66fd48b74c2a8334c241db8263033724f26269ad95b395 -901aa96deb26cda5d9321190ae6624d357a41729d72ef1abfd71bebf6139af6d690798daba53b7bc5923462115ff748a -96c195ec4992728a1eb38cdde42d89a7bce150db43adbc9e61e279ea839e538deec71326b618dd39c50d589f78fc0614 -b6ff8b8aa0837b99a1a8b46fb37f20ad4aecc6a98381b1308697829a59b8442ffc748637a88cb30c9b1f0f28a926c4f6 -8d807e3dca9e7bef277db1d2cfb372408dd587364e8048b304eff00eacde2c723bfc84be9b98553f83cba5c7b3cba248 -8800c96adb0195c4fc5b24511450dee503c32bf47044f5e2e25bd6651f514d79a2dd9b01cd8c09f3c9d3859338490f57 -89fe366096097e38ec28dd1148887112efa5306cc0c3da09562aafa56f4eb000bf46ff79bf0bdd270cbde6bf0e1c8957 -af409a90c2776e1e7e3760b2042507b8709e943424606e31e791d42f17873a2710797f5baaab4cc4a19998ef648556b0 -8d761863c9b6edbd232d35ab853d944f5c950c2b643f84a1a1327ebb947290800710ff01dcfa26dc8e9828481240e8b1 -90b95e9be1e55c463ed857c4e0617d6dc3674e99b6aa62ed33c8e79d6dfcf7d122f4f4cc2ee3e7c5a49170cb617d2e2e -b3ff381efefabc4db38cc4727432e0301949ae4f16f8d1dea9b4f4de611cf5a36d84290a0bef160dac4e1955e516b3b0 -a8a84564b56a9003adcadb3565dc512239fc79572762cda7b5901a255bc82656bb9c01212ad33d6bef4fbbce18dacc87 -90a081890364b222eef54bf0075417f85e340d2fec8b7375995f598aeb33f26b44143ebf56fca7d8b4ebb36b5747b0eb -ade6ee49e1293224ddf2d8ab7f14bb5be6bc6284f60fd5b3a1e0cf147b73cff57cf19763b8a36c5083badc79c606b103 -b2fa99806dd2fa3de09320b615a2570c416c9bcdb052e592b0aead748bbe407ec9475a3d932ae48b71c2627eb81986a6 -91f3b7b73c8ccc9392542711c45fe6f236057e6efad587d661ad5cb4d6e88265f86b807bb1151736b1009ab74fd7acb4 -8800e2a46af96696dfbdcbf2ca2918b3dcf28ad970170d2d1783b52b8d945a9167d052beeb55f56c126da7ffa7059baa -9862267a1311c385956b977c9aa08548c28d758d7ba82d43dbc3d0a0fd1b7a221d39e8399997fea9014ac509ff510ac4 -b7d24f78886fd3e2d283e18d9ad5a25c1a904e7d9b9104bf47da469d74f34162e27e531380dbbe0a9d051e6ffd51d6e7 -b0f445f9d143e28b9df36b0f2c052da87ee2ca374d9d0fbe2eff66ca6fe5fe0d2c1951b428d58f7314b7e74e45d445ea -b63fc4083eabb8437dafeb6a904120691dcb53ce2938b820bb553da0e1eecd476f72495aacb72600cf9cad18698fd3db -b9ffd8108eaebd582d665f8690fe8bb207fd85185e6dd9f0b355a09bac1bbff26e0fdb172bc0498df025414e88fe2eda -967ed453e1f1a4c5b7b6834cc9f75c13f6889edc0cc91dc445727e9f408487bbf05c337103f61397a10011dfbe25d61d -98ceb673aff36e1987d5521a3984a07079c3c6155974bb8b413e8ae1ce84095fe4f7862fba7aefa14753eb26f2a5805f -85f01d28603a8fdf6ce6a50cb5c44f8a36b95b91302e3f4cd95c108ce8f4d212e73aec1b8d936520d9226802a2bd9136 -88118e9703200ca07910345fbb789e7a8f92bd80bbc79f0a9e040e8767d33df39f6eded403a9b636eabf9101e588482a -90833a51eef1b10ed74e8f9bbd6197e29c5292e469c854eed10b0da663e2bceb92539710b1858bbb21887bd538d28d89 -b513b905ec19191167c6193067b5cfdf5a3d3828375360df1c7e2ced5815437dfd37f0c4c8f009d7fb29ff3c8793f560 -b1b6d405d2d18f9554b8a358cc7e2d78a3b34269737d561992c8de83392ac9a2857be4bf15de5a6c74e0c9d0f31f393c -b828bd3e452b797323b798186607849f85d1fb20c616833c0619360dfd6b3e3aa000fd09dafe4b62d74abc41072ff1a9 -8efde67d0cca56bb2c464731879c9ac46a52e75bac702a63200a5e192b4f81c641f855ca6747752b84fe469cb7113b6c -b2762ba1c89ac3c9a983c242e4d1c2610ff0528585ed5c0dfc8a2c0253551142af9b59f43158e8915a1da7cc26b9df67 -8a3f1157fb820d1497ef6b25cd70b7e16bb8b961b0063ad340d82a79ee76eb2359ca9e15e6d42987ed7f154f5eeaa2da -a75e29f29d38f09c879f971c11beb5368affa084313474a5ecafa2896180b9e47ea1995c2733ec46f421e395a1d9cffe -8e8c3dd3e7196ef0b4996b531ec79e4a1f211db5d5635e48ceb80ff7568b2ff587e845f97ee703bb23a60945ad64314a -8e7f32f4a3e3c584af5e3d406924a0aa34024c42eca74ef6cc2a358fd3c9efaf25f1c03aa1e66bb94b023a2ee2a1cace -ab7dce05d59c10a84feb524fcb62478906b3fa045135b23afbede3bb32e0c678d8ebe59feabccb5c8f3550ea76cae44b -b38bb4b44d827f6fd3bd34e31f9186c59e312dbfadd4a7a88e588da10146a78b1f8716c91ad8b806beb8da65cab80c4c -9490ce9442bbbd05438c7f5c4dea789f74a7e92b1886a730544b55ba377840740a3ae4f2f146ee73f47c9278b0e233bc -83c003fab22a7178eed1a668e0f65d4fe38ef3900044e9ec63070c23f2827d36a1e73e5c2b883ec6a2afe2450171b3b3 -9982f02405978ddc4fca9063ebbdb152f524c84e79398955e66fe51bc7c1660ec1afc3a86ec49f58d7b7dde03505731c -ab337bd83ccdd2322088ffa8d005f450ced6b35790f37ab4534313315ee84312adc25e99cce052863a8bedee991729ed -8312ce4bec94366d88f16127a17419ef64285cd5bf9e5eda010319b48085966ed1252ed2f5a9fd3e0259b91bb65f1827 -a60d5a6327c4041b0c00a1aa2f0af056520f83c9ce9d9ccd03a0bd4d9e6a1511f26a422ea86bd858a1f77438adf07e6c -b84a0a0b030bdad83cf5202aa9afe58c9820e52483ab41f835f8c582c129ee3f34aa096d11c1cd922eda02ea1196a882 -8077d105317f4a8a8f1aadeb05e0722bb55f11abcb490c36c0904401107eb3372875b0ac233144829e734f0c538d8c1d -9202503bd29a6ec198823a1e4e098f9cfe359ed51eb5174d1ca41368821bfeebcbd49debfd02952c41359d1c7c06d2b1 -abc28c155e09365cb77ffead8dc8f602335ef93b2f44e4ef767ce8fc8ef9dd707400f3a722e92776c2e0b40192c06354 -b0f6d1442533ca45c9399e0a63a11f85ff288d242cea6cb3b68c02e77bd7d158047cae2d25b3bcd9606f8f66d9b32855 -b01c3d56a0db84dc94575f4b6ee2de4beca3230e86bed63e2066beb22768b0a8efb08ebaf8ac3dedb5fe46708b084807 -8c8634b0432159f66feaabb165842d1c8ac378f79565b1b90c381aa8450eb4231c3dad11ec9317b9fc2b155c3a771e32 -8e67f623d69ecd430c9ee0888520b6038f13a2b6140525b056dc0951f0cfed2822e62cf11d952a483107c5c5acac4826 -9590bb1cba816dd6acd5ac5fba5142c0a19d53573e422c74005e0bcf34993a8138c83124cad35a3df65879dba6134edd -801cd96cde0749021a253027118d3ea135f3fcdbe895db08a6c145641f95ebd368dd6a1568d995e1d0084146aebe224a -848b5d196427f6fc1f762ee3d36e832b64a76ec1033cfedc8b985dea93932a7892b8ef1035c653fb9dcd9ab2d9a44ac8 -a1017eb83d5c4e2477e7bd2241b2b98c4951a3b391081cae7d75965cadc1acaec755cf350f1f3d29741b0828e36fedea -8d6d2785e30f3c29aad17bd677914a752f831e96d46caf54446d967cb2432be2c849e26f0d193a60bee161ea5c6fe90a -935c0ba4290d4595428e034b5c8001cbd400040d89ab00861108e8f8f4af4258e41f34a7e6b93b04bc253d3b9ffc13bf -aac02257146246998477921cef2e9892228590d323b839f3e64ea893b991b463bc2f47e1e5092ddb47e70b2f5bce7622 -b921fde9412970a5d4c9a908ae8ce65861d06c7679af577cf0ad0d5344c421166986bee471fd6a6cecb7d591f06ec985 -8ef4c37487b139d6756003060600bb6ebac7ea810b9c4364fc978e842f13ac196d1264fbe5af60d76ff6d9203d8e7d3f -94b65e14022b5cf6a9b95f94be5ace2711957c96f4211c3f7bb36206bd39cfbd0ea82186cab5ad0577a23214a5c86e9e -a31c166d2a2ca1d5a75a5920fef7532681f62191a50d8555fdaa63ba4581c3391cc94a536fc09aac89f64eafceec3f90 -919a8cc128de01e9e10f5d83b08b52293fdd41bde2b5ae070f3d95842d4a16e5331cf2f3d61c765570c8022403610fa4 -b23d6f8331eef100152d60483cfa14232a85ee712c8538c9b6417a5a7c5b353c2ac401390c6c215cb101f5cee6b5f43e -ab357160c08a18319510a571eafff154298ce1020de8e1dc6138a09fcb0fcbcdd8359f7e9386bda00b7b9cdea745ffdc -ab55079aea34afa5c0bd1124b9cdfe01f325b402fdfa017301bf87812eaa811ea5798c3aaf818074d420d1c782b10ada -ade616010dc5009e7fc4f8d8b00dc716686a5fa0a7816ad9e503e15839d3b909b69d9dd929b7575376434ffec0d2bea8 -863997b97ed46898a8a014599508fa3079f414b1f4a0c4fdc6d74ae8b444afa350f327f8bfc2a85d27f9e2d049c50135 -8d602ff596334efd4925549ed95f2aa762b0629189f0df6dbb162581657cf3ea6863cd2287b4d9c8ad52813d87fcd235 -b70f68c596dcdeed92ad5c6c348578b26862a51eb5364237b1221e840c47a8702f0fbc56eb520a22c0eed99795d3903e -9628088f8e0853cefadee305a8bf47fa990c50fa96a82511bbe6e5dc81ef4b794e7918a109070f92fc8384d77ace226f -97e26a46e068b605ce96007197ecd943c9a23881862f4797a12a3e96ba2b8d07806ad9e2a0646796b1889c6b7d75188c -b1edf467c068cc163e2d6413cc22b16751e78b3312fe47b7ea82b08a1206d64415b2c8f2a677fa89171e82cc49797150 -a44d15ef18745b251429703e3cab188420e2d974de07251501799b016617f9630643fcd06f895634d8ecdd579e1bf000 -abd126df3917ba48c618ee4dbdf87df506193462f792874439043fa1b844466f6f4e0ff2e42516e63b5b23c0892b2695 -a2a67f57c4aa3c2aa1eeddbfd5009a89c26c2ce8fa3c96a64626aba19514beb125f27df8559506f737de3eae0f1fc18f -a633e0132197e6038197304b296ab171f1d8e0d0f34dcf66fe9146ac385b0239232a8470b9205a4802ab432389f4836d -a914b3a28509a906c3821463b936455d58ff45dcbe158922f9efb2037f2eb0ce8e92532d29b5d5a3fcd0d23fa773f272 -a0e1412ce4505daf1a2e59ce4f0fc0e0023e335b50d2b204422f57cd65744cc7a8ed35d5ef131a42c70b27111d3115b7 -a2339e2f2b6072e88816224fdd612c04d64e7967a492b9f8829db15367f565745325d361fd0607b0def1be384d010d9e -a7309fc41203cb99382e8193a1dcf03ac190a7ce04835304eb7e341d78634e83ea47cb15b885601956736d04cdfcaa01 -81f3ccd6c7f5b39e4e873365f8c37b214e8ab122d04a606fbb7339dc3298c427e922ec7418002561d4106505b5c399ee -92c121cf914ca549130e352eb297872a63200e99b148d88fbc9506ad882bec9d0203d65f280fb5b0ba92e336b7f932e8 -a4b330cf3f064f5b131578626ad7043ce2a433b6f175feb0b52d36134a454ca219373fd30d5e5796410e005b69082e47 -86fe5774112403ad83f9c55d58317eeb17ad8e1176d9f2f69c2afb7ed83bc718ed4e0245ceab4b377f5f062dcd4c00e7 -809d152a7e2654c7fd175b57f7928365a521be92e1ed06c05188a95864ddb25f7cab4c71db7d61bbf4cae46f3a1d96ce -b82d663e55c2a5ada7e169e9b1a87bc1c0177baf1ec1c96559b4cb1c5214ce1ddf2ab8d345014cab6402f3774235cf5a -86580af86df1bd2c385adb8f9a079e925981b7184db66fc5fe5b14cddb82e7d836b06eaeef14924ac529487b23dae111 -b5f5f4c5c94944ecc804df6ab8687d64e27d988cbfeae1ba7394e0f6adbf778c5881ead7cd8082dd7d68542b9bb4ecd5 -a6016916146c2685c46e8fdd24186394e2d5496e77e08c0c6a709d4cd7dfa97f1efcef94922b89196819076a91ad37b5 -b778e7367ded3b6eab53d5fc257f7a87e8faf74a593900f2f517220add2125be3f6142022660d8181df8d164ad9441ce -8581b2d36abe6f553add4d24be761bec1b8efaa2929519114346615380b3c55b59e6ad86990e312f7e234d0203bdf59b -9917e74fd45c3f71a829ff5498a7f6b5599b48c098dda2339bf04352bfc7f368ccf1a407f5835901240e76452ae807d7 -afd196ce6f9335069138fd2e3d133134da253978b4ce373152c0f26affe77a336505787594022e610f8feb722f7cc1fb -a477491a1562e329764645e8f24d8e228e5ef28c9f74c6b5b3abc4b6a562c15ffb0f680d372aed04d9e1bf944dece7be -9767440d58c57d3077319d3a330e5322b9ba16981ec74a5a14d53462eab59ae7fd2b14025bfc63b268862094acb444e6 -80986d921be3513ef69264423f351a61cb48390c1be8673aee0f089076086aaebea7ebe268fd0aa7182695606116f679 -a9554c5c921c07b450ee04e34ec58e054ac1541b26ce2ce5a393367a97348ba0089f53db6660ad76b60278b66fd12e3e -95097e7d2999b3e84bf052c775581cf361325325f4a50192521d8f4693c830bed667d88f482dc1e3f833aa2bd22d2cbf -9014c91d0f85aefd28436b5228c12f6353c055a9326c7efbf5e071e089e2ee7c070fcbc84c5fafc336cbb8fa6fec1ca1 -90f57ba36ee1066b55d37384942d8b57ae00f3cf9a3c1d6a3dfee1d1af42d4b5fa9baeb0cd7e46687d1d6d090ddb931d -8e4b1db12fd760a17214c9e47f1fce6e43c0dbb4589a827a13ac61aaae93759345697bb438a00edab92e0b7b62414683 -8022a959a513cdc0e9c705e0fc04eafd05ff37c867ae0f31f6d01cddd5df86138a426cab2ff0ac8ff03a62e20f7e8f51 -914e9a38829834c7360443b8ed86137e6f936389488eccf05b4b4db7c9425611705076ecb3f27105d24b85c852be7511 -957fb10783e2bd0db1ba66b18e794df710bc3b2b05776be146fa5863c15b1ebdd39747b1a95d9564e1772cdfc4f37b8a -b6307028444daed8ed785ac9d0de76bc3fe23ff2cc7e48102553613bbfb5afe0ebe45e4212a27021c8eb870721e62a1f -8f76143597777d940b15a01b39c5e1b045464d146d9a30a6abe8b5d3907250e6c7f858ff2308f8591e8b0a7b3f3c568a -96163138ac0ce5fd00ae9a289648fd9300a0ca0f63a88481d703ecd281c06a52a3b5178e849e331f9c85ca4ba398f4cc -a63ef47c3e18245b0482596a09f488a716df3cbd0f9e5cfabed0d742843e65db8961c556f45f49762f3a6ac8b627b3ef -8cb595466552e7c4d42909f232d4063e0a663a8ef6f6c9b7ce3a0542b2459cde04e0e54c7623d404acb5b82775ac04f6 -b47fe69960eb45f399368807cff16d941a5a4ebad1f5ec46e3dc8a2e4d598a7e6114d8f0ca791e9720fd786070524e2b -89eb5ff83eea9df490e5beca1a1fbbbbcf7184a37e2c8c91ede7a1e654c81e8cd41eceece4042ea7918a4f4646b67fd6 -a84f5d155ed08b9054eecb15f689ba81e44589e6e7207a99790c598962837ca99ec12344105b16641ca91165672f7153 -a6cc8f25c2d5b2d2f220ec359e6a37a52b95fa6af6e173c65e7cd55299eff4aa9e6d9e6f2769e6459313f1f2aecb0fab -afcde944411f017a9f7979755294981e941cc41f03df5e10522ef7c7505e5f1babdd67b3bf5258e8623150062eb41d9b -8fab39f39c0f40182fcd996ade2012643fe7731808afbc53f9b26900b4d4d1f0f5312d9d40b3df8baa4739970a49c732 -ae193af9726da0ebe7df1f9ee1c4846a5b2a7621403baf8e66c66b60f523e719c30c6b4f897bb14b27d3ff3da8392eeb -8ac5adb82d852eba255764029f42e6da92dcdd0e224d387d1ef94174038db9709ac558d90d7e7c57ad4ce7f89bbfc38c -a2066b3458fdf678ee487a55dd5bfb74fde03b54620cb0e25412a89ee28ad0d685e309a51e3e4694be2fa6f1593a344c -88d031745dd0ae07d61a15b594be5d4b2e2a29e715d081649ad63605e3404b0c3a5353f0fd9fad9c05c18e93ce674fa1 -8283cfb0ef743a043f2b77ecaeba3005e2ca50435585b5dd24777ee6bce12332f85e21b446b536da38508807f0f07563 -b376de22d5f6b0af0b59f7d9764561f4244cf8ffe22890ecd3dcf2ff1832130c9b821e068c9d8773136f4796721e5963 -ae3afc50c764f406353965363840bf28ee85e7064eb9d5f0bb3c31c64ab10f48c853e942ee2c9b51bae59651eaa08c2f -948b204d103917461a01a6c57a88f2d66b476eae5b00be20ec8c747650e864bc8a83aee0aff59cb7584b7a3387e0ee48 -81ab098a082b07f896c5ffd1e4446cb7fb44804cbbf38d125208b233fc82f8ec9a6a8d8dd1c9a1162dc28ffeec0dde50 -a149c6f1312821ced2969268789a3151bdda213451760b397139a028da609c4134ac083169feb0ee423a0acafd10eceb -b0ac9e27a5dadaf523010f730b28f0ebac01f460d3bbbe277dc9d44218abb5686f4fac89ae462682fef9edbba663520a -8d0e0073cca273daaaa61b6fc54bfe5a009bc3e20ae820f6c93ba77b19eca517d457e948a2de5e77678e4241807157cb -ad61d3a2edf7c7533a04964b97499503fd8374ca64286dba80465e68fe932e96749b476f458c6fc57cb1a7ca85764d11 -90eb5e121ae46bc01a30881eaa556f46bd8457a4e80787cf634aab355082de34ac57d7f497446468225f7721e68e2a47 -8cdac557de7c42d1f3780e33dec1b81889f6352279be81c65566cdd4952d4c15d79e656cbd46035ab090b385e90245ef -82b67e61b88b84f4f4d4f65df37b3e3dcf8ec91ea1b5c008fdccd52da643adbe6468a1cfdb999e87d195afe2883a3b46 -8503b467e8f5d6048a4a9b78496c58493a462852cab54a70594ae3fd064cfd0deb4b8f336a262155d9fedcaa67d2f6fd -8db56c5ac763a57b6ce6832930c57117058e3e5a81532b7d19346346205e2ec614eb1a2ee836ef621de50a7bc9b7f040 -ad344699198f3c6e8c0a3470f92aaffc805b76266734414c298e10b5b3797ca53578de7ccb2f458f5e0448203f55282b -80602032c43c9e2a09154cc88b83238343b7a139f566d64cb482d87436b288a98f1ea244fd3bff8da3c398686a900c14 -a6385bd50ecd548cfb37174cdbb89e10025b5cadaf3cff164c95d7aef5a33e3d6a9bf0c681b9e11db9ef54ebeee2a0c1 -abf2d95f4aa34b0581eb9257a0cc8462b2213941a5deb8ba014283293e8b36613951b61261cc67bbd09526a54cbbff76 -a3d5de52f48df72c289ff713e445991f142390798cd42bd9d9dbefaee4af4f5faf09042d126b975cf6b98711c3072553 -8e627302ff3d686cff8872a1b7c2a57b35f45bf2fc9aa42b049d8b4d6996a662b8e7cbac6597f0cb79b0cc4e29fbf133 -8510702e101b39a1efbf4e504e6123540c34b5689645e70d0bac1ecc1baf47d86c05cef6c4317a4e99b4edaeb53f2d00 -aa173f0ecbcc6088f878f8726d317748c81ebf501bba461f163b55d66099b191ec7c55f7702f351a9c8eb42cfa3280e2 -b560a697eafab695bcef1416648a0a664a71e311ecbe5823ae903bd0ed2057b9d7574b9a86d3fe22aa3e6ddce38ea513 -8df6304a3d9cf40100f3f687575419c998cd77e5cc27d579cf4f8e98642de3609af384a0337d145dd7c5635172d26a71 -8105c7f3e4d30a29151849673853b457c1885c186c132d0a98e63096c3774bc9deb956cf957367e633d0913680bda307 -95373fc22c0917c3c2044ac688c4f29a63ed858a45c0d6d2d0fe97afd6f532dcb648670594290c1c89010ecc69259bef -8c2fae9bcadab341f49b55230310df93cac46be42d4caa0d42e45104148a91e527af1b4209c0d972448162aed28fab64 -b05a77baab70683f76209626eaefdda2d36a0b66c780a20142d23c55bd479ddd4ad95b24579384b6cf62c8eb4c92d021 -8e6bc6a7ea2755b4aaa19c1c1dee93811fcde514f03485fdc3252f0ab7f032c315614f6336e57cea25dcfb8fb6084eeb -b656a27d06aade55eadae2ad2a1059198918ea6cc3fd22c0ed881294d34d5ac7b5e4700cc24350e27d76646263b223aa -a296469f24f6f56da92d713afcd4dd606e7da1f79dc4e434593c53695847eefc81c7c446486c4b3b8c8d00c90c166f14 -87a326f57713ac2c9dffeb3af44b9f3c613a8f952676fc46343299122b47ee0f8d792abaa4b5db6451ced5dd153aabd0 -b689e554ba9293b9c1f6344a3c8fcb6951d9f9eac4a2e2df13de021aade7c186be27500e81388e5b8bcab4c80f220a31 -87ae0aa0aa48eac53d1ca5a7b93917de12db9e40ceabf8fdb40884ae771cfdf095411deef7c9f821af0b7070454a2608 -a71ffa7eae8ace94e6c3581d4cb2ad25d48cbd27edc9ec45baa2c8eb932a4773c3272b2ffaf077b40f76942a1f3af7f2 -94c218c91a9b73da6b7a495b3728f3028df8ad9133312fc0c03e8c5253b7ccb83ed14688fd4602e2fd41f29a0bc698bd -ae1e77b90ca33728af07a4c03fb2ef71cd92e2618e7bf8ed4d785ce90097fc4866c29999eb84a6cf1819d75285a03af2 -b7a5945b277dab9993cf761e838b0ac6eaa903d7111fca79f9fde3d4285af7a89bf6634a71909d095d7619d913972c9c -8c43b37be02f39b22029b20aca31bff661abce4471dca88aa3bddefd9c92304a088b2dfc8c4795acc301ca3160656af2 -b32e5d0fba024554bd5fe8a793ebe8003335ddd7f585876df2048dcf759a01285fecb53daae4950ba57f3a282a4d8495 -85ea7fd5e10c7b659df5289b2978b2c89e244f269e061b9a15fcab7983fc1962b63546e82d5731c97ec74b6804be63ef -96b89f39181141a7e32986ac02d7586088c5a9662cec39843f397f3178714d02f929af70630c12cbaba0268f8ba2d4fa -929ab1a2a009b1eb37a2817c89696a06426529ebe3f306c586ab717bd34c35a53eca2d7ddcdef36117872db660024af9 -a696dccf439e9ca41511e16bf3042d7ec0e2f86c099e4fc8879d778a5ea79e33aa7ce96b23dc4332b7ba26859d8e674d -a8fe69a678f9a194b8670a41e941f0460f6e2dbc60470ab4d6ae2679cc9c6ce2c3a39df2303bee486dbfde6844e6b31a -95f58f5c82de2f2a927ca99bf63c9fc02e9030c7e46d0bf6b67fe83a448d0ae1c99541b59caf0e1ccab8326231af09a5 -a57badb2c56ca2c45953bd569caf22968f76ed46b9bac389163d6fe22a715c83d5e94ae8759b0e6e8c2f27bff7748f3f -868726fd49963b24acb5333364dffea147e98f33aa19c7919dc9aca0fd26661cfaded74ede7418a5fadbe7f5ae67b67b -a8d8550dcc64d9f1dd7bcdab236c4122f2b65ea404bb483256d712c7518f08bb028ff8801f1da6aed6cbfc5c7062e33b -97e25a87dae23155809476232178538d4bc05d4ff0882916eb29ae515f2a62bfce73083466cc0010ca956aca200aeacc -b4ea26be3f4bd04aa82d7c4b0913b97bcdf5e88b76c57eb1a336cbd0a3eb29de751e1bc47c0e8258adec3f17426d0c71 -99ee555a4d9b3cf2eb420b2af8e3bc99046880536116d0ce7193464ac40685ef14e0e3c442f604e32f8338cb0ef92558 -8c64efa1da63cd08f319103c5c7a761221080e74227bbc58b8fb35d08aa42078810d7af3e60446cbaff160c319535648 -8d9fd88040076c28420e3395cbdfea402e4077a3808a97b7939d49ecbcf1418fe50a0460e1c1b22ac3f6e7771d65169a -ae3c19882d7a9875d439265a0c7003c8d410367627d21575a864b9cb4918de7dbdb58a364af40c5e045f3df40f95d337 -b4f7bfacab7b2cafe393f1322d6dcc6f21ffe69cd31edc8db18c06f1a2b512c27bd0618091fd207ba8df1808e9d45914 -94f134acd0007c623fb7934bcb65ef853313eb283a889a3ffa79a37a5c8f3665f3d5b4876bc66223610c21dc9b919d37 -aa15f74051171daacdc1f1093d3f8e2d13da2833624b80a934afec86fc02208b8f55d24b7d66076444e7633f46375c6a -a32d6bb47ef9c836d9d2371807bafbbbbb1ae719530c19d6013f1d1f813c49a60e4fa51d83693586cba3a840b23c0404 -b61b3599145ea8680011aa2366dc511a358b7d67672d5b0c5be6db03b0efb8ca5a8294cf220ea7409621f1664e00e631 -859cafc3ee90b7ececa1ed8ef2b2fc17567126ff10ca712d5ffdd16aa411a5a7d8d32c9cab1fbf63e87dce1c6e2f5f53 -a2fef1b0b2874387010e9ae425f3a9676d01a095d017493648bcdf3b31304b087ccddb5cf76abc4e1548b88919663b6b -939e18c73befc1ba2932a65ede34c70e4b91e74cc2129d57ace43ed2b3af2a9cc22a40fbf50d79a63681b6d98852866d -b3b4259d37b1b14aee5b676c9a0dd2d7f679ab95c120cb5f09f9fbf10b0a920cb613655ddb7b9e2ba5af4a221f31303c -997255fe51aaca6e5a9cb3359bcbf25b2bb9e30649bbd53a8a7c556df07e441c4e27328b38934f09c09d9500b5fabf66 -abb91be2a2d860fd662ed4f1c6edeefd4da8dc10e79251cf87f06029906e7f0be9b486462718f0525d5e049472692cb7 -b2398e593bf340a15f7801e1d1fbda69d93f2a32a889ec7c6ae5e8a37567ac3e5227213c1392ee86cfb3b56ec2787839 -8ddf10ccdd72922bed36829a36073a460c2118fc7a56ff9c1ac72581c799b15c762cb56cb78e3d118bb9f6a7e56cb25e -93e6bc0a4708d16387cacd44cf59363b994dc67d7ada7b6d6dbd831c606d975247541b42b2a309f814c1bfe205681fc6 -b93fc35c05998cffda2978e12e75812122831523041f10d52f810d34ff71944979054b04de0117e81ddf5b0b4b3e13c0 -92221631c44d60d68c6bc7b287509f37ee44cbe5fdb6935cee36b58b17c7325098f98f7910d2c3ca5dc885ad1d6dabc7 -a230124424a57fad3b1671f404a94d7c05f4c67b7a8fbacfccea28887b78d7c1ed40b92a58348e4d61328891cd2f6cee -a6a230edb8518a0f49d7231bc3e0bceb5c2ac427f045819f8584ba6f3ae3d63ed107a9a62aad543d7e1fcf1f20605706 -845be1fe94223c7f1f97d74c49d682472585d8f772762baad8a9d341d9c3015534cc83d102113c51a9dea2ab10d8d27b -b44262515e34f2db597c8128c7614d33858740310a49cdbdf9c8677c5343884b42c1292759f55b8b4abc4c86e4728033 -805592e4a3cd07c1844bc23783408310accfdb769cca882ad4d07d608e590a288b7370c2cb327f5336e72b7083a0e30f -95153e8b1140df34ee864f4ca601cb873cdd3efa634af0c4093fbaede36f51b55571ab271e6a133020cd34db8411241f -82878c1285cfa5ea1d32175c9401f3cc99f6bb224d622d3fd98cc7b0a27372f13f7ab463ce3a33ec96f9be38dbe2dfe3 -b7588748f55783077c27fc47d33e20c5c0f5a53fc0ac10194c003aa09b9f055d08ec971effa4b7f760553997a56967b3 -b36b4de6d1883b6951f59cfae381581f9c6352fcfcf1524fccdab1571a20f80441d9152dc6b48bcbbf00371337ca0bd5 -89c5523f2574e1c340a955cbed9c2f7b5fbceb260cb1133160dabb7d41c2f613ec3f6e74bbfab3c4a0a6f0626dbe068f -a52f58cc39f968a9813b1a8ddc4e83f4219e4dd82c7aa1dd083bea7edf967151d635aa9597457f879771759b876774e4 -8300a67c2e2e123f89704abfde095463045dbd97e20d4c1157bab35e9e1d3d18f1f4aaba9cbe6aa2d544e92578eaa1b6 -ac6a7f2918768eb6a43df9d3a8a04f8f72ee52f2e91c064c1c7d75cad1a3e83e5aba9fe55bb94f818099ac91ccf2e961 -8d64a2b0991cf164e29835c8ddef6069993a71ec2a7de8157bbfa2e00f6367be646ed74cbaf524f0e9fe13fb09fa15fd -8b2ffe5a545f9f680b49d0a9797a4a11700a2e2e348c34a7a985fc278f0f12def6e06710f40f9d48e4b7fbb71e072229 -8ab8f71cd337fa19178924e961958653abf7a598e3f022138b55c228440a2bac4176cea3aea393549c03cd38a13eb3fc -8419d28318c19ea4a179b7abb43669fe96347426ef3ac06b158d79c0acf777a09e8e770c2fb10e14b3a0421705990b23 -8bacdac310e1e49660359d0a7a17fe3d334eb820e61ae25e84cb52f863a2f74cbe89c2e9fc3283745d93a99b79132354 -b57ace3fa2b9f6b2db60c0d861ace7d7e657c5d35d992588aeed588c6ce3a80b6f0d49f8a26607f0b17167ab21b675e4 -83e265cde477f2ecc164f49ddc7fb255bb05ff6adc347408353b7336dc3a14fdedc86d5a7fb23f36b8423248a7a67ed1 -a60ada971f9f2d79d436de5d3d045f5ab05308cae3098acaf5521115134b2a40d664828bb89895840db7f7fb499edbc5 -a63eea12efd89b62d3952bf0542a73890b104dd1d7ff360d4755ebfa148fd62de668edac9eeb20507967ea37fb220202 -a0275767a270289adc991cc4571eff205b58ad6d3e93778ddbf95b75146d82517e8921bd0d0564e5b75fa0ccdab8e624 -b9b03fd3bf07201ba3a039176a965d736b4ef7912dd9e9bf69fe1b57c330a6aa170e5521fe8be62505f3af81b41d7806 -a95f640e26fb1106ced1729d6053e41a16e4896acac54992279ff873e5a969aad1dcfa10311e28b8f409ac1dab7f03bb -b144778921742418053cb3c70516c63162c187f00db2062193bb2c14031075dbe055d020cde761b26e8c58d0ea6df2c1 -8432fbb799e0435ef428d4fefc309a05dd589bce74d7a87faf659823e8c9ed51d3e42603d878e80f439a38be4321c2fa -b08ddef14e42d4fd5d8bf39feb7485848f0060d43b51ed5bdda39c05fe154fb111d29719ee61a23c392141358c0cfcff -8ae3c5329a5e025b86b5370e06f5e61177df4bda075856fade20a17bfef79c92f54ed495f310130021ba94fb7c33632b -92b6d3c9444100b4d7391febfc1dddaa224651677c3695c47a289a40d7a96d200b83b64e6d9df51f534564f272a2c6c6 -b432bc2a3f93d28b5e506d68527f1efeb2e2570f6be0794576e2a6ef9138926fdad8dd2eabfa979b79ab7266370e86bc -8bc315eacedbcfc462ece66a29662ca3dcd451f83de5c7626ef8712c196208fb3d8a0faf80b2e80384f0dd9772f61a23 -a72375b797283f0f4266dec188678e2b2c060dfed5880fc6bb0c996b06e91a5343ea2b695adaab0a6fd183b040b46b56 -a43445036fbaa414621918d6a897d3692fdae7b2961d87e2a03741360e45ebb19fcb1703d23f1e15bb1e2babcafc56ac -b9636b2ffe305e63a1a84bd44fb402442b1799bd5272638287aa87ca548649b23ce8ce7f67be077caed6aa2dbc454b78 -99a30bf0921d854c282b83d438a79f615424f28c2f99d26a05201c93d10378ab2cd94a792b571ddae5d4e0c0013f4006 -8648e3c2f93d70b392443be116b48a863e4b75991bab5db656a4ef3c1e7f645e8d536771dfe4e8d1ceda3be8d32978b0 -ab50dc9e6924c1d2e9d2e335b2d679fc7d1a7632e84964d3bac0c9fe57e85aa5906ec2e7b0399d98ddd022e9b19b5904 -ab729328d98d295f8f3272afaf5d8345ff54d58ff9884da14f17ecbdb7371857fdf2f3ef58080054e9874cc919b46224 -83fa5da7592bd451cad3ad7702b4006332b3aae23beab4c4cb887fa6348317d234bf62a359e665b28818e5410c278a09 -8bdbff566ae9d368f114858ef1f009439b3e9f4649f73efa946e678d6c781d52c69af195df0a68170f5f191b2eac286b -91245e59b4425fd4edb2a61d0d47c1ccc83d3ced8180de34887b9655b5dcda033d48cde0bdc3b7de846d246c053a02e8 -a2cb00721e68f1cad8933947456f07144dc69653f96ceed845bd577d599521ba99cdc02421118971d56d7603ed118cbf -af8cd66d303e808b22ec57860dd909ca64c27ec2c60e26ffecfdc1179d8762ffd2739d87b43959496e9fee4108df71df -9954136812dffcd5d3f167a500e7ab339c15cfc9b3398d83f64b0daa3dd5b9a851204f424a3493b4e326d3de81e50a62 -93252254d12511955f1aa464883ad0da793f84d900fea83e1df8bca0f2f4cf5b5f9acbaec06a24160d33f908ab5fea38 -997cb55c26996586ba436a95566bd535e9c22452ca5d2a0ded2bd175376557fa895f9f4def4519241ff386a063f2e526 -a12c78ad451e0ac911260ade2927a768b50cb4125343025d43474e7f465cdc446e9f52a84609c5e7e87ae6c9b3f56cda -a789d4ca55cbba327086563831b34487d63d0980ba8cf55197c016702ed6da9b102b1f0709ce3da3c53ff925793a3d73 -a5d76acbb76741ce85be0e655b99baa04f7f587347947c0a30d27f8a49ae78cce06e1cde770a8b618d3db402be1c0c4b -873c0366668c8faddb0eb7c86f485718d65f8c4734020f1a18efd5fa123d3ea8a990977fe13592cd01d17e60809cb5ff -b659b71fe70f37573ff7c5970cc095a1dc0da3973979778f80a71a347ef25ad5746b2b9608bad4ab9a4a53a4d7df42d7 -a34cbe05888e5e5f024a2db14cb6dcdc401a9cbd13d73d3c37b348f68688f87c24ca790030b8f84fef9e74b4eab5e412 -94ce8010f85875c045b0f014db93ef5ab9f1f6842e9a5743dce9e4cb872c94affd9e77c1f1d1ab8b8660b52345d9acb9 -adefa9b27a62edc0c5b019ddd3ebf45e4de846165256cf6329331def2e088c5232456d3de470fdce3fa758bfdd387512 -a6b83821ba7c1f83cc9e4529cf4903adb93b26108e3d1f20a753070db072ad5a3689643144bdd9c5ea06bb9a7a515cd0 -a3a9ddedc2a1b183eb1d52de26718151744db6050f86f3580790c51d09226bf05f15111691926151ecdbef683baa992c -a64bac89e7686932cdc5670d07f0b50830e69bfb8c93791c87c7ffa4913f8da881a9d8a8ce8c1a9ce5b6079358c54136 -a77b5a63452cb1320b61ab6c7c2ef9cfbcade5fd4727583751fb2bf3ea330b5ca67757ec1f517bf4d503ec924fe32fbd -8746fd8d8eb99639d8cd0ca34c0d9c3230ed5a312aab1d3d925953a17973ee5aeb66e68667e93caf9cb817c868ea8f3d -88a2462a26558fc1fbd6e31aa8abdc706190a17c27fdc4217ffd2297d1b1f3321016e5c4b2384c5454d5717dc732ed03 -b78893a97e93d730c8201af2e0d3b31cb923d38dc594ffa98a714e627c473d42ea82e0c4d2eeb06862ee22a9b2c54588 -920cc8b5f1297cf215a43f6fc843e379146b4229411c44c0231f6749793d40f07b9af7699fd5d21fd69400b97febe027 -a0f0eafce1e098a6b58c7ad8945e297cd93aaf10bc55e32e2e32503f02e59fc1d5776936577d77c0b1162cb93b88518b -98480ba0064e97a2e7a6c4769b4d8c2a322cfc9a3b2ca2e67e9317e2ce04c6e1108169a20bd97692e1cb1f1423b14908 -83dbbb2fda7e287288011764a00b8357753a6a44794cc8245a2275237f11affdc38977214e463ad67aec032f3dfa37e9 -86442fff37598ce2b12015ff19b01bb8a780b40ad353d143a0f30a06f6d23afd5c2b0a1253716c855dbf445cc5dd6865 -b8a4c60c5171189414887847b9ed9501bff4e4c107240f063e2d254820d2906b69ef70406c585918c4d24f1dd052142b -919f33a98e84015b2034b57b5ffe9340220926b2c6e45f86fd79ec879dbe06a148ae68b77b73bf7d01bd638a81165617 -95c13e78d89474a47fbc0664f6f806744b75dede95a479bbf844db4a7f4c3ae410ec721cb6ffcd9fa9c323da5740d5ae -ab7151acc41fffd8ec6e90387700bcd7e1cde291ea669567295bea1b9dd3f1df2e0f31f3588cd1a1c08af8120aca4921 -80e74c5c47414bd6eeef24b6793fb1fa2d8fb397467045fcff887c52476741d5bc4ff8b6d3387cb53ad285485630537f -a296ad23995268276aa351a7764d36df3a5a3cffd7dbeddbcea6b1f77adc112629fdeffa0918b3242b3ccd5e7587e946 -813d2506a28a2b01cb60f49d6bd5e63c9b056aa56946faf2f33bd4f28a8d947569cfead3ae53166fc65285740b210f86 -924b265385e1646287d8c09f6c855b094daaee74b9e64a0dddcf9ad88c6979f8280ba30c8597b911ef58ddb6c67e9fe3 -8d531513c70c2d3566039f7ca47cd2352fd2d55b25675a65250bdb8b06c3843db7b2d29c626eed6391c238fc651cf350 -82b338181b62fdc81ceb558a6843df767b6a6e3ceedc5485664b4ea2f555904b1a45fbb35f6cf5d96f27da10df82a325 -92e62faaedea83a37f314e1d3cb4faaa200178371d917938e59ac35090be1db4b4f4e0edb78b9c991de202efe4f313d8 -99d645e1b642c2dc065bac9aaa0621bc648c9a8351efb6891559c3a41ba737bd155fb32d7731950514e3ecf4d75980e4 -b34a13968b9e414172fb5d5ece9a39cf2eb656128c3f2f6cc7a9f0c69c6bae34f555ecc8f8837dc34b5e470e29055c78 -a2a0bb7f3a0b23a2cbc6585d59f87cd7e56b2bbcb0ae48f828685edd9f7af0f5edb4c8e9718a0aaf6ef04553ba71f3b7 -8e1a94bec053ed378e524b6685152d2b52d428266f2b6eadd4bcb7c4e162ed21ab3e1364879673442ee2162635b7a4d8 -9944adaff14a85eab81c73f38f386701713b52513c4d4b838d58d4ffa1d17260a6d056b02334850ea9a31677c4b078bd -a450067c7eceb0854b3eca3db6cf38669d72cb7143c3a68787833cbca44f02c0be9bfbe082896f8a57debb13deb2afb1 -8be4ad3ac9ef02f7df09254d569939757101ee2eda8586fefcd8c847adc1efe5bdcb963a0cafa17651befaafb376a531 -90f6de91ea50255f148ac435e08cf2ac00c772a466e38155bd7e8acf9197af55662c7b5227f88589b71abe9dcf7ba343 -86e5a24f0748b106dee2d4d54e14a3b0af45a96cbee69cac811a4196403ebbee17fd24946d7e7e1b962ac7f66dbaf610 -afdd96fbcda7aa73bf9eeb2292e036c25753d249caee3b9c013009cc22e10d3ec29e2aa6ddbb21c4e949b0c0bccaa7f4 -b5a4e7436d5473647c002120a2cb436b9b28e27ad4ebdd7c5f122b91597c507d256d0cbd889d65b3a908531936e53053 -b632414c3da704d80ac2f3e5e0e9f18a3637cdc2ebeb613c29300745582427138819c4e7b0bec3099c1b8739dac1807b -a28df1464d3372ce9f37ef1db33cc010f752156afae6f76949d98cd799c0cf225c20228ae86a4da592d65f0cffe3951b -898b93d0a31f7d3f11f253cb7a102db54b669fd150da302d8354d8e02b1739a47cb9bd88015f3baf12b00b879442464e -96fb88d89a12049091070cb0048a381902965e67a8493e3991eaabe5d3b7ff7eecd5c94493a93b174df3d9b2c9511755 -b899cb2176f59a5cfba3e3d346813da7a82b03417cad6342f19cc8f12f28985b03bf031e856a4743fd7ebe16324805b0 -a60e2d31bc48e0c0579db15516718a03b73f5138f15037491f4dae336c904e312eda82d50862f4debd1622bb0e56d866 -979fc8b987b5cef7d4f4b58b53a2c278bd25a5c0ea6f41c715142ea5ff224c707de38451b0ad3aa5e749aa219256650a -b2a75bff18e1a6b9cf2a4079572e41205741979f57e7631654a3c0fcec57c876c6df44733c9da3d863db8dff392b44a3 -b7a0f0e811222c91e3df98ff7f286b750bc3b20d2083966d713a84a2281744199e664879401e77470d44e5a90f3e5181 -82b74ba21c9d147fbc338730e8f1f8a6e7fc847c3110944eb17a48bea5e06eecded84595d485506d15a3e675fd0e5e62 -a7f44eef817d5556f0d1abcf420301217d23c69dd2988f44d91ea1f1a16c322263cbacd0f190b9ba22b0f141b9267b4f -aadb68164ede84fc1cb3334b3194d84ba868d5a88e4c9a27519eef4923bc4abf81aab8114449496c073c2a6a0eb24114 -b5378605fabe9a8c12a5dc55ef2b1de7f51aedb61960735c08767a565793cea1922a603a6983dc25f7cea738d0f7c40d -a97a4a5cd8d51302e5e670aee78fe6b5723f6cc892902bbb4f131e82ca1dfd5de820731e7e3367fb0c4c1922a02196e3 -8bdfeb15c29244d4a28896f2b2cb211243cd6a1984a3f5e3b0ebe5341c419beeab3304b390a009ffb47588018034b0ea -a9af3022727f2aa2fca3b096968e97edad3f08edcbd0dbca107b892ae8f746a9c0485e0d6eb5f267999b23a845923ed0 -8e7594034feef412f055590fbb15b6322dc4c6ab7a4baef4685bd13d71a83f7d682b5781bdfa0d1c659489ce9c2b8000 -84977ca6c865ebee021c58106c1a4ad0c745949ecc5332948002fd09bd9b890524878d0c29da96fd11207621136421fe -8687551a79158e56b2375a271136756313122132a6670fa51f99a1b5c229ed8eea1655a734abae13228b3ebfd2a825dd -a0227d6708979d99edfc10f7d9d3719fd3fc68b0d815a7185b60307e4c9146ad2f9be2b8b4f242e320d4288ceeb9504c -89f75583a16735f9dd8b7782a130437805b34280ccea8dac6ecaee4b83fe96947e7b53598b06fecfffdf57ffc12cc445 -a0056c3353227f6dd9cfc8e3399aa5a8f1d71edf25d3d64c982910f50786b1e395c508d3e3727ac360e3e040c64b5298 -b070e61a6d813626144b312ded1788a6d0c7cec650a762b2f8df6e4743941dd82a2511cd956a3f141fc81e15f4e092da -b4e6db232e028a1f989bb5fc13416711f42d389f63564d60851f009dcffac01acfd54efa307aa6d4c0f932892d4e62b0 -89b5991a67db90024ddd844e5e1a03ef9b943ad54194ae0a97df775dde1addf31561874f4e40fbc37a896630f3bbda58 -ad0e8442cb8c77d891df49cdb9efcf2b0d15ac93ec9be1ad5c3b3cca1f4647b675e79c075335c1f681d56f14dc250d76 -b5d55a6ae65bb34dd8306806cb49b5ccb1c83a282ee47085cf26c4e648e19a52d9c422f65c1cd7e03ca63e926c5e92ea -b749501347e5ec07e13a79f0cb112f1b6534393458b3678a77f02ca89dca973fa7b30e55f0b25d8b92b97f6cb0120056 -94144b4a3ffc5eec6ba35ce9c245c148b39372d19a928e236a60e27d7bc227d18a8cac9983851071935d8ffb64b3a34f -92bb4f9f85bc8c028a3391306603151c6896673135f8a7aefedd27acb322c04ef5dac982fc47b455d6740023e0dd3ea3 -b9633a4a101461a782fc2aa092e9dbe4e2ad00987578f18cd7cf0021a909951d60fe79654eb7897806795f93c8ff4d1c -809f0196753024821b48a016eca5dbb449a7c55750f25981bb7a4b4c0e0846c09b8f6128137905055fc43a3f0deb4a74 -a27dc9cdd1e78737a443570194a03d89285576d3d7f3a3cf15cc55b3013e42635d4723e2e8fe1d0b274428604b630db9 -861f60f0462e04cd84924c36a28163def63e777318d00884ab8cb64c8df1df0bce5900342163edb60449296484a6c5bf -b7bc23fb4e14af4c4704a944253e760adefeca8caee0882b6bbd572c84434042236f39ae07a8f21a560f486b15d82819 -b9a6eb492d6dd448654214bd01d6dc5ff12067a11537ab82023fc16167507ee25eed2c91693912f4155d1c07ed9650b3 -97678af29c68f9a5e213bf0fb85c265303714482cfc4c2c00b4a1e8a76ed08834ee6af52357b143a1ca590fb0265ea5a -8a15b499e9eca5b6cac3070b5409e8296778222018ad8b53a5d1f6b70ad9bb10c68a015d105c941ed657bf3499299e33 -b487fefede2e8091f2c7bfe85770db2edff1db83d4effe7f7d87bff5ab1ace35e9b823a71adfec6737fede8d67b3c467 -8b51b916402aa2c437fce3bcad6dad3be8301a1a7eab9d163085b322ffb6c62abf28637636fe6114573950117fc92898 -b06a2106d031a45a494adec0881cb2f82275dff9dcdd2bc16807e76f3bec28a6734edd3d54f0be8199799a78cd6228ad -af0a185391bbe2315eb97feac98ad6dd2e5d931d012c621abd6e404a31cc188b286fef14871762190acf086482b2b5e2 -8e78ee8206506dd06eb7729e32fceda3bebd8924a64e4d8621c72e36758fda3d0001af42443851d6c0aea58562870b43 -a1ba52a569f0461aaf90b49b92be976c0e73ec4a2c884752ee52ffb62dd137770c985123d405dfb5de70692db454b54a -8d51b692fa1543c51f6b62b9acb8625ed94b746ef96c944ca02859a4133a5629da2e2ce84e111a7af8d9a5b836401c64 -a7a20d45044cf6492e0531d0b8b26ffbae6232fa05a96ed7f06bdb64c2b0f5ca7ec59d5477038096a02579e633c7a3ff -84df867b98c53c1fcd4620fef133ee18849c78d3809d6aca0fb6f50ff993a053a455993f216c42ab6090fa5356b8d564 -a7227c439f14c48e2577d5713c97a5205feb69acb0b449152842e278fa71e8046adfab468089c8b2288af1fc51fa945b -855189b3a105670779997690876dfaa512b4a25a24931a912c2f0f1936971d2882fb4d9f0b3d9daba77eaf660e9d05d5 -b5696bd6706de51c502f40385f87f43040a5abf99df705d6aac74d88c913b8ecf7a99a63d7a37d9bdf3a941b9e432ff5 -ab997beb0d6df9c98d5b49864ef0b41a2a2f407e1687dfd6089959757ba30ed02228940b0e841afe6911990c74d536c4 -b36b65f85546ebfdbe98823d5555144f96b4ab39279facd19c0de3b8919f105ba0315a0784dce4344b1bc62d8bb4a5a3 -b8371f0e4450788720ac5e0f6cd3ecc5413d33895083b2c168d961ec2b5c3de411a4cc0712481cbe8df8c2fa1a7af006 -98325d8026b810a8b7a114171ae59a57e8bbc9848e7c3df992efc523621729fd8c9f52114ce01d7730541a1ada6f1df1 -8d0e76dbd37806259486cd9a31bc8b2306c2b95452dc395546a1042d1d17863ef7a74c636b782e214d3aa0e8d717f94a -a4e15ead76da0214d702c859fb4a8accdcdad75ed08b865842bd203391ec4cba2dcc916455e685f662923b96ee0c023f -8618190972086ebb0c4c1b4a6c94421a13f378bc961cc8267a301de7390c5e73c3333864b3b7696d81148f9d4843fd02 -85369d6cc7342e1aa15b59141517d8db8baaaeb7ab9670f3ba3905353948d575923d283b7e5a05b13a30e7baf1208a86 -87c51ef42233c24a6da901f28c9a075d9ba3c625687c387ad6757b72ca6b5a8885e6902a3082da7281611728b1e45f26 -aa6348a4f71927a3106ad0ea8b02fc8d8c65531e4ab0bd0a17243e66f35afe252e40ab8eef9f13ae55a72566ffdaff5c -96a3bc976e9d03765cc3fee275fa05b4a84c94fed6b767e23ca689394501e96f56f7a97cffddc579a6abff632bf153be -97dbf96c6176379fdb2b888be4e757b2bca54e74124bd068d3fa1dbd82a011bbeb75079da38e0cd22a761fe208ecad9b -b70cf0a1d14089a4129ec4e295313863a59da8c7e26bf74cc0e704ed7f0ee4d7760090d0ddf7728180f1bf2c5ac64955 -882d664714cc0ffe53cbc9bef21f23f3649824f423c4dbad1f893d22c4687ab29583688699efc4d5101aa08b0c3e267a -80ecb7cc963e677ccaddbe3320831dd6ee41209acf4ed41b16dc4817121a3d86a1aac9c4db3d8c08a55d28257088af32 -a25ba667d832b145f9ce18c3f9b1bd00737aa36db020e1b99752c8ef7d27c6c448982bd8d352e1b6df266b8d8358a8d5 -83734841c13dee12759d40bdd209b277e743b0d08cc0dd1e0b7afd2d65bfa640400eefcf6be4a52e463e5b3d885eeac6 -848d16505b04804afc773aebabb51b36fd8aacfbb0e09b36c0d5d57df3c0a3b92f33e7d5ad0a7006ec46ebb91df42b8c -909a8d793f599e33bb9f1dc4792a507a97169c87cd5c087310bc05f30afcd247470b4b56dec59894c0fb1d48d39bb54e -8e558a8559df84a1ba8b244ece667f858095c50bb33a5381e60fcc6ba586b69693566d8819b4246a27287f16846c1dfa -84d6b69729f5aaa000cd710c2352087592cfbdf20d5e1166977e195818e593fa1a50d1e04566be23163a2523dc1612f1 -9536d262b7a42125d89f4f32b407d737ba8d9242acfc99d965913ab3e043dcac9f7072a43708553562cac4cba841df30 -9598548923ca119d6a15fd10861596601dd1dedbcccca97bb208cdc1153cf82991ea8cc17686fbaa867921065265970c -b87f2d4af6d026e4d2836bc3d390a4a18e98a6e386282ce96744603bab74974272e97ac2da281afa21885e2cbb3a8001 -991ece62bf07d1a348dd22191868372904b9f8cf065ae7aa4e44fd24a53faf6d851842e35fb472895963aa1992894918 -a8c53dea4c665b30e51d22ca6bc1bc78aaf172b0a48e64a1d4b93439b053877ec26cb5221c55efd64fa841bbf7d5aff4 -93487ec939ed8e740f15335b58617c3f917f72d07b7a369befd479ae2554d04deb240d4a14394b26192efae4d2f4f35d -a44793ab4035443f8f2968a40e043b4555960193ffa3358d22112093aadfe2c136587e4139ffd46d91ed4107f61ea5e0 -b13fe033da5f0d227c75927d3dacb06dbaf3e1322f9d5c7c009de75cdcba5e308232838785ab69a70f0bedea755e003f -970a29b075faccd0700fe60d1f726bdebf82d2cc8252f4a84543ebd3b16f91be42a75c9719a39c4096139f0f31393d58 -a4c3eb1f7160f8216fc176fb244df53008ff32f2892363d85254002e66e2de21ccfe1f3b1047589abee50f29b9d507e3 -8c552885eab04ba40922a8f0c3c38c96089c95ff1405258d3f1efe8d179e39e1295cbf67677894c607ae986e4e6b1fb0 -b3671746fa7f848c4e2ae6946894defadd815230b906b419143523cc0597bc1d6c0a4c1e09d49b66b4a2c11cde3a4de3 -937a249a95813a5e2ef428e355efd202e15a37d73e56cfb7e57ea9f943f2ce5ca8026f2f1fd25bf164ba89d07077d858 -83646bdf6053a04aa9e2f112499769e5bd5d0d10f2e13db3ca89bd45c0b3b7a2d752b7d137fb3909f9c62b78166c9339 -b4eac4b91e763666696811b7ed45e97fd78310377ebea1674b58a2250973f80492ac35110ed1240cd9bb2d17493d708c -82db43a99bc6573e9d92a3fd6635dbbb249ac66ba53099c3c0c8c8080b121dd8243cd5c6e36ba0a4d2525bae57f5c89c -a64d6a264a681b49d134c655d5fc7756127f1ee7c93d328820f32bca68869f53115c0d27fef35fe71f7bc4fdaed97348 -8739b7a9e2b4bc1831e7f04517771bc7cde683a5e74e052542517f8375a2f64e53e0d5ac925ef722327e7bb195b4d1d9 -8f337cdd29918a2493515ebb5cf702bbe8ecb23b53c6d18920cc22f519e276ca9b991d3313e2d38ae17ae8bdfa4f8b7e -b0edeab9850e193a61f138ef2739fc42ceec98f25e7e8403bfd5fa34a7bc956b9d0898250d18a69fa4625a9b3d6129da -a9920f26fe0a6d51044e623665d998745c9eca5bce12051198b88a77d728c8238f97d4196f26e43b24f8841500b998d0 -86e655d61502b979eeeeb6f9a7e1d0074f936451d0a1b0d2fa4fb3225b439a3770767b649256fe481361f481a8dbc276 -84d3b32fa62096831cc3bf013488a9f3f481dfe293ae209ed19585a03f7db8d961a7a9dd0db82bd7f62d612707575d9c -81c827826ec9346995ffccf62a241e3b2d32f7357acd1b1f8f7a7dbc97022d3eb51b8a1230e23ce0b401d2e535e8cd78 -94a1e40c151191c5b055b21e86f32e69cbc751dcbdf759a48580951834b96a1eed75914c0d19a38aefd21fb6c8d43d0c -ab890222b44bc21b71f7c75e15b6c6e16bb03371acce4f8d4353ff3b8fcd42a14026589c5ed19555a3e15e4d18bfc3a3 -accb0be851e93c6c8cc64724cdb86887eea284194b10e7a43c90528ed97e9ec71ca69c6fac13899530593756dd49eab2 -b630220aa9e1829c233331413ee28c5efe94ea8ea08d0c6bfd781955078b43a4f92915257187d8526873e6c919c6a1de -add389a4d358c585f1274b73f6c3c45b58ef8df11f9d11221f620e241bf3579fba07427b288c0c682885a700cc1fa28d -a9fe6ca8bf2961a3386e8b8dcecc29c0567b5c0b3bcf3b0f9169f88e372b80151af883871fc5229815f94f43a6f5b2b0 -ad839ae003b92b37ea431fa35998b46a0afc3f9c0dd54c3b3bf7a262467b13ff3c323ada1c1ae02ac7716528bdf39e3e -9356d3fd0edcbbb65713c0f2a214394f831b26f792124b08c5f26e7f734b8711a87b7c4623408da6a091c9aef1f6af3c -896b25b083c35ac67f0af3784a6a82435b0e27433d4d74cd6d1eafe11e6827827799490fb1c77c11de25f0d75f14e047 -8bfa019391c9627e8e5f05c213db625f0f1e51ec68816455f876c7e55b8f17a4f13e5aae9e3fb9e1cf920b1402ee2b40 -8ba3a6faa6a860a8f3ce1e884aa8769ceded86380a86520ab177ab83043d380a4f535fe13884346c5e51bee68da6ab41 -a8292d0844084e4e3bb7af92b1989f841a46640288c5b220fecfad063ee94e86e13d3d08038ec2ac82f41c96a3bfe14d -8229bb030b2fc566e11fd33c7eab7a1bb7b49fed872ea1f815004f7398cb03b85ea14e310ec19e1f23e0bdaf60f8f76c -8cfbf869ade3ec551562ff7f63c2745cc3a1f4d4dc853a0cd42dd5f6fe54228f86195ea8fe217643b32e9f513f34a545 -ac52a3c8d3270ddfe1b5630159da9290a5ccf9ccbdef43b58fc0a191a6c03b8a5974cf6e2bbc7bd98d4a40a3581482d7 -ab13decb9e2669e33a7049b8eca3ca327c40dea15ad6e0e7fa63ed506db1d258bc36ac88b35f65cae0984e937eb6575d -b5e748eb1a7a1e274ff0cc56311c198f2c076fe4b7e73e5f80396fe85358549df906584e6bb2c8195b3e2be7736850a5 -b5cb911325d8f963c41f691a60c37831c7d3bbd92736efa33d1f77a22b3fde7f283127256c2f47e197571e6fe0b46149 -8a01dc6ed1b55f26427a014faa347130738b191a06b800e32042a46c13f60b49534520214359d68eb2e170c31e2b8672 -a72fa874866e19b2efb8e069328362bf7921ec375e3bcd6b1619384c3f7ee980f6cf686f3544e9374ff54b4d17a1629c -8db21092f7c5f110fba63650b119e82f4b42a997095d65f08f8237b02dd66fdf959f788df2c35124db1dbd330a235671 -8c65d50433d9954fe28a09fa7ba91a70a590fe7ba6b3060f5e4be0f6cef860b9897fa935fb4ebc42133524eb071dd169 -b4614058e8fa21138fc5e4592623e78b8982ed72aa35ee4391b164f00c68d277fa9f9eba2eeefc890b4e86eba5124591 -ab2ad3a1bce2fbd55ca6b7c23786171fe1440a97d99d6df4d80d07dd56ac2d7203c294b32fc9e10a6c259381a73f24a1 -812ae3315fdc18774a8da3713a4679e8ed10b9405edc548c00cacbe25a587d32040566676f135e4723c5dc25df5a22e9 -a464b75f95d01e5655b54730334f443c8ff27c3cb79ec7af4b2f9da3c2039c609908cd128572e1fd0552eb597e8cef8d -a0db3172e93ca5138fe419e1c49a1925140999f6eff7c593e5681951ee0ec1c7e454c851782cbd2b8c9bc90d466e90e0 -806db23ba7d00b87d544eed926b3443f5f9c60da6b41b1c489fba8f73593b6e3b46ebfcab671ee009396cd77d5e68aa1 -8bfdf2c0044cc80260994e1c0374588b6653947b178e8b312be5c2a05e05767e98ea15077278506aee7df4fee1aaf89e -827f6558c16841b5592ff089c9c31e31eb03097623524394813a2e4093ad2d3f8f845504e2af92195aaa8a1679d8d692 -925c4f8eab2531135cd71a4ec88e7035b5eea34ba9d799c5898856080256b4a15ed1a746e002552e2a86c9c157e22e83 -a9f9a368f0e0b24d00a35b325964c85b69533013f9c2cfad9708be5fb87ff455210f8cb8d2ce3ba58ca3f27495552899 -8ac0d3bebc1cae534024187e7c71f8927ba8fcc6a1926cb61c2b6c8f26bb7831019e635a376146c29872a506784a4aaa -97c577be2cbbfdb37ad754fae9df2ada5fc5889869efc7e18a13f8e502fbf3f4067a509efbd46fd990ab47ce9a70f5a8 -935e7d82bca19f16614aa43b4a3474e4d20d064e4bfdf1cea2909e5c9ab72cfe3e54dc50030e41ee84f3588cebc524e9 -941aafc08f7c0d94cebfbb1f0aad5202c02e6e37f2c12614f57e727efa275f3926348f567107ee6d8914dd71e6060271 -af0fbc1ba05b4b5b63399686df3619968be5d40073de0313cbf5f913d3d4b518d4c249cdd2176468ccaa36040a484f58 -a0c414f23f46ca6d69ce74c6f8a00c036cb0edd098af0c1a7d39c802b52cfb2d5dbdf93fb0295453d4646e2af7954d45 -909cf39e11b3875bb63b39687ae1b5d1f5a15445e39bf164a0b14691b4ddb39a8e4363f584ef42213616abc4785b5d66 -a92bac085d1194fbd1c88299f07a061d0bdd3f980b663e81e6254dbb288bf11478c0ee880e28e01560f12c5ccb3c0103 -841705cd5cd76b943e2b7c5e845b9dd3c8defe8ef67e93078d6d5e67ade33ad4b0fd413bc196f93b0a4073c855cd97d4 -8e7eb8364f384a9161e81d3f1d52ceca9b65536ae49cc35b48c3e2236322ba4ae9973e0840802d9fa4f4d82ea833544f -aed3ab927548bc8bec31467ba80689c71a168e34f50dcb6892f19a33a099f5aa6b3f9cb79f5c0699e837b9a8c7f27efe -b8fbf7696210a36e20edabd77839f4dfdf50d6d015cdf81d587f90284a9bcef7d2a1ff520728d7cc69a4843d6c20dedd -a9d533769ce6830211c884ae50a82a7bf259b44ac71f9fb11f0296fdb3981e6b4c1753fe744647b247ebc433a5a61436 -8b4bdf90d33360b7f428c71cde0a49fb733badba8c726876945f58c620ce7768ae0e98fc8c31fa59d8955a4823336bb1 -808d42238e440e6571c59e52a35ae32547d502dc24fd1759d8ea70a7231a95859baf30b490a4ba55fa2f3aaa11204597 -85594701f1d2fee6dc1956bc44c7b31db93bdeec2f3a7d622c1a08b26994760773e3d57521a44cfd7e407ac3fd430429 -a66de045ce7173043a6825e9dc440ac957e2efb6df0a337f4f8003eb0c719d873a52e6eba3cb0d69d977ca37d9187674 -87a1c6a1fdff993fa51efa5c3ba034c079c0928a7d599b906336af7c2dcab9721ceaf3108c646490af9dff9a754f54b3 -926424223e462ceb75aed7c22ade8a7911a903b7e5dd4bc49746ddce8657f4616325cd12667d4393ac52cdd866396d0e -b5dc96106593b42b30f06f0b0a1e0c1aafc70432e31807252d3674f0b1ea5e58eac8424879d655c9488d85a879a3e572 -997ca0987735cc716507cb0124b1d266d218b40c9d8e0ecbf26a1d65719c82a637ce7e8be4b4815d307df717bde7c72a -92994d3f57a569b7760324bb5ae4e8e14e1633d175dab06aa57b8e391540e05f662fdc08b8830f489a063f59b689a688 -a8087fcc6aa4642cb998bea11facfe87eb33b90a9aa428ab86a4124ad032fc7d2e57795311a54ec9f55cc120ebe42df1 -a9bd7d1de6c0706052ca0b362e2e70e8c8f70f1f026ea189b4f87a08ce810297ebfe781cc8004430776c54c1a05ae90c -856d33282e8a8e33a3d237fb0a0cbabaf77ba9edf2fa35a831fdafcadf620561846aa6cbb6bdc5e681118e1245834165 -9524a7aa8e97a31a6958439c5f3339b19370f03e86b89b1d02d87e4887309dbbe9a3a8d2befd3b7ed5143c8da7e0a8ad -824fdf433e090f8acbd258ac7429b21f36f9f3b337c6d0b71d1416a5c88a767883e255b2888b7c906dd2e9560c4af24c -88c7fee662ca7844f42ed5527996b35723abffd0d22d4ca203b9452c639a5066031207a5ae763dbc0865b3299d19b1ec -919dca5c5595082c221d5ab3a5bc230f45da7f6dec4eb389371e142c1b9c6a2c919074842479c2844b72c0d806170c0c -b939be8175715e55a684578d8be3ceff3087f60fa875fff48e52a6e6e9979c955efef8ff67cfa2b79499ea23778e33b0 -873b6db725e7397d11bc9bed9ac4468e36619135be686790a79bc6ed4249058f1387c9a802ea86499f692cf635851066 -aeae06db3ec47e9e5647323fa02fac44e06e59b885ad8506bf71b184ab3895510c82f78b6b22a5d978e8218e7f761e9f -b99c0a8359c72ab88448bae45d4bf98797a26bca48b0d4460cd6cf65a4e8c3dd823970ac3eb774ae5d0cea4e7fadf33e -8f10c8ec41cdfb986a1647463076a533e6b0eec08520c1562401b36bb063ac972aa6b28a0b6ce717254e35940b900e3c -a106d9be199636d7add43b942290269351578500d8245d4aae4c083954e4f27f64740a3138a66230391f2d0e6043a8de -a469997908244578e8909ff57cffc070f1dbd86f0098df3cfeb46b7a085cfecc93dc69ee7cad90ff1dc5a34d50fe580c -a4ef087bea9c20eb0afc0ee4caba7a9d29dfa872137828c721391273e402fb6714afc80c40e98bbd8276d3836bffa080 -b07a013f73cd5b98dae0d0f9c1c0f35bff8a9f019975c4e1499e9bee736ca6fcd504f9bc32df1655ff333062382cff04 -b0a77188673e87cc83348c4cc5db1eecf6b5184e236220c8eeed7585e4b928db849944a76ec60ef7708ef6dac02d5592 -b1284b37e59b529f0084c0dacf0af6c0b91fc0f387bf649a8c74819debf606f7b07fc3e572500016fb145ec2b24e9f17 -97b20b5b4d6b9129da185adfbf0d3d0b0faeba5b9715f10299e48ea0521709a8296a9264ce77c275a59c012b50b6519a -b9d37e946fae5e4d65c1fbfacc8a62e445a1c9d0f882e60cca649125af303b3b23af53c81d7bac544fb7fcfc7a314665 -8e5acaac379f4bb0127efbef26180f91ff60e4c525bc9b798fc50dfaf4fe8a5aa84f18f3d3cfb8baead7d1e0499af753 -b0c0b8ab1235bf1cda43d4152e71efc1a06c548edb964eb4afceb201c8af24240bf8ab5cae30a08604e77432b0a5faf0 -8cc28d75d5c8d062d649cbc218e31c4d327e067e6dbd737ec0a35c91db44fbbd0d40ec424f5ed79814add16947417572 -95ae6219e9fd47efaa9cb088753df06bc101405ba50a179d7c9f7c85679e182d3033f35b00dbba71fdcd186cd775c52e -b5d28fa09f186ebc5aa37453c9b4d9474a7997b8ae92748ecb940c14868792292ac7d10ade01e2f8069242b308cf97e5 -8c922a0faa14cc6b7221f302df3342f38fc8521ec6c653f2587890192732c6da289777a6cd310747ea7b7d104af95995 -b9ad5f660b65230de54de535d4c0fcae5bc6b59db21dea5500fdc12eea4470fb8ea003690fdd16d052523418d5e01e8c -a39a9dd41a0ff78c82979483731f1cd68d3921c3e9965869662c22e02dde3877802e180ba93f06e7346f96d9fa9261d2 -8b32875977ec372c583b24234c27ed73aef00cdff61eb3c3776e073afbdeade548de9497c32ec6d703ff8ad0a5cb7fe4 -9644cbe755a5642fe9d26cfecf170d3164f1848c2c2e271d5b6574a01755f3980b3fc870b98cf8528fef6ecef4210c16 -81ea9d1fdd9dd66d60f40ce0712764b99da9448ae0b300f8324e1c52f154e472a086dda840cb2e0b9813dc8ce8afd4b5 -906aaa4a7a7cdf01909c5cfbc7ded2abc4b869213cbf7c922d4171a4f2e637e56f17020b852ad339d83b8ac92f111666 -939b5f11acbdeff998f2a080393033c9b9d8d5c70912ea651c53815c572d36ee822a98d6dfffb2e339f29201264f2cf4 -aba4898bf1ccea9b9e2df1ff19001e05891581659c1cbbde7ee76c349c7fc7857261d9785823c9463a8aea3f40e86b38 -83ca1a56b8a0be4820bdb5a9346357c68f9772e43f0b887729a50d2eb2a326bbcede676c8bf2e51d7c89bbd8fdb778a6 -94e86e9fe6addfe2c3ee3a547267ed921f4230d877a85bb4442c2d9350c2fa9a9c54e6fe662de82d1a2407e4ab1691c2 -a0cc3bdef671a59d77c6984338b023fa2b431b32e9ed2abe80484d73edc6540979d6f10812ecc06d4d0c5d4eaca7183c -b5343413c1b5776b55ea3c7cdd1f3af1f6bd802ea95effe3f2b91a523817719d2ecc3f8d5f3cc2623ace7e35f99ca967 -92085d1ed0ed28d8cabe3e7ff1905ed52c7ceb1eac5503760c52fb5ee3a726aba7c90b483c032acc3f166b083d7ec370 -8ec679520455275cd957fca8122724d287db5df7d29f1702a322879b127bff215e5b71d9c191901465d19c86c8d8d404 -b65eb2c63d8a30332eb24ee8a0c70156fc89325ebbb38bacac7cf3f8636ad8a472d81ccca80423772abc00192d886d8a -a9fe1c060b974bee4d590f2873b28635b61bfcf614e61ff88b1be3eee4320f4874e21e8d666d8ac8c9aba672efc6ecae -b3fe2a9a389c006a831dea7e777062df84b5c2803f9574d7fbe10b7e1c125817986af8b6454d6be9d931a5ac94cfe963 -95418ad13b734b6f0d33822d9912c4c49b558f68d08c1b34a0127fcfa666bcae8e6fda8832d2c75bb9170794a20e4d7c -a9a7df761e7f18b79494bf429572140c8c6e9d456c4d4e336184f3f51525a65eb9582bea1e601bdb6ef8150b7ca736a5 -a0de03b1e75edf7998c8c1ac69b4a1544a6fa675a1941950297917366682e5644a4bda9cdeedfaf9473d7fccd9080b0c -a61838af8d95c95edf32663a68f007d95167bf6e41b0c784a30b22d8300cfdd5703bd6d16e86396638f6db6ae7e42a85 -8866d62084d905c145ff2d41025299d8b702ac1814a7dec4e277412c161bc9a62fed735536789cb43c88693c6b423882 -91da22c378c81497fe363e7f695c0268443abee50f8a6625b8a41e865638a643f07b157ee566de09ba09846934b4e2d7 -941d21dd57c9496aa68f0c0c05507405fdd413acb59bc668ce7e92e1936c68ec4b065c3c30123319884149e88228f0b2 -a77af9b094bc26966ddf2bf9e1520c898194a5ccb694915950dadc204facbe3066d3d89f50972642d76b14884cfbaa21 -8e76162932346869f4618bde744647f7ab52ab498ad654bdf2a4feeb986ac6e51370841e5acbb589e38b6e7142bb3049 -b60979ace17d6937ece72e4f015da4657a443dd01cebc7143ef11c09e42d4aa8855999a65a79e2ea0067f31c9fc2ab0f -b3e2ffdd5ee6fd110b982fd4fad4b93d0fca65478f986d086eeccb0804960bfaa1919afa743c2239973ea65091fe57d2 -8ce0ce05e7d7160d44574011da687454dbd3c8b8290aa671731b066e2c82f8cf2d63cb8e932d78c6122ec610e44660e6 -ab005dd8d297045c39e2f72fb1c48edb501ccf3575d3d04b9817b3afee3f0bb0f3f53f64bda37d1d9cde545aae999bae -95bd7edb4c4cd60e3cb8a72558845a3cce6bb7032ccdf33d5a49ebb6ddf203bc3c79e7b7e550735d2d75b04c8b2441e8 -889953ee256206284094e4735dbbb17975bafc7c3cb94c9fbfee4c3e653857bfd49e818f64a47567f721b98411a3b454 -b188423e707640ab0e75a061e0b62830cde8afab8e1ad3dae30db69ffae4e2fc005bababbdcbd7213b918ed4f70e0c14 -a97e0fafe011abd70d4f99a0b36638b3d6e7354284588f17a88970ed48f348f88392779e9a038c6cbc9208d998485072 -87db11014a91cb9b63e8dfaa82cdebca98272d89eb445ee1e3ff9dbaf2b3fad1a03b888cffc128e4fe208ed0dddece0f -aad2e40364edd905d66ea4ac9d51f9640d6fda9a54957d26ba233809851529b32c85660fa401dbee3679ec54fa6dd966 -863e99336ca6edf03a5a259e59a2d0f308206e8a2fb320cfc0be06057366df8e0f94b33a28f574092736b3c5ada84270 -b34bcc56a057589f34939a1adc51de4ff6a9f4fee9c7fa9aa131e28d0cf0759a0c871b640162acdfbf91f3f1b59a3703 -935dd28f2896092995c5eff1618e5b6efe7a40178888d7826da9b0503c2d6e68a28e7fac1a334e166d0205f0695ef614 -b842cd5f8f5de5ca6c68cb4a5c1d7b451984930eb4cc18fd0934d52fdc9c3d2d451b1c395594d73bc3451432bfba653f -9014537885ce2debad736bc1926b25fdab9f69b216bf024f589c49dc7e6478c71d595c3647c9f65ff980b14f4bb2283b -8e827ccca1dd4cd21707140d10703177d722be0bbe5cac578db26f1ef8ad2909103af3c601a53795435b27bf95d0c9ed -8a0b8ad4d466c09d4f1e9167410dbe2edc6e0e6229d4b3036d30f85eb6a333a18b1c968f6ca6d6889bb08fecde017ef4 -9241ee66c0191b06266332dc9161dede384c4bb4e116dbd0890f3c3790ec5566da4568243665c4725b718ac0f6b5c179 -aeb4d5fad81d2b505d47958a08262b6f1b1de9373c2c9ba6362594194dea3e002ab03b8cbb43f867be83065d3d370f19 -8781bc83bb73f7760628629fe19e4714b494dbed444c4e4e4729b7f6a8d12ee347841a199888794c2234f51fa26fc2b9 -b58864f0acd1c2afa29367e637cbde1968d18589245d9936c9a489c6c495f54f0113ecdcbe4680ac085dd3c397c4d0c3 -94a24284afaeead61e70f3e30f87248d76e9726759445ca18cdb9360586c60cc9f0ec1c397f9675083e0b56459784e2e -aed358853f2b54dcbddf865e1816c2e89be12e940e1abfa661e2ee63ffc24a8c8096be2072fa83556482c0d89e975124 -b95374e6b4fc0765708e370bc881e271abf2e35c08b056a03b847e089831ef4fe3124b9c5849d9c276eb2e35b3daf264 -b834cdbcfb24c8f84bfa4c552e7fadc0028a140952fd69ed13a516e1314a4cd35d4b954a77d51a1b93e1f5d657d0315d -8fb6d09d23bfa90e7443753d45a918d91d75d8e12ec7d016c0dfe94e5c592ba6aaf483d2f16108d190822d955ad9cdc3 -aa315cd3c60247a6ad4b04f26c5404c2713b95972843e4b87b5a36a89f201667d70f0adf20757ebe1de1b29ae27dda50 -a116862dca409db8beff5b1ccd6301cdd0c92ca29a3d6d20eb8b87f25965f42699ca66974dd1a355200157476b998f3b -b4c2f5fe173c4dc8311b60d04a65ce1be87f070ac42e13cd19c6559a2931c6ee104859cc2520edebbc66a13dc7d30693 -8d4a02bf99b2260c334e7d81775c5cf582b00b0c982ce7745e5a90624919028278f5e9b098573bad5515ce7fa92a80c8 -8543493bf564ce6d97bd23be9bff1aba08bd5821ca834f311a26c9139c92a48f0c2d9dfe645afa95fec07d675d1fd53b -9344239d13fde08f98cb48f1f87d34cf6abe8faecd0b682955382a975e6eed64e863fa19043290c0736261622e00045c -aa49d0518f343005ca72b9e6c7dcaa97225ce6bb8b908ebbe7b1a22884ff8bfb090890364e325a0d414ad180b8f161d1 -907d7fd3e009355ab326847c4a2431f688627faa698c13c03ffdd476ecf988678407f029b8543a475dcb3dafdf2e7a9c -845f1f10c6c5dad2adc7935f5cd2e2b32f169a99091d4f1b05babe7317b9b1cdce29b5e62f947dc621b9acbfe517a258 -8f3be8e3b380ea6cdf9e9c237f5e88fd5a357e5ded80ea1fc2019810814de82501273b4da38916881125b6fa0cfd4459 -b9c7f487c089bf1d20c822e579628db91ed9c82d6ca652983aa16d98b4270c4da19757f216a71b9c13ddee3e6e43705f -8ba2d8c88ad2b872db104ea8ddbb006ec2f3749fd0e19298a804bb3a5d94de19285cc7fb19fee58a66f7851d1a66c39f -9375ecd3ed16786fe161af5d5c908f56eeb467a144d3bbddfc767e90065b7c94fc53431adebecba2b6c9b5821184d36e -a49e069bfadb1e2e8bff6a4286872e2a9765d62f0eaa4fcb0e5af4bbbed8be3510fb19849125a40a8a81d1e33e81c3eb -9522cc66757b386aa6b88619525c8ce47a5c346d590bb3647d12f991e6c65c3ab3c0cfc28f0726b6756c892eae1672be -a9a0f1f51ff877406fa83a807aeb17b92a283879f447b8a2159653db577848cc451cbadd01f70441e351e9ed433c18bc -8ff7533dcff6be8714df573e33f82cf8e9f2bcaaa43e939c4759d52b754e502717950de4b4252fb904560fc31dce94a4 -959724671e265a28d67c29d95210e97b894b360da55e4cf16e6682e7912491ed8ca14bfaa4dce9c25a25b16af580494f -92566730c3002f4046c737032487d0833c971e775de59fe02d9835c9858e2e3bc37f157424a69764596c625c482a2219 -a84b47ceff13ed9c3e5e9cdf6739a66d3e7c2bd8a6ba318fefb1a9aecf653bb2981da6733ddb33c4b0a4523acc429d23 -b4ddf571317e44f859386d6140828a42cf94994e2f1dcbcc9777f4eebbfc64fc1e160b49379acc27c4672b8e41835c5d -8ab95c94072b853d1603fdd0a43b30db617d13c1d1255b99075198e1947bfa5f59aed2b1147548a1b5e986cd9173d15c -89511f2eab33894fd4b3753d24249f410ff7263052c1fef6166fc63a79816656b0d24c529e45ccce6be28de6e375d916 -a0866160ca63d4f2be1b4ea050dac6b59db554e2ebb4e5b592859d8df339b46fd7cb89aaed0951c3ee540aee982c238a -8fcc5cbba1b94970f5ff2eb1922322f5b0aa7d918d4b380c9e7abfd57afd8b247c346bff7b87af82efbce3052511cd1b -99aeb2a5e846b0a2874cca02c66ed40d5569eb65ab2495bc3f964a092e91e1517941f2688e79f8cca49cd3674c4e06dc -b7a096dc3bad5ca49bee94efd884aa3ff5615cf3825cf95fbe0ce132e35f46581d6482fa82666c7ef5f1643eaee8f1ca -94393b1da6eaac2ffd186b7725eca582f1ddc8cdd916004657f8a564a7c588175cb443fc6943b39029f5bbe0add3fad8 -884b85fe012ccbcd849cb68c3ad832d83b3ef1c40c3954ffdc97f103b1ed582c801e1a41d9950f6bddc1d11f19d5ec76 -b00061c00131eded8305a7ce76362163deb33596569afb46fe499a7c9d7a0734c084d336b38d168024c2bb42b58e7660 -a439153ac8e6ca037381e3240e7ba08d056c83d7090f16ed538df25901835e09e27de2073646e7d7f3c65056af6e4ce7 -830fc9ca099097d1f38b90e6843dc86f702be9d20bdacc3e52cae659dc41df5b8d2c970effa6f83a5229b0244a86fe22 -b81ea2ffaaff2bb00dd59a9ab825ba5eed4db0d8ac9c8ed1a632ce8f086328a1cddd045fbe1ace289083c1325881b7e7 -b51ea03c58daf2db32c99b9c4789b183365168cb5019c72c4cc91ac30b5fb7311d3db76e6fa41b7cd4a8c81e2f6cdc94 -a4170b2c6d09ca5beb08318730419b6f19215ce6c631c854116f904be3bc30dd85a80c946a8ab054d3e307afaa3f8fbc -897cc42ff28971ff54d2a55dd6b35cfb8610ac902f3c06e3a5cea0e0a257e870c471236a8e84709211c742a09c5601a6 -a18f2e98d389dace36641621488664ecbb422088ab03b74e67009b8b8acacaaa24fdcf42093935f355207d934adc52a8 -92adcfb678cc2ba19c866f3f2b988fdcb4610567f3ab436cc0cb9acaf5a88414848d71133ebdbec1983e38e6190f1b5f -a86d43c2ce01b366330d3b36b3ca85f000c3548b8297e48478da1ee7d70d8576d4650cba7852ed125c0d7cb6109aa7f3 -8ed31ceed9445437d7732dce78a762d72ff32a7636bfb3fd7974b7ae15db414d8184a1766915244355deb354fbc5803b -9268f70032584f416e92225d65af9ea18c466ebc7ae30952d56a4e36fd9ea811dde0a126da9220ba3c596ec54d8a335e -9433b99ee94f2d3fbdd63b163a2bdf440379334c52308bd24537f7defd807145a062ff255a50d119a7f29f4b85d250e3 -90ce664f5e4628a02278f5cf5060d1a34f123854634b1870906e5723ac9afd044d48289be283b267d45fcbf3f4656aaf -aaf21c4d59378bb835d42ae5c5e5ab7a3c8c36a59e75997989313197752b79a472d866a23683b329ea69b048b87fa13e -b83c0589b304cec9ede549fde54f8a7c2a468c6657da8c02169a6351605261202610b2055c639b9ed2d5b8c401fb8f56 -9370f326ea0f170c2c05fe2c5a49189f20aec93b6b18a5572a818cd4c2a6adb359e68975557b349fb54f065d572f4c92 -ac3232fa5ce6f03fca238bef1ce902432a90b8afce1c85457a6bee5571c033d4bceefafc863af04d4e85ac72a4d94d51 -80d9ea168ff821b22c30e93e4c7960ce3ad3c1e6deeebedd342a36d01bd942419b187e2f382dbfd8caa34cca08d06a48 -a387a3c61676fb3381eefa2a45d82625635a666e999aba30e3b037ec9e040f414f9e1ad9652abd3bcad63f95d85038db -a1b229fe32121e0b391b0f6e0180670b9dc89d79f7337de4c77ea7ad0073e9593846f06797c20e923092a08263204416 -92164a9d841a2b828cedf2511213268b698520f8d1285852186644e9a0c97512cafa4bfbe29af892c929ebccd102e998 -82ee2fa56308a67c7db4fd7ef539b5a9f26a1c2cc36da8c3206ba4b08258fbb3cec6fe5cdbd111433fb1ba2a1e275927 -8c77bfe9e191f190a49d46f05600603fa42345592539b82923388d72392404e0b29a493a15e75e8b068dddcd444c2928 -80b927f93ccf79dcf5c5b20bcf5a7d91d7a17bc0401bb7cc9b53a6797feac31026eb114257621f5a64a52876e4474cc1 -b6b68b6501c37804d4833d5a063dd108a46310b1400549074e3cac84acc6d88f73948b7ad48d686de89c1ec043ae8c1a -ab3da00f9bdc13e3f77624f58a3a18fc3728956f84b5b549d62f1033ae4b300538e53896e2d943f160618e05af265117 -b6830e87233b8eace65327fdc764159645b75d2fd4024bf8f313b2dd5f45617d7ecfb4a0b53ccafb5429815a9a1adde6 -b9251cfe32a6dc0440615aadcd98b6b1b46e3f4e44324e8f5142912b597ee3526bea2431e2b0282bb58f71be5b63f65e -af8d70711e81cdddfb39e67a1b76643292652584c1ce7ce4feb1641431ad596e75c9120e85f1a341e7a4da920a9cdd94 -98cd4e996594e89495c078bfd52a4586b932c50a449a7c8dfdd16043ca4cda94dafbaa8ad1b44249c99bbcc52152506e -b9fc6d1c24f48404a4a64fbe3e43342738797905db46e4132aee5f086aaa4c704918ad508aaefa455cfe1b36572e6242 -a365e871d30ba9291cedaba1be7b04e968905d003e9e1af7e3b55c5eb048818ae5b913514fb08b24fb4fbdccbb35d0b8 -93bf99510971ea9af9f1e364f1234c898380677c8e8de9b0dd24432760164e46c787bc9ec42a7ad450500706cf247b2d -b872f825a5b6e7b9c7a9ddfeded3516f0b1449acc9b4fd29fc6eba162051c17416a31e5be6d3563f424d28e65bab8b8f -b06b780e5a5e8eb4f4c9dc040f749cf9709c8a4c9ef15e925f442b696e41e5095db0778a6c73bcd329b265f2c6955c8b -848f1a981f5fc6cd9180cdddb8d032ad32cdfa614fc750d690dbae36cc0cd355cbf1574af9b3ffc8b878f1b2fafb9544 -a03f48cbff3e9e8a3a655578051a5ae37567433093ac500ed0021c6250a51b767afac9bdb194ee1e3eac38a08c0eaf45 -b5be78ce638ff8c4aa84352b536628231d3f7558c5be3bf010b28feac3022e64691fa672f358c8b663904aebe24a54ed -a9d4da70ff676fa55d1728ba6ab03b471fa38b08854d99e985d88c2d050102d8ccffbe1c90249a5607fa7520b15fe791 -8fe9f7092ffb0b69862c8e972fb1ecf54308c96d41354ed0569638bb0364f1749838d6d32051fff1599112978c6e229c -ae6083e95f37770ecae0df1e010456f165d96cfe9a7278c85c15cffd61034081ce5723e25e2bede719dc9341ec8ed481 -a260891891103089a7afbd9081ea116cfd596fd1015f5b65e10b0961eb37fab7d09c69b7ce4be8bf35e4131848fb3fe4 -8d729fa32f6eb9fd2f6a140bef34e8299a2f3111bffd0fe463aa8622c9d98bfd31a1df3f3e87cd5abc52a595f96b970e -a30ec6047ae4bc7da4daa7f4c28c93aedb1112cfe240e681d07e1a183782c9ff6783ac077c155af23c69643b712a533f -ac830726544bfe7b5467339e5114c1a75f2a2a8d89453ce86115e6a789387e23551cd64620ead6283dfa4538eb313d86 -8445c135b7a48068d8ed3e011c6d818cfe462b445095e2fbf940301e50ded23f272d799eea47683fc027430ce14613ef -95785411715c9ae9d8293ce16a693a2aa83e3cb1b4aa9f76333d0da2bf00c55f65e21e42e50e6c5772ce213dd7b4f7a0 -b273b024fa18b7568c0d1c4d2f0c4e79ec509dafac8c5951f14192d63ddbcf2d8a7512c1c1b615cc38fa3e336618e0c5 -a78b9d3ea4b6a90572eb27956f411f1d105fdb577ee2ffeec9f221da9b45db84bfe866af1f29597220c75e0c37a628d8 -a4be2bf058c36699c41513c4d667681ce161a437c09d81383244fc55e1c44e8b1363439d0cce90a3e44581fb31d49493 -b6eef13040f17dd4eba22aaf284d2f988a4a0c4605db44b8d2f4bf9567ac794550b543cc513c5f3e2820242dd704152e -87eb00489071fa95d008c5244b88e317a3454652dcb1c441213aa16b28cd3ecaa9b22fec0bdd483c1df71c37119100b1 -92d388acdcb49793afca329cd06e645544d2269234e8b0b27d2818c809c21726bc9cf725651b951e358a63c83dedee24 -ae27e219277a73030da27ab5603c72c8bd81b6224b7e488d7193806a41343dff2456132274991a4722fdb0ef265d04cd -97583e08ecb82bbc27c0c8476d710389fa9ffbead5c43001bd36c1b018f29faa98de778644883e51870b69c5ffb558b5 -90a799a8ce73387599babf6b7da12767c0591cadd36c20a7990e7c05ea1aa2b9645654ec65308ee008816623a2757a6a -a1b47841a0a2b06efd9ab8c111309cc5fc9e1d5896b3e42ed531f6057e5ade8977c29831ce08dbda40348386b1dcc06d -b92b8ef59bbddb50c9457691bc023d63dfcc54e0fd88bd5d27a09e0d98ac290fc90e6a8f6b88492043bf7c87fac8f3e4 -a9d6240b07d62e22ec8ab9b1f6007c975a77b7320f02504fc7c468b4ee9cfcfd945456ff0128bc0ef2174d9e09333f8d -8e96534c94693226dc32bca79a595ca6de503af635f802e86442c67e77564829756961d9b701187fe91318da515bf0e6 -b6ba290623cd8dd5c2f50931c0045d1cfb0c30877bc8fe58cbc3ff61ee8da100045a39153916efa1936f4aee0892b473 -b43baa7717fac02d4294f5b3bb5e58a65b3557747e3188b482410388daac7a9c177f762d943fd5dcf871273921213da8 -b9cf00f8fb5e2ef2b836659fece15e735060b2ea39b8e901d3dcbdcf612be8bf82d013833718c04cd46ffaa70b85f42e -8017d0c57419e414cbba504368723e751ef990cc6f05dad7b3c2de6360adc774ad95512875ab8337d110bf39a42026fa -ae7401048b838c0dcd4b26bb6c56d79d51964a0daba780970b6c97daee4ea45854ea0ac0e4139b3fe60dac189f84df65 -887b237b0cd0f816b749b21db0b40072f9145f7896c36916296973f9e6990ede110f14e5976c906d08987c9836cca57f -a88c3d5770148aee59930561ca1223aceb2c832fb5417e188dca935905301fc4c6c2c9270bc1dff7add490a125eb81c6 -b6cf9b02c0cd91895ad209e38c54039523f137b5848b9d3ad33ae43af6c20c98434952db375fe378de7866f2d0e8b18a -84ef3d322ff580c8ad584b1fe4fe346c60866eb6a56e982ba2cf3b021ecb1fdb75ecc6c29747adda86d9264430b3f816 -a0561c27224baf0927ad144cb71e31e54a064c598373fcf0d66aebf98ab7af1d8e2f343f77baefff69a6da750a219e11 -aa5cc43f5b8162b016f5e1b61214c0c9d15b1078911c650b75e6cdfb49b85ee04c6739f5b1687d15908444f691f732de -ad4ac099b935589c7b8fdfdf3db332b7b82bb948e13a5beb121ebd7db81a87d278024a1434bcf0115c54ca5109585c3d -8a00466abf3f109a1dcd19e643b603d3af23d42794ef8ca2514dd507ecea44a031ac6dbc18bd02f99701168b25c1791e -b00b5900dfad79645f8bee4e5adc7b84eb22e5b1e67df77ccb505b7fc044a6c08a8ea5faca662414eb945f874f884cea -950e204e5f17112250b22ea6bb8423baf522fc0af494366f18fe0f949f51d6e6812074a80875cf1ed9c8e7420058d541 -91e5cbf8bb1a1d50c81608c9727b414d0dd2fb467ebc92f100882a3772e54f94979cfdf8e373fdef7c7fcdd60fec9e00 -a093f6a857b8caaff80599c2e89c962b415ecbaa70d8fd973155fa976a284c6b29a855f5f7a3521134d00d2972755188 -b4d55a3551b00da54cc010f80d99ddd2544bde9219a3173dfaadf3848edc7e4056ab532fb75ac26f5f7141e724267663 -a03ea050fc9b011d1b04041b5765d6f6453a93a1819cd9bd6328637d0b428f08526466912895dcc2e3008ee58822e9a7 -99b12b3665e473d01bc6985844f8994fb65cb15745024fb7af518398c4a37ff215da8f054e8fdf3286984ae36a73ca5e -9972c7e7a7fb12e15f78d55abcaf322c11249cd44a08f62c95288f34f66b51f146302bce750ff4d591707075d9123bd2 -a64b4a6d72354e596d87cda213c4fc2814009461570ccb27d455bbe131f8d948421a71925425b546d8cf63d5458cd64b -91c215c73b195795ede2228b7ed1f6e37892e0c6b0f4a0b5a16c57aa1100c84df9239054a173b6110d6c2b7f4bf1ce52 -88807198910ec1303480f76a3683870246a995e36adaeadc29c22f0bdba8152fe705bd070b75de657b04934f7d0ccf80 -b37c0026c7b32eb02cacac5b55cb5fe784b8e48b2945c64d3037af83ece556a117f0ff053a5968c2f5fa230e291c1238 -94c768384ce212bc2387e91ce8b45e4ff120987e42472888a317abc9dcdf3563b62e7a61c8e98d7cdcbe272167d91fc6 -a10c2564936e967a390cb14ef6e8f8b04ea9ece5214a38837eda09e79e0c7970b1f83adf017c10efd6faa8b7ffa2c567 -a5085eed3a95f9d4b1269182ea1e0d719b7809bf5009096557a0674bde4201b0ddc1f0f16a908fc468846b3721748ce3 -87468eb620b79a0a455a259a6b4dfbc297d0d53336537b771254dd956b145dc816b195b7002647ea218552e345818a3f -ace2b77ffb87366af0a9cb5d27d6fc4a14323dbbf1643f5f3c4559306330d86461bb008894054394cbfaefeaa0bc2745 -b27f56e840a54fbd793f0b7a7631aa4cee64b5947e4382b2dfb5eb1790270288884c2a19afebe5dc0c6ef335d4531c1c -876e438633931f7f895062ee16c4b9d10428875f7bc79a8e156a64d379a77a2c45bf5430c5ab94330f03da352f1e9006 -a2512a252587d200d2092b44c914df54e04ff8bcef36bf631f84bde0cf5a732e3dc7f00f662842cfd74b0b0f7f24180e -827f1bc8f54a35b7a4bd8154f79bcc055e45faed2e74adf7cf21cca95df44d96899e847bd70ead6bb27b9c0ed97bbd8b -a0c92cf5a9ed843714f3aea9fe7b880f622d0b4a3bf66de291d1b745279accf6ba35097849691370f41732ba64b5966b -a63f5c1e222775658421c487b1256b52626c6f79cb55a9b7deb2352622cedffb08502042d622eb3b02c97f9c09f9c957 -8cc093d52651e65fb390e186db6cc4de559176af4624d1c44cb9b0e836832419dacac7b8db0627b96288977b738d785d -aa7b6a17dfcec146134562d32a12f7bd7fe9522e300859202a02939e69dbd345ed7ff164a184296268f9984f9312e8fc -8ac76721f0d2b679f023d06cbd28c85ae5f4b43c614867ccee88651d4101d4fd352dbdb65bf36bfc3ebc0109e4b0c6f9 -8d350f7c05fc0dcd9a1170748846fb1f5d39453e4cb31e6d1457bed287d96fc393b2ecc53793ca729906a33e59c6834a -b9913510dfc5056d7ec5309f0b631d1ec53e3a776412ada9aefdaf033c90da9a49fdde6719e7c76340e86599b1f0eec2 -94955626bf4ce87612c5cfffcf73bf1c46a4c11a736602b9ba066328dc52ad6d51e6d4f53453d4ed55a51e0aad810271 -b0fcab384fd4016b2f1e53f1aafd160ae3b1a8865cd6c155d7073ecc1664e05b1d8bca1def39c158c7086c4e1103345e -827de3f03edfbde08570b72de6662c8bfa499b066a0a27ebad9b481c273097d17a5a0a67f01553da5392ec3f149b2a78 -ab7940384c25e9027c55c40df20bd2a0d479a165ced9b1046958353cd69015eeb1e44ed2fd64e407805ba42df10fc7bf -8ad456f6ff8cd58bd57567d931f923d0c99141978511b17e03cab7390a72b9f62498b2893e1b05c7c22dd274e9a31919 -ac75399e999effe564672db426faa17a839e57c5ef735985c70cd559a377adec23928382767b55ed5a52f7b11b54b756 -b17f975a00b817299ac7af5f2024ea820351805df58b43724393bfb3920a8cd747a3bbd4b8286e795521489db3657168 -a2bed800a6d95501674d9ee866e7314063407231491d794f8cf57d5be020452729c1c7cefd8c50dc1540181f5caab248 -9743f5473171271ffdd3cc59a3ae50545901a7b45cd4bc3570db487865f3b73c0595bebabbfe79268809ee1862e86e4a -b7eab77c2d4687b60d9d7b04e842b3880c7940140012583898d39fcc22d9b9b0a9be2c2e3788b3e6f30319b39c338f09 -8e2b8f797a436a1b661140e9569dcf3e1eea0a77c7ff2bc4ff0f3e49af04ed2de95e255df8765f1d0927fb456a9926b1 -8aefea201d4a1f4ff98ffce94e540bb313f2d4dfe7e9db484a41f13fc316ed02b282e1acc9bc6f56cad2dc2e393a44c9 -b950c17c0e5ca6607d182144aa7556bb0efe24c68f06d79d6413a973b493bfdf04fd147a4f1ab03033a32004cc3ea66f -b7b8dcbb179a07165f2dc6aa829fad09f582a71b05c3e3ea0396bf9e6fe73076f47035c031c2101e8e38e0d597eadd30 -a9d77ed89c77ec1bf8335d08d41c3c94dcca9fd1c54f22837b4e54506b212aa38d7440126c80648ab7723ff18e65ed72 -a819d6dfd4aef70e52b8402fe5d135f8082d40eb7d3bb5c4d7997395b621e2bb10682a1bad2c9caa33dd818550fc3ec6 -8f6ee34128fac8bbf13ce2d68b2bb363eb4fd65b297075f88e1446ddeac242500eeb4ef0735e105882ff5ba8c44c139b -b4440e48255c1644bcecf3a1e9958f1ec4901cb5b1122ee5b56ffd02cad1c29c4266999dbb85aa2605c1b125490074d4 -a43304a067bede5f347775d5811cf65a6380a8d552a652a0063580b5c5ef12a0867a39c7912fa219e184f4538eba1251 -a891ad67a790089ffc9f6d53e6a3d63d3556f5f693e0cd8a7d0131db06fd4520e719cfcc3934f0a8f62a95f90840f1d4 -aea6df8e9bb871081aa0fc5a9bafb00be7d54012c5baf653791907d5042a326aeee966fd9012a582cc16695f5baf7042 -8ffa2660dc52ed1cd4eff67d6a84a8404f358a5f713d04328922269bee1e75e9d49afeec0c8ad751620f22352a438e25 -87ec6108e2d63b06abed350f8b363b7489d642486f879a6c3aa90e5b0f335efc2ff2834eef9353951a42136f8e6a1b32 -865619436076c2760d9e87ddc905023c6de0a8d56eef12c98a98c87837f2ca3f27fd26a2ad752252dbcbe2b9f1d5a032 -980437dce55964293cb315c650c5586ffd97e7a944a83f6618af31c9d92c37b53ca7a21bb5bc557c151b9a9e217e7098 -95d128fc369df4ad8316b72aea0ca363cbc7b0620d6d7bb18f7076a8717a6a46956ff140948b0cc4f6d2ce33b5c10054 -8c7212d4a67b9ec70ebbca04358ad2d36494618d2859609163526d7b3acc2fc935ca98519380f55e6550f70a9bc76862 -893a2968819401bf355e85eee0f0ed0406a6d4a7d7f172d0017420f71e00bb0ba984f6020999a3cdf874d3cd8ebcd371 -9103c1af82dece25d87274e89ea0acd7e68c2921c4af3d8d7c82ab0ed9990a5811231b5b06113e7fa43a6bd492b4564f -99cfd87a94eab7d35466caa4ed7d7bb45e5c932b2ec094258fb14bf205659f83c209b83b2f2c9ccb175974b2a33e7746 -874b6b93e4ee61be3f00c32dd84c897ccd6855c4b6251eb0953b4023634490ed17753cd3223472873cbc6095b2945075 -84a32c0dc4ea60d33aac3e03e70d6d639cc9c4cc435c539eff915017be3b7bdaba33349562a87746291ebe9bc5671f24 -a7057b24208928ad67914e653f5ac1792c417f413d9176ba635502c3f9c688f7e2ee81800d7e3dc0a340c464da2fd9c5 -a03fb9ed8286aacfa69fbd5d953bec591c2ae4153400983d5dbb6cd9ea37fff46ca9e5cceb9d117f73e9992a6c055ad2 -863b2de04e89936c9a4a2b40380f42f20aefbae18d03750fd816c658aee9c4a03df7b12121f795c85d01f415baaeaa59 -8526eb9bd31790fe8292360d7a4c3eed23be23dd6b8b8f01d2309dbfdc0cfd33ad1568ddd7f8a610f3f85a9dfafc6a92 -b46ab8c5091a493d6d4d60490c40aa27950574a338ea5bbc045be3a114af87bdcb160a8c80435a9b7ad815f3cb56a3f3 -aeadc47b41a8d8b4176629557646202f868b1d728b2dda58a347d937e7ffc8303f20d26d6c00b34c851b8aeec547885d -aebb19fc424d72c1f1822aa7adc744cd0ef7e55727186f8df8771c784925058c248406ebeeaf3c1a9ee005a26e9a10c6 -8ff96e81c1a4a2ab1b4476c21018fae0a67e92129ee36120cae8699f2d7e57e891f5c624902cb1b845b944926a605cc3 -8251b8d2c43fadcaa049a9e7aff838dae4fb32884018d58d46403ac5f3beb5c518bfd45f03b8abb710369186075eb71c -a8b2a64f865f51a5e5e86a66455c093407933d9d255d6b61e1fd81ffafc9538d73caaf342338a66ba8ee166372a3d105 -aad915f31c6ba7fdc04e2aaac62e84ef434b7ee76a325f07dc430d12c84081999720181067b87d792efd0117d7ee1eab -a13db3bb60389883fd41d565c54fb5180d9c47ce2fe7a169ae96e01d17495f7f4fa928d7e556e7c74319c4c25d653eb2 -a4491b0198459b3f552855d680a59214eb74e6a4d6c5fa3b309887dc50ebea2ecf6d26c040550f7dc478b452481466fb -8f017f13d4b1e3f0c087843582b52d5f8d13240912254d826dd11f8703a99a2f3166dfbdfdffd9a3492979d77524276b -96c3d5dcd032660d50d7cd9db2914f117240a63439966162b10c8f1f3cf74bc83b0f15451a43b31dbd85e4a7ce0e4bb1 -b479ec4bb79573d32e0ec93b92bdd7ec8c26ddb5a2d3865e7d4209d119fd3499eaac527615ffac78c440e60ef3867ae0 -b2c49c4a33aa94b52b6410b599e81ff15490aafa7e43c8031c865a84e4676354a9c81eb4e7b8be6825fdcefd1e317d44 -906dc51d6a90c089b6704b47592805578a6eed106608eeb276832f127e1b8e858b72e448edcbefb497d152447e0e68ff -b0e81c63b764d7dfbe3f3fddc9905aef50f3633e5d6a4af6b340495124abedcff5700dfd1577bbbed7b6bf97d02719cb -9304c64701e3b4ed6d146e48a881f7d83a17f58357cca0c073b2bb593afd2d94f6e2a7a1ec511d0a67ad6ff4c3be5937 -b6fdbd12ba05aa598d80b83f70a15ef90e5cba7e6e75fa038540ee741b644cd1f408a6cecfd2a891ef8d902de586c6b5 -b80557871a6521b1b3c74a1ba083ae055b575df607f1f7b04c867ba8c8c181ea68f8d90be6031f4d25002cca27c44da2 -aa7285b8e9712e06b091f64163f1266926a36607f9d624af9996856ed2aaf03a580cb22ce407d1ade436c28b44ca173f -8148d72b975238b51e6ea389e5486940d22641b48637d7dfadfa603a605bfc6d74a016480023945d0b85935e396aea5d -8a014933a6aea2684b5762af43dcf4bdbb633cd0428d42d71167a2b6fc563ece5e618bff22f1db2ddb69b845b9a2db19 -990d91740041db770d0e0eb9d9d97d826f09fd354b91c41e0716c29f8420e0e8aac0d575231efba12fe831091ec38d5a -9454d0d32e7e308ddec57cf2522fb1b67a2706e33fb3895e9e1f18284129ab4f4c0b7e51af25681d248d7832c05eb698 -a5bd434e75bac105cb3e329665a35bce6a12f71dd90c15165777d64d4c13a82bceedb9b48e762bd24034e0fc9fbe45f4 -b09e3b95e41800d4dc29c6ffdaab2cd611a0050347f6414f154a47ee20ee59bf8cf7181454169d479ebce1eb5c777c46 -b193e341d6a047d15eea33766d656d807b89393665a783a316e9ba10518e5515c8e0ade3d6e15641d917a8a172a5a635 -ade435ec0671b3621dde69e07ead596014f6e1daa1152707a8c18877a8b067bde2895dd47444ffa69db2bbef1f1d8816 -a7fd3d6d87522dfc56fb47aef9ce781a1597c56a8bbfd796baba907afdc872f753d732bfda1d3402aee6c4e0c189f52d -a298cb4f4218d0464b2fab393e512bbc477c3225aa449743299b2c3572f065bc3a42d07e29546167ed9e1b6b3b3a3af3 -a9ee57540e1fd9c27f4f0430d194b91401d0c642456c18527127d1f95e2dba41c2c86d1990432eb38a692fda058fafde -81d6c1a5f93c04e6d8e5a7e0678c1fc89a1c47a5c920bcd36180125c49fcf7c114866b90e90a165823560b19898a7c16 -a4b7a1ec9e93c899b9fd9aaf264c50e42c36c0788d68296a471f7a3447af4dbc81e4fa96070139941564083ec5b5b5a1 -b3364e327d381f46940c0e11e29f9d994efc6978bf37a32586636c0070b03e4e23d00650c1440f448809e1018ef9f6d8 -8056e0913a60155348300e3a62e28b5e30629a90f7dd4fe11289097076708110a1d70f7855601782a3cdc5bdb1ca9626 -b4980fd3ea17bac0ba9ee1c470b17e575bb52e83ebdd7d40c93f4f87bebeaff1c8a679f9d3d09d635f068d37d5bd28bd -905a9299e7e1853648e398901dfcd437aa575c826551f83520df62984f5679cb5f0ea86aa45ed3e18b67ddc0dfafe809 -ab99553bf31a84f2e0264eb34a08e13d8d15e2484aa9352354becf9a15999c76cc568d68274b70a65e49703fc23540d0 -a43681597bc574d2dae8964c9a8dc1a07613d7a1272bdcb818d98c85d44e16d744250c33f3b5e4d552d97396b55e601f -a54e5a31716fccb50245898c99865644405b8dc920ded7a11f3d19bdc255996054b268e16f2e40273f11480e7145f41e -8134f3ad5ef2ad4ba12a8a4e4d8508d91394d2bcdc38b7c8c8c0b0a820357ac9f79d286c65220f471eb1adca1d98fc68 -94e2f755e60471578ab2c1adb9e9cea28d4eec9b0e92e0140770bca7002c365fcabfe1e5fb4fe6cfe79a0413712aa3ef -ad48f8d0ce7eb3cc6e2a3086ad96f562e5bed98a360721492ae2e74dc158586e77ec8c35d5fd5927376301b7741bad2b -8614f0630bdd7fbad3a31f55afd9789f1c605dc85e7dc67e2edfd77f5105f878bb79beded6e9f0b109e38ea7da67e8d5 -9804c284c4c5e77dabb73f655b12181534ca877c3e1e134aa3f47c23b7ec92277db34d2b0a5d38d2b69e5d1c3008a3e3 -a51b99c3088e473afdaa9e0a9f7e75a373530d3b04e44e1148da0726b95e9f5f0c7e571b2da000310817c36f84b19f7f -ac4ff909933b3b76c726b0a382157cdc74ab851a1ac6cef76953c6444441804cc43abb883363f416592e8f6cfbc4550b -ae7d915eb9fc928b65a29d6edbc75682d08584d0014f7bcf17d59118421ae07d26a02137d1e4de6938bcd1ab8ef48fad -852f7e453b1af89b754df6d11a40d5d41ea057376e8ecacd705aacd2f917457f4a093d6b9a8801837fa0f62986ad7149 -92c6bf5ada5d0c3d4dd8058483de36c215fa98edab9d75242f3eff9db07c734ad67337da6f0eefe23a487bf75a600dee -a2b42c09d0db615853763552a48d2e704542bbd786aae016eb58acbf6c0226c844f5fb31e428cb6450b9db855f8f2a6f -880cc07968266dbfdcfbc21815cd69e0eddfee239167ac693fb0413912d816f2578a74f7716eecd6deefa68c6eccd394 -b885b3ace736cd373e8098bf75ba66fa1c6943ca1bc4408cd98ac7074775c4478594f91154b8a743d9c697e1b29f5840 -a51ce78de512bd87bfa0835de819941dffbf18bec23221b61d8096fc9436af64e0693c335b54e7bfc763f287bdca2db6 -a3c76166a3bdb9b06ef696e57603b58871bc72883ee9d45171a30fe6e1d50e30bc9c51b4a0f5a7270e19a77b89733850 -acefc5c6f8a1e7c24d7b41e0fc7f6f3dc0ede6cf3115ffb9a6e54b1d954cbca9bda8ad7a084be9be245a1b8e9770d141 -b420ed079941842510e31cfad117fa11fb6b4f97dfbc6298cb840f27ebaceba23eeaf3f513bcffbf5e4aae946310182d -95c3bb5ef26c5ed2f035aa5d389c6b3c15a6705b9818a3fefaed28922158b35642b2e8e5a1a620fdad07e75ad4b43af4 -825149f9081ecf07a2a4e3e8b5d21bade86c1a882475d51c55ee909330b70c5a2ac63771c8600c6f38df716af61a3ea1 -873b935aae16d9f08adbc25353cee18af2f1b8d5f26dec6538d6bbddc515f2217ed7d235dcfea59ae61b428798b28637 -9294150843a2bedcedb3bb74c43eb28e759cf9499582c5430bccefb574a8ddd4f11f9929257ff4c153990f9970a2558f -b619563a811cc531da07f4f04e5c4c6423010ff9f8ed7e6ec9449162e3d501b269fb1c564c09c0429431879b0f45df02 -91b509b87eb09f007d839627514658c7341bc76d468920fe8a740a8cb96a7e7e631e0ea584a7e3dc1172266f641d0f5c -8b8aceace9a7b9b4317f1f01308c3904d7663856946afbcea141a1c615e21ccad06b71217413e832166e9dd915fbe098 -87b3b36e725833ea0b0f54753c3728c0dbc87c52d44d705ffc709f2d2394414c652d3283bab28dcce09799504996cee0 -b2670aad5691cbf308e4a6a77a075c4422e6cbe86fdba24e9f84a313e90b0696afb6a067eebb42ba2d10340d6a2f6e51 -876784a9aff3d54faa89b2bacd3ff5862f70195d0b2edc58e8d1068b3c9074c0da1cfa23671fe12f35e33b8a329c0ccd -8b48b9e758e8a8eae182f5cbec96f67d20cca6d3eee80a2d09208eb1d5d872e09ef23d0df8ebbb9b01c7449d0e3e3650 -b79303453100654c04a487bdcadc9e3578bc80930c489a7069a52e8ca1dba36c492c8c899ce025f8364599899baa287d -961b35a6111da54ece6494f24dacd5ea46181f55775b5f03df0e370c34a5046ac2b4082925855325bb42bc2a2c98381d -a31feb1be3f5a0247a1f7d487987eb622e34fca817832904c6ee3ee60277e5847945a6f6ea1ac24542c72e47bdf647df -a12a2aa3e7327e457e1aae30e9612715dd2cfed32892c1cd6dcda4e9a18203af8a44afb46d03b2eed89f6b9c5a2c0c23 -a08265a838e69a2ca2f80fead6ccf16f6366415b920c0b22ee359bcd8d4464ecf156f400a16a7918d52e6d733dd64211 -b723d6344e938d801cca1a00032af200e541d4471fd6cbd38fb9130daa83f6a1dffbbe7e67fc20f9577f884acd7594b2 -a6733d83ec78ba98e72ddd1e7ff79b7adb0e559e256760d0c590a986e742445e8cdf560d44b29439c26d87edd0b07c8c -a61c2c27d3f7b9ff4695a17afedf63818d4bfba390507e1f4d0d806ce8778d9418784430ce3d4199fd3bdbc2504d2af3 -8332f3b63a6dc985376e8b1b25eeae68be6160fbe40053ba7bcf6f073204f682da72321786e422d3482fd60c9e5aa034 -a280f44877583fbb6b860d500b1a3f572e3ee833ec8f06476b3d8002058e25964062feaa1e5bec1536d734a5cfa09145 -a4026a52d277fcea512440d2204f53047718ebfcae7b48ac57ea7f6bfbc5de9d7304db9a9a6cbb273612281049ddaec5 -95cdf69c831ab2fad6c2535ede9c07e663d2ddccc936b64e0843d2df2a7b1c31f1759c3c20f1e7a57b1c8f0dbb21b540 -95c96cec88806469c277ab567863c5209027cecc06c7012358e5f555689c0d9a5ffb219a464f086b45817e8536b86d2f -afe38d4684132a0f03d806a4c8df556bf589b25271fbc6fe2e1ed16de7962b341c5003755da758d0959d2e6499b06c68 -a9b77784fda64987f97c3a23c5e8f61b918be0f7c59ba285084116d60465c4a2aaafc8857eb16823282cc83143eb9126 -a830f05881ad3ce532a55685877f529d32a5dbe56cea57ffad52c4128ee0fad0eeaf0da4362b55075e77eda7babe70e5 -992b3ad190d6578033c13ed5abfee4ef49cbc492babb90061e3c51ee4b5790cdd4c8fc1abff1fa2c00183b6b64f0bbbe -b1015424d9364aeff75de191652dc66484fdbec3e98199a9eb9671ec57bec6a13ff4b38446e28e4d8aedb58dd619cd90 -a745304604075d60c9db36cada4063ac7558e7ec2835d7da8485e58d8422e817457b8da069f56511b02601289fbb8981 -a5ba4330bc5cb3dbe0486ddf995632a7260a46180a08f42ae51a2e47778142132463cc9f10021a9ad36986108fefa1a9 -b419e9fd4babcaf8180d5479db188bb3da232ae77a1c4ed65687c306e6262f8083070a9ac32220cddb3af2ec73114092 -a49e23dc5f3468f3bf3a0bb7e4a114a788b951ff6f23a3396ae9e12cbff0abd1240878a3d1892105413dbc38818e807c -b7ecc7b4831f650202987e85b86bc0053f40d983f252e9832ef503aea81c51221ce93279da4aa7466c026b2d2070e55d -96a8c35cb87f84fa84dcd6399cc2a0fd79cc9158ef4bdde4bae31a129616c8a9f2576cd19baa3f497ca34060979aed7d -8681b2c00aa62c2b519f664a95dcb8faef601a3b961bb4ce5d85a75030f40965e2983871d41ea394aee934e859581548 -85c229a07efa54a713d0790963a392400f55fbb1a43995a535dc6c929f20d6a65cf4efb434e0ad1cb61f689b8011a3bc -90856f7f3444e5ad44651c28e24cc085a5db4d2ffe79aa53228c26718cf53a6e44615f3c5cda5aa752d5f762c4623c66 -978999b7d8aa3f28a04076f74d11c41ef9c89fdfe514936c4238e0f13c38ec97e51a5c078ebc6409e517bfe7ccb42630 -a099914dd7ed934d8e0d363a648e9038eb7c1ec03fa04dbcaa40f7721c618c3ef947afef7a16b4d7ac8c12aa46637f03 -ab2a104fed3c83d16f2cda06878fa5f30c8c9411de71bfb67fd2fc9aa454dcbcf3d299d72f8cc12e919466a50fcf7426 -a4471d111db4418f56915689482f6144efc4664cfb0311727f36c864648d35734351becc48875df96f4abd3cfcf820f9 -83be11727cd30ea94ccc8fa31b09b81c9d6a9a5d3a4686af9da99587332fe78c1f94282f9755854bafd6033549afec91 -88020ff971dc1a01a9e993cd50a5d2131ffdcbb990c1a6aaa54b20d8f23f9546a70918ea57a21530dcc440c1509c24ad -ae24547623465e87905eaffa1fa5d52bb7c453a8dbd89614fa8819a2abcedaf455c2345099b7324ae36eb0ad7c8ef977 -b59b0c60997de1ee00b7c388bc7101d136c9803bf5437b1d589ba57c213f4f835a3e4125b54738e78abbc21b000f2016 -a584c434dfe194546526691b68fa968c831c31da42303a1d735d960901c74011d522246f37f299555416b8cf25c5a548 -80408ce3724f4837d4d52376d255e10f69eb8558399ae5ca6c11b78b98fe67d4b93157d2b9b639f1b5b64198bfe87713 -abb941e8d406c2606e0ddc35c113604fdd9d249eacc51cb64e2991e551b8639ce44d288cc92afa7a1e7fc599cfc84b22 -b223173f560cacb1c21dba0f1713839e348ad02cbfdef0626748604c86f89e0f4c919ed40b583343795bdd519ba952c8 -af1c70512ec3a19d98b8a1fc3ff7f7f5048a27d17d438d43f561974bbdd116fcd5d5c21040f3447af3f0266848d47a15 -8a44809568ebe50405bede19b4d2607199159b26a1b33e03d180e6840c5cf59d991a4fb150d111443235d75ecad085b7 -b06207cdca46b125a27b3221b5b50cf27af4c527dd7c80e2dbcebbb09778a96df3af67e50f07725239ce3583dad60660 -993352d9278814ec89b26a11c4a7c4941bf8f0e6781ae79559d14749ee5def672259792db4587f85f0100c7bb812f933 -9180b8a718b971fd27bc82c8582d19c4b4f012453e8c0ffeeeffe745581fc6c07875ab28be3af3fa3896d19f0c89ac5b -8b8e1263eb48d0fe304032dd5ea1f30e73f0121265f7458ba9054d3626894e8a5fef665340abd2ede9653045c2665938 -99a2beee4a10b7941c24b2092192faf52b819afd033e4a2de050fd6c7f56d364d0cf5f99764c3357cf32399e60fc5d74 -946a4aad7f8647ea60bee2c5fcdeb6f9a58fb2cfca70c4d10e458027a04846e13798c66506151be3df9454b1e417893f -a672a88847652d260b5472d6908d1d57e200f1e492d30dd1cecc441cdfc9b76e016d9bab560efd4d7f3c30801de884a9 -9414e1959c156cde1eb24e628395744db75fc24b9df4595350aaad0bc38e0246c9b4148f6443ef68b8e253a4a6bcf11c -9316e9e4ec5fab4f80d6540df0e3a4774db52f1d759d2e5b5bcd3d7b53597bb007eb1887cb7dc61f62497d51ffc8d996 -902d6d77bb49492c7a00bc4b70277bc28c8bf9888f4307bb017ac75a962decdedf3a4e2cf6c1ea9f9ba551f4610cbbd7 -b07025a18b0e32dd5e12ec6a85781aa3554329ea12c4cd0d3b2c22e43d777ef6f89876dd90a9c8fb097ddf61cf18adc5 -b355a849ad3227caa4476759137e813505ec523cbc2d4105bc7148a4630f9e81918d110479a2d5f5e4cd9ccec9d9d3e3 -b49532cfdf02ee760109881ad030b89c48ee3bb7f219ccafc13c93aead754d29bdafe345be54c482e9d5672bd4505080 -9477802410e263e4f938d57fa8f2a6cac7754c5d38505b73ee35ea3f057aad958cb9722ba6b7b3cfc4524e9ca93f9cdc -9148ea83b4436339580f3dbc9ba51509e9ab13c03063587a57e125432dd0915f5d2a8f456a68f8fff57d5f08c8f34d6e -b00b6b5392b1930b54352c02b1b3b4f6186d20bf21698689bbfc7d13e86538a4397b90e9d5c93fd2054640c4dbe52a4f -926a9702500441243cd446e7cbf15dde16400259726794694b1d9a40263a9fc9e12f7bcbf12a27cb9aaba9e2d5848ddc -a0c6155f42686cbe7684a1dc327100962e13bafcf3db97971fc116d9f5c0c8355377e3d70979cdbd58fd3ea52440901c -a277f899f99edb8791889d0817ea6a96c24a61acfda3ad8c3379e7c62b9d4facc4b965020b588651672fd261a77f1bfc -8f528cebb866b501f91afa50e995234bef5bf20bff13005de99cb51eaac7b4f0bf38580cfd0470de40f577ead5d9ba0f -963fc03a44e9d502cc1d23250efef44d299befd03b898d07ce63ca607bb474b5cf7c965a7b9b0f32198b04a8393821f7 -ab087438d0a51078c378bf4a93bd48ef933ff0f1fa68d02d4460820df564e6642a663b5e50a5fe509527d55cb510ae04 -b0592e1f2c54746bb076be0fa480e1c4bebc4225e1236bcda3b299aa3853e3afb401233bdbcfc4a007b0523a720fbf62 -851613517966de76c1c55a94dc4595f299398a9808f2d2f0a84330ba657ab1f357701d0895f658c18a44cb00547f6f57 -a2fe9a1dd251e72b0fe4db27be508bb55208f8f1616b13d8be288363ec722826b1a1fd729fc561c3369bf13950bf1fd6 -b896cb2bc2d0c77739853bc59b0f89b2e008ba1f701c9cbe3bef035f499e1baee8f0ff1e794854a48c320586a2dfc81a -a1b60f98e5e5106785a9b81a85423452ee9ef980fa7fa8464f4366e73f89c50435a0c37b2906052b8e58e212ebd366cf -a853b0ebd9609656636df2e6acd5d8839c0fda56f7bf9288a943b06f0b67901a32b95e016ca8bc99bd7b5eab31347e72 -b290fa4c1346963bd5225235e6bdf7c542174dab4c908ab483d1745b9b3a6015525e398e1761c90e4b49968d05e30eea -b0f65a33ad18f154f1351f07879a183ad62e5144ad9f3241c2d06533dad09cbb2253949daff1bb02d24d16a3569f7ef0 -a00db59b8d4218faf5aeafcd39231027324408f208ec1f54d55a1c41228b463b88304d909d16b718cfc784213917b71e -b8d695dd33dc2c3bc73d98248c535b2770ad7fa31aa726f0aa4b3299efb0295ba9b4a51c71d314a4a1bd5872307534d1 -b848057cca2ca837ee49c42b88422303e58ea7d2fc76535260eb5bd609255e430514e927cc188324faa8e657396d63ec -92677836061364685c2aaf0313fa32322746074ed5666fd5f142a7e8f87135f45cd10e78a17557a4067a51dfde890371 -a854b22c9056a3a24ab164a53e5c5cf388616c33e67d8ebb4590cb16b2e7d88b54b1393c93760d154208b5ca822dc68f -86fff174920388bfab841118fb076b2b0cdec3fdb6c3d9a476262f82689fb0ed3f1897f7be9dbf0932bb14d346815c63 -99661cf4c94a74e182752bcc4b98a8c2218a8f2765642025048e12e88ba776f14f7be73a2d79bd21a61def757f47f904 -8a8893144d771dca28760cba0f950a5d634195fd401ec8cf1145146286caffb0b1a6ba0c4c1828d0a5480ce49073c64c -938a59ae761359ee2688571e7b7d54692848eb5dde57ffc572b473001ea199786886f8c6346a226209484afb61d2e526 -923f68a6aa6616714cf077cf548aeb845bfdd78f2f6851d8148cba9e33a374017f2f3da186c39b82d14785a093313222 -ac923a93d7da7013e73ce8b4a2b14b8fd0cc93dc29d5de941a70285bdd19be4740fedfe0c56b046689252a3696e9c5bc -b49b32c76d4ec1a2c68d4989285a920a805993bc6fcce6dacd3d2ddae73373050a5c44ba8422a3781050682fa0ef6ba2 -8a367941c07c3bdca5712524a1411bad7945c7c48ffc7103b1d4dff2c25751b0624219d1ccde8c3f70c465f954be5445 -b838f029df455efb6c530d0e370bbbf7d87d61a9aea3d2fe5474c5fe0a39cf235ceecf9693c5c6c5820b1ba8f820bd31 -a8983b7c715eaac7f13a001d2abc462dfc1559dab4a6b554119c271aa8fe00ffcf6b6949a1121f324d6d26cb877bcbae -a2afb24ad95a6f14a6796315fbe0d8d7700d08f0cfaf7a2abe841f5f18d4fecf094406cbd54da7232a159f9c5b6e805e -87e8e95ad2d62f947b2766ff405a23f7a8afba14e7f718a691d95369c79955cdebe24c54662553c60a3f55e6322c0f6f -87c2cbcecb754e0cc96128e707e5c5005c9de07ffd899efa3437cadc23362f5a1d3fcdd30a1f5bdc72af3fb594398c2a -91afd6ee04f0496dc633db88b9370d41c428b04fd991002502da2e9a0ef051bcd7b760e860829a44fbe5539fa65f8525 -8c50e5d1a24515a9dd624fe08b12223a75ca55196f769f24748686315329b337efadca1c63f88bee0ac292dd0a587440 -8a07e8f912a38d94309f317c32068e87f68f51bdfa082d96026f5f5f8a2211621f8a3856dda8069386bf15fb2d28c18f -94ad1dbe341c44eeaf4dc133eed47d8dbfe752575e836c075745770a6679ff1f0e7883b6aa917462993a7f469d74cab5 -8745f8bd86c2bb30efa7efb7725489f2654f3e1ac4ea95bd7ad0f3cfa223055d06c187a16192d9d7bdaea7b050c6a324 -900d149c8d79418cda5955974c450a70845e02e5a4ecbcc584a3ca64d237df73987c303e3eeb79da1af83bf62d9e579f -8f652ab565f677fb1a7ba03b08004e3cda06b86c6f1b0b9ab932e0834acf1370abb2914c15b0d08327b5504e5990681c -9103097d088be1f75ab9d3da879106c2f597e2cc91ec31e73430647bdd5c33bcfd771530d5521e7e14df6acda44f38a6 -b0fec7791cfb0f96e60601e1aeced9a92446b61fedab832539d1d1037558612d78419efa87ff5f6b7aab8fd697d4d9de -b9d2945bdb188b98958854ba287eb0480ef614199c4235ce5f15fc670b8c5ffe8eeb120c09c53ea8a543a022e6a321ac -a9461bb7d5490973ebaa51afc0bb4a5e42acdccb80e2f939e88b77ac28a98870e103e1042899750f8667a8cc9123bae9 -a37fdf11d4bcb2aed74b9f460a30aa34afea93386fa4cdb690f0a71bc58f0b8df60bec56e7a24f225978b862626fa00e -a214420e183e03d531cf91661466ea2187d84b6e814b8b20b3730a9400a7d25cf23181bb85589ebc982cec414f5c2923 -ad09a45a698a6beb3e0915f540ef16e9af7087f53328972532d6b5dfe98ce4020555ece65c6cbad8bd6be8a4dfefe6fd -ab6742800b02728c92d806976764cb027413d6f86edd08ad8bb5922a2969ee9836878cd39db70db0bd9a2646862acc4f -974ca9305bd5ea1dc1755dff3b63e8bfe9f744321046c1395659bcea2a987b528e64d5aa96ac7b015650b2253b37888d -84eee9d6bce039c52c2ebc4fccc0ad70e20c82f47c558098da4be2f386a493cbc76adc795b5488c8d11b6518c2c4fab8 -875d7bda46efcb63944e1ccf760a20144df3b00d53282b781e95f12bfc8f8316dfe6492c2efbf796f1150e36e436e9df -b68a2208e0c587b5c31b5f6cb32d3e6058a9642e2d9855da4f85566e1412db528475892060bb932c55b3a80877ad7b4a -ba006368ecab5febb6ab348644d9b63de202293085ed468df8bc24d992ae8ce468470aa37f36a73630c789fb9c819b30 -90a196035150846cd2b482c7b17027471372a8ce7d914c4d82b6ea7fa705d8ed5817bd42d63886242585baf7d1397a1c -a223b4c85e0daa8434b015fd9170b5561fe676664b67064974a1e9325066ecf88fc81f97ab5011c59fad28cedd04b240 -82e8ec43139cf15c6bbeed484b62e06cded8a39b5ce0389e4cbe9c9e9c02f2f0275d8d8d4e8dfec8f69a191bef220408 -81a3fc07a7b68d92c6ee4b6d28f5653ee9ec85f7e2ee1c51c075c1b130a8c5097dc661cf10c5aff1c7114b1a6a19f11a -8ed2ef8331546d98819a5dd0e6c9f8cb2630d0847671314a28f277faf68da080b53891dd75c82cbcf7788b255490785d -acecabf84a6f9bbed6b2fc2e7e4b48f02ef2f15e597538a73aea8f98addc6badda15e4695a67ecdb505c1554e8f345ec -b8f51019b2aa575f8476e03dcadf86cc8391f007e5f922c2a36b2daa63f5a503646a468990cd5c65148d323942193051 -aaa595a84b403ec65729bc1c8055a94f874bf9adddc6c507b3e1f24f79d3ad359595a672b93aab3394db4e2d4a7d8970 -895144c55fcbd0f64d7dd69e6855cfb956e02b5658eadf0f026a70703f3643037268fdd673b0d21b288578a83c6338dd -a2e92ae6d0d237d1274259a8f99d4ea4912a299816350b876fba5ebc60b714490e198a916e1c38c6e020a792496fa23c -a45795fda3b5bb0ad1d3c628f6add5b2a4473a1414c1a232e80e70d1cfffd7f8a8d9861f8df2946999d7dbb56bf60113 -b6659bf7f6f2fef61c39923e8c23b8c70e9c903028d8f62516d16755cd3fba2fe41c285aa9432dc75ab08f8a1d8a81fc -a735609a6bc5bfd85e58234fc439ff1f58f1ff1dd966c5921d8b649e21f006bf2b8642ad8a75063c159aaf6935789293 -a3c622eb387c9d15e7bda2e3e84d007cb13a6d50d655c3f2f289758e49d3b37b9a35e4535d3cc53d8efd51f407281f19 -8afe147b53ad99220f5ef9d763bfc91f9c20caecbcf823564236fb0e6ede49414c57d71eec4772c8715cc65a81af0047 -b5f0203233cf71913951e9c9c4e10d9243e3e4a1f2cb235bf3f42009120ba96e04aa414c9938ea8873b63148478927e8 -93c52493361b458d196172d7ba982a90a4f79f03aa8008edc322950de3ce6acf4c3977807a2ffa9e924047e02072b229 -b9e72b805c8ac56503f4a86c82720afbd5c73654408a22a2ac0b2e5caccdfb0e20b59807433a6233bc97ae58cf14c70a -af0475779b5cee278cca14c82da2a9f9c8ef222eb885e8c50cca2315fea420de6e04146590ed0dd5a29c0e0812964df5 -b430ccab85690db02c2d0eb610f3197884ca12bc5f23c51e282bf3a6aa7e4a79222c3d8761454caf55d6c01a327595f9 -830032937418b26ee6da9b5206f3e24dc76acd98589e37937e963a8333e5430abd6ce3dd93ef4b8997bd41440eed75d6 -8820a6d73180f3fe255199f3f175c5eb770461ad5cfdde2fb11508041ed19b8c4ce66ad6ecebf7d7e836cc2318df47ca -aef1393e7d97278e77bbf52ef6e1c1d5db721ccf75fe753cf47a881fa034ca61eaa5098ee5a344c156d2b14ff9e284ad -8a4a26c07218948c1196c45d927ef4d2c42ade5e29fe7a91eaebe34a29900072ce5194cf28d51f746f4c4c649daf4396 -84011dc150b7177abdcb715efbd8c201f9cb39c36e6069af5c50a096021768ba40cef45b659c70915af209f904ede3b6 -b1bd90675411389bb66910b21a4bbb50edce5330850c5ab0b682393950124252766fc81f5ecfc72fb7184387238c402e -8dfdcd30583b696d2c7744655f79809f451a60c9ad5bf1226dc078b19f4585d7b3ef7fa9d54e1ac09520d95cbfd20928 -b351b4dc6d98f75b8e5a48eb7c6f6e4b78451991c9ba630e5a1b9874c15ac450cd409c1a024713bf2cf82dc400e025ef -a462b8bc97ac668b97b28b3ae24b9f5de60e098d7b23ecb600d2194cd35827fb79f77c3e50d358f5bd72ee83fef18fa0 -a183753265c5f7890270821880cce5f9b2965b115ba783c6dba9769536f57a04465d7da5049c7cf8b3fcf48146173c18 -a8a771b81ed0d09e0da4d79f990e58eabcd2be3a2680419502dd592783fe52f657fe55125b385c41d0ba3b9b9cf54a83 -a71ec577db46011689d073245e3b1c3222a9b1fe6aa5b83629adec5733dd48617ebea91346f0dd0e6cdaa86e4931b168 -a334b8b244f0d598a02da6ae0f918a7857a54dce928376c4c85df15f3b0f2ba3ac321296b8b7c9dd47d770daf16c8f8c -a29037f8ef925c417c90c4df4f9fb27fb977d04e2b3dd5e8547d33e92ab72e7a00f5461de21e28835319eae5db145eb7 -b91054108ae78b00e3298d667b913ebc44d8f26e531eae78a8fe26fdfb60271c97efb2dee5f47ef5a3c15c8228138927 -926c13efbe90604f6244be9315a34f72a1f8d1aab7572df431998949c378cddbf2fe393502c930fff614ff06ae98a0ce -995c758fd5600e6537089b1baa4fbe0376ab274ff3e82a17768b40df6f91c2e443411de9cafa1e65ea88fb8b87d504f4 -9245ba307a7a90847da75fca8d77ec03fdfc812c871e7a2529c56a0a79a6de16084258e7a9ac4ae8a3756f394336e21c -99e0cfa2bb57a7e624231317044c15e52196ecce020db567c8e8cb960354a0be9862ee0c128c60b44777e65ac315e59f -ad4f6b3d27bbbb744126601053c3dc98c07ff0eb0b38a898bd80dce778372846d67e5ab8fb34fb3ad0ef3f235d77ba7f -a0f12cae3722bbbca2e539eb9cc7614632a2aefe51410430070a12b5bc5314ecec5857b7ff8f41e9980cac23064f7c56 -b487f1bc59485848c98222fd3bc36c8c9bb3d2912e2911f4ceca32c840a7921477f9b1fe00877e05c96c75d3eecae061 -a6033db53925654e18ecb3ce715715c36165d7035db9397087ac3a0585e587998a53973d011ac6d48af439493029cee6 -a6b4d09cd01c70a3311fd131d3710ccf97bde3e7b80efd5a8c0eaeffeb48cca0f951ced905290267b115b06d46f2693b -a9dff1df0a8f4f218a98b6f818a693fb0d611fed0fc3143537cbd6578d479af13a653a8155e535548a2a0628ae24fa58 -a58e469f65d366b519f9a394cacb7edaddac214463b7b6d62c2dbc1316e11c6c5184ce45c16de2d77f990dcdd8b55430 -989e71734f8119103586dc9a3c5f5033ddc815a21018b34c1f876cdfc112efa868d5751bf6419323e4e59fa6a03ece1c -a2da00e05036c884369e04cf55f3de7d659cd5fa3f849092b2519dd263694efe0f051953d9d94b7e121f0aee8b6174d7 -968f3c029f57ee31c4e1adea89a7f92e28483af9a74f30fbdb995dc2d40e8e657dff8f8d340d4a92bf65f54440f2859f -932778df6f60ac1639c1453ef0cbd2bf67592759dcccb3e96dcc743ff01679e4c7dd0ef2b0833dda548d32cb4eba49e2 -a805a31139f8e0d6dae1ac87d454b23a3dc9fc653d4ca18d4f8ebab30fc189c16e73981c2cb7dd6f8c30454a5208109d -a9ba0991296caa2aaa4a1ceacfb205544c2a2ec97088eace1d84ee5e2767656a172f75d2f0c4e16a3640a0e0dec316e0 -b1e49055c968dced47ec95ae934cf45023836d180702e20e2df57e0f62fb85d7ac60d657ba3ae13b8560b67210449459 -a94e1da570a38809c71e37571066acabff7bf5632737c9ab6e4a32856924bf6211139ab3cedbf083850ff2d0e0c0fcfc -88ef1bb322000c5a5515b310c838c9af4c1cdbb32eab1c83ac3b2283191cd40e9573747d663763a28dad0d64adc13840 -a987ce205f923100df0fbd5a85f22c9b99b9b9cbe6ddfa8dfda1b8fe95b4f71ff01d6c5b64ca02eb24edb2b255a14ef0 -84fe8221a9e95d9178359918a108de4763ebfa7a6487facb9c963406882a08a9a93f492f8e77cf9e7ea41ae079c45993 -aa1cf3dc7c5dcfa15bbbc811a4bb6dbac4fba4f97fb1ed344ab60264d7051f6eef19ea9773441d89929ee942ed089319 -8f6a7d610d59d9f54689bbe6a41f92d9f6096cde919c1ab94c3c7fcecf0851423bc191e5612349e10f855121c0570f56 -b5af1fa7894428a53ea520f260f3dc3726da245026b6d5d240625380bfb9c7c186df0204bb604efac5e613a70af5106e -a5bce6055ff812e72ce105f147147c7d48d7a2313884dd1f488b1240ee320f13e8a33f5441953a8e7a3209f65b673ce1 -b9b55b4a1422677d95821e1d042ab81bbf0bf087496504021ec2e17e238c2ca6b44fb3b635a5c9eac0871a724b8d47c3 -941c38e533ce4a673a3830845b56786585e5fe49c427f2e5c279fc6db08530c8f91db3e6c7822ec6bb4f956940052d18 -a38e191d66c625f975313c7007bbe7431b5a06ed2da1290a7d5d0f2ec73770d476efd07b8e632de64597d47df175cbb0 -94ba76b667abf055621db4c4145d18743a368d951565632ed4e743dd50dd3333507c0c34f286a5c5fdbf38191a2255cd -a5ca38c60be5602f2bfa6e00c687ac96ac36d517145018ddbee6f12eb0faa63dd57909b9eeed26085fe5ac44e55d10ab -b00fea3b825e60c1ed1c5deb4b551aa65a340e5af36b17d5262c9cd2c508711e4dc50dc2521a2c16c7c901902266e64a -971b86fc4033485e235ccb0997a236206ba25c6859075edbcdf3c943116a5030b7f75ebca9753d863a522ba21a215a90 -b3b31f52370de246ee215400975b674f6da39b2f32514fe6bd54e747752eedca22bb840493b44a67df42a3639c5f901f -affbbfac9c1ba7cbfa1839d2ae271dd6149869b75790bf103230637da41857fc326ef3552ff31c15bda0694080198143 -a95d42aa7ef1962520845aa3688f2752d291926f7b0d73ea2ee24f0612c03b43f2b0fe3c9a9a99620ffc8d487b981bc2 -914a266065caf64985e8c5b1cb2e3f4e3fe94d7d085a1881b1fefa435afef4e1b39a98551d096a62e4f5cc1a7f0fdc2e -81a0b4a96e2b75bc1bf2dbd165d58d55cfd259000a35504d1ffb18bc346a3e6f07602c683723864ffb980f840836fd8d -91c1556631cddd4c00b65b67962b39e4a33429029d311c8acf73a18600e362304fb68bccb56fde40f49e95b7829e0b87 -8befbacc19e57f7c885d1b7a6028359eb3d80792fe13b92a8400df21ce48deb0bb60f2ddb50e3d74f39f85d7eab23adc -92f9458d674df6e990789690ec9ca73dacb67fc9255b58c417c555a8cc1208ace56e8e538f86ba0f3615573a0fbac00d -b4b1b3062512d6ae7417850c08c13f707d5838e43d48eb98dd4621baf62eee9e82348f80fe9b888a12874bfa538771f8 -a13c4a3ac642ede37d9c883f5319e748d2b938f708c9d779714108a449b343f7b71a6e3ef4080fee125b416762920273 -af44983d5fc8cceee0551ef934e6e653f2d3efa385e5c8a27a272463a6f333e290378cc307c2b664eb923c78994e706e -a389fd6c59fe2b4031cc244e22d3991e541bd203dd5b5e73a6159e72df1ab41d49994961500dcde7989e945213184778 -8d2141e4a17836c548de9598d7b298b03f0e6c73b7364979a411c464e0628e21cff6ac3d6decdba5d1c4909eff479761 -980b22ef53b7bdf188a3f14bc51b0dbfdf9c758826daa3cbc1e3986022406a8aa9a6a79e400567120b88c67faa35ce5f -a28882f0a055f96df3711de5d0aa69473e71245f4f3e9aa944e9d1fb166e02caa50832e46da6d3a03b4801735fd01b29 -8db106a37d7b88f5d995c126abb563934dd8de516af48e85695d02b1aea07f79217e3cdd03c6f5ca57421830186c772b -b5a7e50da0559a675c472f7dfaee456caab6695ab7870541b2be8c2b118c63752427184aad81f0e1afc61aef1f28c46f -9962118780e20fe291d10b64f28d09442a8e1b5cffd0f3dd68d980d0614050a626c616b44e9807fbee7accecae00686a -b38ddf33745e8d2ad6a991aefaf656a33c5f8cbe5d5b6b6fd03bd962153d8fd0e01b5f8f96d80ae53ab28d593ab1d4e7 -857dc12c0544ff2c0c703761d901aba636415dee45618aba2e3454ff9cbc634a85c8b05565e88520ff9be2d097c8b2b1 -a80d465c3f8cc63af6d74a6a5086b626c1cb4a8c0fee425964c3bd203d9d7094e299f81ce96d58afc20c8c9a029d9dae -89e1c8fbde8563763be483123a3ed702efac189c6d8ab4d16c85e74bbaf856048cc42d5d6e138633a38572ba5ec3f594 -893a594cf495535f6d216508f8d03c317dcf03446668cba688da90f52d0111ac83d76ad09bf5ea47056846585ee5c791 -aadbd8be0ae452f7f9450c7d2957598a20cbf10139a4023a78b4438172d62b18b0de39754dd2f8862dbd50a3a0815e53 -ae7d39670ecca3eb6db2095da2517a581b0e8853bdfef619b1fad9aacd443e7e6a40f18209fadd44038a55085c5fe8b2 -866ef241520eacb6331593cfcb206f7409d2f33d04542e6e52cba5447934e02d44c471f6c9a45963f9307e9809ab91d9 -b1a09911ad3864678f7be79a9c3c3eb5c84a0a45f8dcb52c67148f43439aeaaa9fd3ed3471276b7e588b49d6ebe3033a -add07b7f0dbb34049cd8feeb3c18da5944bf706871cfd9f14ff72f6c59ad217ebb1f0258b13b167851929387e4e34cfe -ae048892d5c328eefbdd4fba67d95901e3c14d974bfc0a1fc68155ca9f0d59e61d7ba17c6c9948b120cf35fd26e6fee9 -9185b4f3b7da0ddb4e0d0f09b8a9e0d6943a4611e43f13c3e2a767ed8592d31e0ba3ebe1914026a3627680274291f6e5 -a9c022d4e37b0802284ce3b7ee9258628ab4044f0db4de53d1c3efba9de19d15d65cc5e608dbe149c21c2af47d0b07b5 -b24dbd5852f8f24921a4e27013b6c3fa8885b973266cb839b9c388efad95821d5d746348179dcc07542bd0d0aefad1ce -b5fb4f279300876a539a27a441348764908bc0051ebd66dc51739807305e73db3d2f6f0f294ffb91b508ab150eaf8527 -ace50841e718265b290c3483ed4b0fdd1175338c5f1f7530ae9a0e75d5f80216f4de37536adcbc8d8c95982e88808cd0 -b19cadcde0f63bd1a9c24bd9c2806f53c14c0b9735bf351601498408ba503ddbd2037c891041cbba47f58b8c483f3b21 -b6061e63558d312eb891b97b39aa552fa218568d79ee26fe6dd5b864aea9e3216d8f2e2f3b093503be274766dac41426 -89730fdb2876ab6f0fe780d695f6e12090259027e789b819956d786e977518057e5d1d7f5ab24a3ae3d5d4c97773bd2b -b6fa841e81f9f2cad0163a02a63ae96dc341f7ae803b616efc6e1da2fbea551c1b96b11ad02c4afbdf6d0cc9f23da172 -8fb66187182629c861ddb6896d7ed3caf2ad050c3dba8ab8eb0d7a2c924c3d44c48d1a148f9e33fb1f061b86972f8d21 -86022ac339c1f84a7fa9e05358c1a5b316b4fc0b83dbe9c8c7225dc514f709d66490b539359b084ce776e301024345fa -b50b9c321468da950f01480bb62b6edafd42f83c0001d6e97f2bd523a1c49a0e8574fb66380ea28d23a7c4d54784f9f0 -a31c05f7032f30d1dac06678be64d0250a071fd655e557400e4a7f4c152be4d5c7aa32529baf3e5be7c4bd49820054f6 -b95ac0848cd322684772119f5b682d90a66bbf9dac411d9d86d2c34844bbd944dbaf8e47aa41380455abd51687931a78 -ae4a6a5ce9553b65a05f7935e61e496a4a0f6fd8203367a2c627394c9ce1e280750297b74cdc48fd1d9a31e93f97bef4 -a22daf35f6e9b05e52e0b07f7bd1dbbebd2c263033fb0e1b2c804e2d964e2f11bc0ece6aca6af079dd3a9939c9c80674 -902150e0cb1f16b9b59690db35281e28998ce275acb313900da8b2d8dfd29fa1795f8ca3ff820c31d0697de29df347c1 -b17b5104a5dc665cdd7d47e476153d715eb78c6e5199303e4b5445c21a7fa7cf85fe7cfd08d7570f4e84e579b005428c -a03f49b81c15433f121680aa02d734bb9e363af2156654a62bcb5b2ba2218398ccb0ff61104ea5d7df5b16ea18623b1e -802101abd5d3c88876e75a27ffc2f9ddcce75e6b24f23dba03e5201281a7bd5cc7530b6a003be92d225093ca17d3c3bb -a4d183f63c1b4521a6b52226fc19106158fc8ea402461a5cccdaa35fee93669df6a8661f45c1750cd01308149b7bf08e -8d17c22e0c8403b69736364d460b3014775c591032604413d20a5096a94d4030d7c50b9fe3240e31d0311efcf9816a47 -947225acfcce5992eab96276f668c3cbe5f298b90a59f2bb213be9997d8850919e8f496f182689b5cbd54084a7332482 -8df6f4ed216fc8d1905e06163ba1c90d336ab991a18564b0169623eb39b84e627fa267397da15d3ed754d1f3423bff07 -83480007a88f1a36dea464c32b849a3a999316044f12281e2e1c25f07d495f9b1710b4ba0d88e9560e72433addd50bc2 -b3019d6e591cf5b33eb972e49e06c6d0a82a73a75d78d383dd6f6a4269838289e6e07c245f54fed67f5c9bb0fd5e1c5f -92e8ce05e94927a9fb02debadb99cf30a26172b2705003a2c0c47b3d8002bf1060edb0f6a5750aad827c98a656b19199 -ac2aff801448dbbfc13cca7d603fd9c69e82100d997faf11f465323b97255504f10c0c77401e4d1890339d8b224f5803 -b0453d9903d08f508ee27e577445dc098baed6cde0ac984b42e0f0efed62760bd58d5816cf1e109d204607b7b175e30c -ae68dc4ba5067e825d46d2c7c67f1009ceb49d68e8d3e4c57f4bcd299eb2de3575d42ea45e8722f8f28497a6e14a1cfe -b22486c2f5b51d72335ce819bbafb7fa25eb1c28a378a658f13f9fc79cd20083a7e573248d911231b45a5cf23b561ca7 -89d1201d1dbd6921867341471488b4d2fd0fc773ae1d4d074c78ae2eb779a59b64c00452c2a0255826fca6b3d03be2b1 -a2998977c91c7a53dc6104f5bc0a5b675e5350f835e2f0af69825db8af4aeb68435bdbcc795f3dd1f55e1dd50bc0507f -b0be4937a925b3c05056ed621910d535ccabf5ab99fd3b9335080b0e51d9607d0fd36cb5781ff340018f6acfca4a9736 -aea145a0f6e0ba9df8e52e84bb9c9de2c2dc822f70d2724029b153eb68ee9c17de7d35063dcd6a39c37c59fdd12138f7 -91cb4545d7165ee8ffbc74c874baceca11fdebbc7387908d1a25877ca3c57f2c5def424dab24148826832f1e880bede0 -b3b579cb77573f19c571ad5eeeb21f65548d7dff9d298b8d7418c11f3e8cd3727c5b467f013cb87d6861cfaceee0d2e3 -b98a1eeec2b19fecc8378c876d73645aa52fb99e4819903735b2c7a885b242787a30d1269a04bfb8573d72d9bbc5f0f0 -940c1f01ed362bd588b950c27f8cc1d52276c71bb153d47f07ec85b038c11d9a8424b7904f424423e714454d5e80d1cd -aa343a8ecf09ce11599b8cf22f7279cf80f06dbf9f6d62cb05308dbbb39c46fd0a4a1240b032665fbb488a767379b91b -87c3ac72084aca5974599d3232e11d416348719e08443acaba2b328923af945031f86432e170dcdd103774ec92e988c9 -91d6486eb5e61d2b9a9e742c20ec974a47627c6096b3da56209c2b4e4757f007e793ebb63b2b246857c9839b64dc0233 -aebcd3257d295747dd6fc4ff910d839dd80c51c173ae59b8b2ec937747c2072fa85e3017f9060aa509af88dfc7529481 -b3075ba6668ca04eff19efbfa3356b92f0ab12632dcda99cf8c655f35b7928c304218e0f9799d68ef9f809a1492ff7db -93ba7468bb325639ec2abd4d55179c69fd04eaaf39fc5340709227bbaa4ad0a54ea8b480a1a3c8d44684e3be0f8d1980 -a6aef86c8c0d92839f38544d91b767c582568b391071228ff5a5a6b859c87bf4f81a7d926094a4ada1993ddbd677a920 -91dcd6d14207aa569194aa224d1e5037b999b69ade52843315ca61ba26abe9a76412c9e88259bc5cf5d7b95b97d9c3bc -b3b483d31c88f78d49bd065893bc1e3d2aa637e27dedb46d9a7d60be7660ce7a10aaaa7deead362284a52e6d14021178 -8e5730070acf8371461ef301cc4523e8e672aa0e3d945d438a0e0aa6bdf8cb9c685dcf38df429037b0c8aff3955c6f5b -b8c6d769890a8ee18dc4f9e917993315877c97549549b34785a92543cbeec96a08ae3a28d6e809c4aacd69de356c0012 -95ca86cd384eaceaa7c077c5615736ca31f36824bd6451a16142a1edc129fa42b50724aeed7c738f08d7b157f78b569e -94df609c6d71e8eee7ab74226e371ccc77e01738fe0ef1a6424435b4570fe1e5d15797b66ed0f64eb88d4a3a37631f0e -89057b9783212add6a0690d6bb99097b182738deff2bd9e147d7fd7d6c8eacb4c219923633e6309ad993c24572289901 -83a0f9f5f265c5a0e54defa87128240235e24498f20965009fef664f505a360b6fb4020f2742565dfc7746eb185bcec0 -91170da5306128931349bc3ed50d7df0e48a68b8cc8420975170723ac79d8773e4fa13c5f14dc6e3fafcad78379050b1 -b7178484d1b55f7e56a4cc250b6b2ec6040437d96bdfddfa7b35ed27435860f3855c2eb86c636f2911b012eb83b00db8 -ac0b00c4322d1e4208e09cd977b4e54d221133ff09551f75b32b0b55d0e2be80941dda26257b0e288c162e63c7e9cf68 -9690ed9e7e53ed37ff362930e4096b878b12234c332fd19d5d064824084245952eda9f979e0098110d6963e468cf513e -b6fa547bb0bb83e5c5be0ed462a8783fba119041c136a250045c09d0d2af330c604331e7de960df976ff76d67f8000cd -814603907c21463bcf4e59cfb43066dfe1a50344ae04ef03c87c0f61b30836c3f4dea0851d6fa358c620045b7f9214c8 -9495639e3939fad2a3df00a88603a5a180f3c3a0fe4d424c35060e2043e0921788003689887b1ed5be424d9a89bb18bb -aba4c02d8d57f2c92d5bc765885849e9ff8393d6554f5e5f3e907e5bfac041193a0d8716d7861104a4295d5a03c36b03 -8ead0b56c1ca49723f94a998ba113b9058059321da72d9e395a667e6a63d5a9dac0f5717cec343f021695e8ced1f72af -b43037f7e3852c34ed918c5854cd74e9d5799eeddfe457d4f93bb494801a064735e326a76e1f5e50a339844a2f4a8ec9 -99db8422bb7302199eb0ff3c3d08821f8c32f53a600c5b6fb43e41205d96adae72be5b460773d1280ad1acb806af9be8 -8a9be08eae0086c0f020838925984df345c5512ff32e37120b644512b1d9d4fecf0fd30639ca90fc6cf334a86770d536 -81b43614f1c28aa3713a309a88a782fb2bdfc4261dd52ddc204687791a40cf5fd6a263a8179388596582cccf0162efc2 -a9f3a8b76912deb61d966c75daf5ddb868702ebec91bd4033471c8e533183df548742a81a2671de5be63a502d827437d -902e2415077f063e638207dc7e14109652e42ab47caccd6204e2870115791c9defac5425fd360b37ac0f7bd8fe7011f8 -aa18e4fdc1381b59c18503ae6f6f2d6943445bd00dd7d4a2ad7e5adad7027f2263832690be30d456e6d772ad76f22350 -a348b40ba3ba7d81c5d4631f038186ebd5e5f314f1ea737259151b07c3cc8cf0c6ed4201e71bcc1c22fefda81a20cde6 -aa1306f7ac1acbfc47dc6f7a0cb6d03786cec8c8dc8060388ccda777bca24bdc634d03e53512c23dba79709ff64f8620 -818ccfe46e700567b7f3eb400e5a35f6a5e39b3db3aa8bc07f58ace35d9ae5a242faf8dbccd08d9a9175bbce15612155 -b7e3da2282b65dc8333592bb345a473f03bd6df69170055fec60222de9897184536bf22b9388b08160321144d0940279 -a4d976be0f0568f4e57de1460a1729129252b44c552a69fceec44e5b97c96c711763360d11f9e5bf6d86b4976bf40d69 -85d185f0397c24c2b875b09b6328a23b87982b84ee880f2677a22ff4c9a1ba9f0fea000bb3f7f66375a00d98ebafce17 -b4ccbb8c3a2606bd9b87ce022704663af71d418351575f3b350d294f4efc68c26f9a2ce49ff81e6ff29c3b63d746294e -93ffd3265fddb63724dfde261d1f9e22f15ecf39df28e4d89e9fea03221e8e88b5dd9b77628bacaa783c6f91802d47cc -b1fd0f8d7a01378e693da98d03a2d2fda6b099d03454b6f2b1fa6472ff6bb092751ce6290059826b74ac0361eab00e1e -a89f440c71c561641589796994dd2769616b9088766e983c873fae0716b95c386c8483ab8a4f367b6a68b72b7456dd32 -af4fe92b01d42d03dd5d1e7fa55e96d4bbcb7bf7d4c8c197acd16b3e0f3455807199f683dcd263d74547ef9c244b35cc -a8227f6e0a344dfe76bfbe7a1861be32c4f4bed587ccce09f9ce2cf481b2dda8ae4f566154bc663d15f962f2d41761bd -a7b361663f7495939ed7f518ba45ea9ff576c4e628995b7aea026480c17a71d63fc2c922319f0502eb7ef8f14a406882 -8ddcf382a9f39f75777160967c07012cfa89e67b19714a7191f0c68eaf263935e5504e1104aaabd0899348c972a8d3c6 -98c95b9f6f5c91f805fb185eedd06c6fc4457d37dd248d0be45a6a168a70031715165ea20606245cbdf8815dc0ac697f -805b44f96e001e5909834f70c09be3efcd3b43632bcac5b6b66b6d227a03a758e4b1768ce2a723045681a1d34562aaeb -b0e81b07cdc45b3dca60882676d9badb99f25c461b7efe56e3043b80100bb62d29e1873ae25eb83087273160ece72a55 -b0c53f0abe78ee86c7b78c82ae1f7c070bb0b9c45c563a8b3baa2c515d482d7507bb80771e60b38ac13f78b8af92b4a9 -a7838ef6696a9e4d2e5dfd581f6c8d6a700467e8fd4e85adabb5f7a56f514785dd4ab64f6f1b48366f7d94728359441b -88c76f7700a1d23c30366a1d8612a796da57b2500f97f88fdf2d76b045a9d24e7426a8ffa2f4e86d3046937a841dad58 -ad8964baf98c1f02e088d1d9fcb3af6b1dfa44cdfe0ed2eae684e7187c33d3a3c28c38e8f4e015f9c04d451ed6f85ff6 -90e9d00a098317ececaa9574da91fc149eda5b772dedb3e5a39636da6603aa007804fa86358550cfeff9be5a2cb7845e -a56ff4ddd73d9a6f5ab23bb77efa25977917df63571b269f6a999e1ad6681a88387fcc4ca3b26d57badf91b236503a29 -97ad839a6302c410a47e245df84c01fb9c4dfef86751af3f9340e86ff8fc3cd52fa5ff0b9a0bd1d9f453e02ca80658a6 -a4c8c44cbffa804129e123474854645107d1f0f463c45c30fd168848ebea94880f7c0c5a45183e9eb837f346270bdb35 -a72e53d0a1586d736e86427a93569f52edd2f42b01e78aee7e1961c2b63522423877ae3ac1227a2cf1e69f8e1ff15bc3 -8559f88a7ef13b4f09ac82ae458bbae6ab25671cfbf52dae7eac7280d6565dd3f0c3286aec1a56a8a16dc3b61d78ce47 -8221503f4cdbed550876c5dc118a3f2f17800c04e8be000266633c83777b039a432d576f3a36c8a01e8fd18289ebc10b -99bfbe5f3e46d4d898a578ba86ed26de7ed23914bd3bcdf3c791c0bcd49398a52419077354a5ab75cea63b6c871c6e96 -aa134416d8ff46f2acd866c1074af67566cfcf4e8be8d97329dfa0f603e1ff208488831ce5948ac8d75bfcba058ddcaa -b02609d65ebfe1fe8e52f21224a022ea4b5ea8c1bd6e7b9792eed8975fc387cdf9e3b419b8dd5bcce80703ab3a12a45f -a4f14798508698fa3852e5cac42a9db9797ecee7672a54988aa74037d334819aa7b2ac7b14efea6b81c509134a6b7ad2 -884f01afecbcb987cb3e7c489c43155c416ed41340f61ecb651d8cba884fb9274f6d9e7e4a46dd220253ae561614e44c -a05523c9e71dce1fe5307cc71bd721feb3e1a0f57a7d17c7d1c9fb080d44527b7dbaa1f817b1af1c0b4322e37bc4bb1e -8560aec176a4242b39f39433dd5a02d554248c9e49d3179530815f5031fee78ba9c71a35ceeb2b9d1f04c3617c13d8f0 -996aefd402748d8472477cae76d5a2b92e3f092fc834d5222ae50194dd884c9fb8b6ed8e5ccf8f6ed483ddbb4e80c747 -8fd09900320000cbabc40e16893e2fcf08815d288ec19345ad7b6bb22f7d78a52b6575a3ca1ca2f8bc252d2eafc928ec -939e51f73022bc5dc6862a0adf8fb8a3246b7bfb9943cbb4b27c73743926cc20f615a036c7e5b90c80840e7f1bfee0e7 -a0a6258700cadbb9e241f50766573bf9bdb7ad380b1079dc3afb4054363d838e177b869cad000314186936e40359b1f2 -972699a4131c8ed27a2d0e2104d54a65a7ff1c450ad9da3a325c662ab26869c21b0a84d0700b98c8b5f6ce3b746873d7 -a454c7fe870cb8aa6491eafbfb5f7872d6e696033f92e4991d057b59d70671f2acdabef533e229878b60c7fff8f748b1 -a167969477214201f09c79027b10221e4707662e0c0fde81a0f628249f2f8a859ce3d30a7dcc03b8ecca8f7828ad85c7 -8ff6b7265175beb8a63e1dbf18c9153fb2578c207c781282374f51b40d57a84fd2ef2ea2b9c6df4a54646788a62fd17f -a3d7ebeccde69d73d8b3e76af0da1a30884bb59729503ff0fb0c3bccf9221651b974a6e72ea33b7956fc3ae758226495 -b71ef144c9a98ce5935620cb86c1590bd4f48e5a2815d25c0cdb008fde628cf628c31450d3d4f67abbfeb16178a74cfd -b5e0a16d115134f4e2503990e3f2035ed66b9ccf767063fe6747870d97d73b10bc76ed668550cb82eedc9a2ca6f75524 -b30ffaaf94ee8cbc42aa2c413175b68afdb207dbf351fb20be3852cb7961b635c22838da97eaf43b103aff37e9e725cc -98aa7d52284f6c1f22e272fbddd8c8698cf8f5fbb702d5de96452141fafb559622815981e50b87a72c2b1190f59a7deb -81fbacda3905cfaf7780bb4850730c44166ed26a7c8d07197a5d4dcd969c09e94a0461638431476c16397dd7bdc449f9 -95e47021c1726eac2e5853f570d6225332c6e48e04c9738690d53e07c6b979283ebae31e2af1fc9c9b3e59f87e5195b1 -ac024a661ba568426bb8fce21780406537f518075c066276197300841e811860696f7588188bc01d90bace7bc73d56e3 -a4ebcaf668a888dd404988ab978594dee193dad2d0aec5cdc0ccaf4ec9a7a8228aa663db1da8ddc52ec8472178e40c32 -a20421b8eaf2199d93b083f2aff37fb662670bd18689d046ae976d1db1fedd2c2ff897985ecc6277b396db7da68bcb27 -8bc33d4b40197fd4d49d1de47489d10b90d9b346828f53a82256f3e9212b0cbc6930b895e879da9cec9fedf026aadb3e -aaafdd1bec8b757f55a0433eddc0a39f818591954fd4e982003437fcceb317423ad7ee74dbf17a2960380e7067a6b4e2 -aad34277ebaed81a6ec154d16736866f95832803af28aa5625bf0461a71d02b1faba02d9d9e002be51c8356425a56867 -976e9c8b150d08706079945bd0e84ab09a648ecc6f64ded9eb5329e57213149ae409ae93e8fbd8eda5b5c69f5212b883 -8097fae1653247d2aed4111533bc378171d6b2c6d09cbc7baa9b52f188d150d645941f46d19f7f5e27b7f073c1ebd079 -83905f93b250d3184eaba8ea7d727c4464b6bdb027e5cbe4f597d8b9dc741dcbea709630bd4fd59ce24023bec32fc0f3 -8095030b7045cff28f34271386e4752f9a9a0312f8df75de4f424366d78534be2b8e1720a19cb1f9a2d21105d790a225 -a7b7b73a6ae2ed1009c49960374b0790f93c74ee03b917642f33420498c188a169724945a975e5adec0a1e83e07fb1b2 -856a41c54df393b6660b7f6354572a4e71c8bfca9cabaffb3d4ef2632c015e7ee2bc10056f3eccb3dbed1ad17d939178 -a8f7a55cf04b38cd4e330394ee6589da3a07dc9673f74804fdf67b364e0b233f14aec42e783200a2e4666f7c5ff62490 -82c529f4e543c6bca60016dc93232c115b359eaee2798a9cf669a654b800aafe6ab4ba58ea8b9cdda2b371c8d62fa845 -8caab020c1baddce77a6794113ef1dfeafc5f5000f48e97f4351b588bf02f1f208101745463c480d37f588d5887e6d8c -8fa91b3cc400f48b77b6fd77f3b3fbfb3f10cdff408e1fd22d38f77e087b7683adad258804409ba099f1235b4b4d6fea -8aa02787663d6be9a35677d9d8188b725d5fcd770e61b11b64e3def8808ea5c71c0a9afd7f6630c48634546088fcd8e2 -b5635b7b972e195cab878b97dea62237c7f77eb57298538582a330b1082f6207a359f2923864630136d8b1f27c41b9aa -8257bb14583551a65975946980c714ecd6e5b629672bb950b9caacd886fbd22704bc9e3ba7d30778adab65dc74f0203a -ab5fe1cd12634bfa4e5c60d946e2005cbd38f1063ec9a5668994a2463c02449a0a185ef331bd86b68b6e23a8780cb3ba -a7d3487da56cda93570cc70215d438204f6a2709bfb5fda6c5df1e77e2efc80f4235c787e57fbf2c74aaff8cbb510a14 -b61cff7b4c49d010e133319fb828eb900f8a7e55114fc86b39c261a339c74f630e1a7d7e1350244ada566a0ff3d46c4b -8d4d1d55d321d278db7a85522ccceca09510374ca81d4d73e3bb5249ace7674b73900c35a531ec4fa6448fabf7ad00dc -966492248aee24f0f56c8cfca3c8ec6ba3b19abb69ae642041d4c3be8523d22c65c4dafcab4c58989ccc4e0bd2f77919 -b20c320a90cb220b86e1af651cdc1e21315cd215da69f6787e28157172f93fc8285dcd59b039c626ed8ca4633cba1a47 -aae9e6b22f018ceb5c0950210bb8182cb8cb61014b7e14581a09d36ebd1bbfebdb2b82afb7fdb0cf75e58a293d9c456d -875547fb67951ad37b02466b79f0c9b985ccbc500cfb431b17823457dc79fb9597ec42cd9f198e15523fcd88652e63a4 -92afce49773cb2e20fb21e4f86f18e0959ebb9c33361547ddb30454ee8e36b1e234019cbdca0e964cb292f7f77df6b90 -8af85343dfe1821464c76ba11c216cbef697b5afc69c4d821342e55afdac047081ec2e3f7b09fc14b518d9a23b78c003 -b7de4a1648fd63f3a918096ea669502af5357438e69dac77cb8102b6e6c15c76e033cfaa80dafc806e535ede5c1a20aa -ac80e9b545e8bd762951d96c9ce87f629d01ffcde07efc2ef7879ca011f1d0d8a745abf26c9d452541008871304fac00 -a4cf0f7ed724e481368016c38ea5816698a5f68eb21af4d3c422d2ba55f96a33e427c2aa40de1b56a7cfac7f7cf43ab0 -899b0a678bb2db2cae1b44e75a661284844ebcdd87abf308fedeb2e4dbe5c5920c07db4db7284a7af806a2382e8b111a -af0588a2a4afce2b1b13c1230816f59e8264177e774e4a341b289a101dcf6af813638fed14fb4d09cb45f35d5d032609 -a4b8df79e2be76e9f5fc5845f06fe745a724cf37c82fcdb72719b77bdebea3c0e763f37909373e3a94480cc5e875cba0 -83e42c46d88930c8f386b19fd999288f142d325e2ebc86a74907d6d77112cb0d449bc511c95422cc810574031a8cbba9 -b5e39534070de1e5f6e27efbdd3dc917d966c2a9b8cf2d893f964256e95e954330f2442027dc148c776d63a95bcde955 -958607569dc28c075e658cd4ae3927055c6bc456eef6212a6fea8205e48ed8777a8064f584cda38fe5639c371e2e7fba -812adf409fa63575113662966f5078a903212ffb65c9b0bbe62da0f13a133443a7062cb8fd70f5e5dd5559a32c26d2c8 -a679f673e5ce6a3cce7fa31f22ee3785e96bcb55e5a776e2dd3467bef7440e3555d1a9b87cb215e86ee9ed13a090344b -afedbb34508b159eb25eb2248d7fe328f86ef8c7d84c62d5b5607d74aae27cc2cc45ee148eb22153b09898a835c58df4 -b75505d4f6b67d31e665cfaf5e4acdb5838ae069166b7fbcd48937c0608a59e40a25302fcc1873d2e81c1782808c70f0 -b62515d539ec21a155d94fc00ea3c6b7e5f6636937bce18ed5b618c12257fb82571886287fd5d1da495296c663ebc512 -ab8e1a9446bbdd588d1690243b1549d230e6149c28f59662b66a8391a138d37ab594df38e7720fae53217e5c3573b5be -b31e8abf4212e03c3287bb2c0a153065a7290a16764a0bac8f112a72e632185a654bb4e88fdd6053e6c7515d9719fadb -b55165477fe15b6abd2d0f4fddaa9c411710dcc4dd712daba3d30e303c9a3ee5415c256f9dc917ecf18c725b4dbab059 -a0939d4f57cacaae549b78e87cc234de4ff6a35dc0d9cd5d7410abc30ebcd34c135e008651c756e5a9d2ca79c40ef42b -8cf10e50769f3443340844aad4d56ec790850fed5a41fcbd739abac4c3015f0a085a038fbe7fae9f5ad899cce5069f6b -924055e804d82a99ea4bb160041ea4dc14b568abf379010bc1922fde5d664718c31d103b8b807e3a1ae809390e708c73 -8ec0f9d26f71b0f2e60a179e4fd1778452e2ffb129d50815e5d7c7cb9415fa69ae5890578086e8ef6bfde35ad2a74661 -98c7f12b15ec4426b59f737f73bf5faea4572340f4550b7590dfb7f7ffedb2372e3e555977c63946d579544c53210ad0 -8a935f7a955c78f69d66f18eee0092e5e833fa621781c9581058e219af4d7ceee48b84e472e159dda6199715fb2f9acf -b78d4219f95a2dbfaa7d0c8a610c57c358754f4f43c2af312ab0fe8f10a5f0177e475332fb8fd23604e474fc2abeb051 -8d086a14803392b7318c28f1039a17e3cfdcece8abcaca3657ec3d0ac330842098a85c0212f889fabb296dfb133ce9aa -a53249f417aac82f2c2a50c244ce21d3e08a5e5a8bd33bec2a5ab0d6cd17793e34a17edfa3690899244ce201e2fb9986 -8619b0264f9182867a1425be514dc4f1ababc1093138a728a28bd7e4ecc99b9faaff68c23792264bc6e4dce5f52a5c52 -8c171edbbbde551ec19e31b2091eb6956107dd9b1f853e1df23bff3c10a3469ac77a58335eee2b79112502e8e163f3de -a9d19ec40f0ca07c238e9337c6d6a319190bdba2db76fb63902f3fb459aeeb50a1ac30db5b25ee1b4201f3ca7164a7f4 -b9c6ec14b1581a03520b8d2c1fbbc31fb8ceaef2c0f1a0d0080b6b96e18442f1734bea7ef7b635d787c691de4765d469 -8cb437beb4cfa013096f40ccc169a713dc17afee6daa229a398e45fd5c0645a9ad2795c3f0cd439531a7151945d7064d -a6e8740cc509126e146775157c2eb278003e5bb6c48465c160ed27888ca803fa12eee1f6a8dd7f444f571664ed87fdc1 -b75c1fecc85b2732e96b3f23aefb491dbd0206a21d682aee0225838dc057d7ed3b576176353e8e90ae55663f79e986e4 -ad8d249b0aea9597b08358bce6c77c1fd552ef3fbc197d6a1cfe44e5e6f89b628b12a6fb04d5dcfcbacc51f46e4ae7bb -b998b2269932cbd58d04b8e898d373ac4bb1a62e8567484f4f83e224061bc0f212459f1daae95abdbc63816ae6486a55 -827988ef6c1101cddc96b98f4a30365ff08eea2471dd949d2c0a9b35c3bbfa8c07054ad1f4c88c8fbf829b20bb5a9a4f -8692e638dd60babf7d9f2f2d2ce58e0ac689e1326d88311416357298c6a2bffbfebf55d5253563e7b3fbbf5072264146 -a685d75b91aea04dbc14ab3c1b1588e6de96dae414c8e37b8388766029631b28dd860688079b12d09cd27f2c5af11adf -b57eced93eec3371c56679c259b34ac0992286be4f4ff9489d81cf9712403509932e47404ddd86f89d7c1c3b6391b28c -a1c8b4e42ebcbd8927669a97f1b72e236fb19249325659e72be7ddaaa1d9e81ca2abb643295d41a8c04a2c01f9c0efd7 -877c33de20d4ed31674a671ba3e8f01a316581e32503136a70c9c15bf0b7cb7b1cba6cd4eb641fad165fb3c3c6c235fd -a2a469d84ec478da40838f775d11ad38f6596eb41caa139cc190d6a10b5108c09febae34ffdafac92271d2e73c143693 -972f817caedb254055d52e963ed28c206848b6c4cfdb69dbc961c891f8458eaf582a6d4403ce1177d87bc2ea410ef60a -accbd739e138007422f28536381decc54bb6bd71d93edf3890e54f9ef339f83d2821697d1a4ac1f5a98175f9a9ecb9b5 -8940f8772e05389f823b62b3adc3ed541f91647f0318d7a0d3f293aeeb421013de0d0a3664ea53dd24e5fbe02d7efef6 -8ecce20f3ef6212edef07ec4d6183fda8e0e8cad2c6ccd0b325e75c425ee1faba00b5c26b4d95204238931598d78f49d -97cc72c36335bd008afbed34a3b0c7225933faba87f7916d0a6d2161e6f82e0cdcda7959573a366f638ca75d30e9dab1 -9105f5de8699b5bdb6bd3bb6cc1992d1eac23929c29837985f83b22efdda92af64d9c574aa9640475087201bbbe5fd73 -8ffb33c4f6d05c413b9647eb6933526a350ed2e4278ca2ecc06b0e8026d8dbe829c476a40e45a6df63a633090a3f82ef -8bfc6421fdc9c2d2aaa68d2a69b1a2728c25b84944cc3e6a57ff0c94bfd210d1cbf4ff3f06702d2a8257024d8be7de63 -a80e1dc1dddfb41a70220939b96dc6935e00b32fb8be5dff4eed1f1c650002ff95e4af481c43292e3827363b7ec4768a -96f714ebd54617198bd636ba7f7a7f8995a61db20962f2165078d9ed8ee764d5946ef3cbdc7ebf8435bb8d5dd4c1deac -8cdb0890e33144d66391d2ae73f5c71f5a861f72bc93bff6cc399fc25dd1f9e17d8772592b44593429718784802ac377 -8ccf9a7f80800ee770b92add734ed45a73ecc31e2af0e04364eefc6056a8223834c7c0dc9dfc52495bdec6e74ce69994 -aa0875f423bd68b5f10ba978ddb79d3b96ec093bfbac9ff366323193e339ed7c4578760fb60f60e93598bdf1e5cc4995 -a9214f523957b59c7a4cb61a40251ad72aba0b57573163b0dc0f33e41d2df483fb9a1b85a5e7c080e9376c866790f8cb -b6224b605028c6673a536cc8ff9aeb94e7a22e686fda82cf16068d326469172f511219b68b2b3affb7933af0c1f80d07 -b6d58968d8a017c6a34e24c2c09852f736515a2c50f37232ac6b43a38f8faa7572cc31dade543b594b61b5761c4781d0 -8a97cefe5120020c38deeb861d394404e6c993c6cbd5989b6c9ebffe24f46ad11b4ba6348e2991cbf3949c28cfc3c99d -95bf046f8c3a9c0ce2634be4de3713024daec3fc4083e808903b25ce3ac971145af90686b451efcc72f6b22df0216667 -a6a4e2f71b8fa28801f553231eff2794c0f10d12e7e414276995e21195abc9c2983a8997e41af41e78d19ff6fbb2680b -8e5e62a7ca9c2f58ebaab63db2ff1fb1ff0877ae94b7f5e2897f273f684ae639dff44cc65718f78a9c894787602ab26a -8542784383eec4f565fcb8b9fc2ad8d7a644267d8d7612a0f476fc8df3aff458897a38003d506d24142ad18f93554f2b -b7db68ba4616ea072b37925ec4fb39096358c2832cc6d35169e032326b2d6614479f765ae98913c267105b84afcb9bf2 -8b31dbb9457d23d416c47542c786e07a489af35c4a87dadb8ee91bea5ac4a5315e65625d78dad2cf8f9561af31b45390 -a8545a1d91ac17257732033d89e6b7111db8242e9c6ebb0213a88906d5ef407a2c6fdb444e29504b06368b6efb4f4839 -b1bd85d29ebb28ccfb05779aad8674906b267c2bf8cdb1f9a0591dd621b53a4ee9f2942687ee3476740c0b4a7621a3ae -a2b54534e152e46c50d91fff03ae9cd019ff7cd9f4168b2fe7ac08ef8c3bbc134cadd3f9d6bd33d20ae476c2a8596c8a -b19b571ff4ae3e9f5d95acda133c455e72c9ea9973cae360732859836c0341c4c29ab039224dc5bc3deb824e031675d8 -940b5f80478648bac025a30f3efeb47023ce20ee98be833948a248bca6979f206bb28fc0f17b90acf3bb4abd3d14d731 -8f106b40588586ac11629b96d57808ad2808915d89539409c97414aded90b4ff23286a692608230a52bff696055ba5d6 -ae6bda03aa10da3d2abbc66d764ca6c8d0993e7304a1bdd413eb9622f3ca1913baa6da1e9f4f9e6cf847f14f44d6924d -a18e7796054a340ef826c4d6b5a117b80927afaf2ebd547794c400204ae2caf277692e2eabb55bc2f620763c9e9da66d -8d2d25180dc2c65a4844d3e66819ccfcf48858f0cc89e1c77553b463ec0f7feb9a4002ce26bc618d1142549b9850f232 -863f413a394de42cc8166c1c75d513b91d545fff1de6b359037a742c70b008d34bf8e587afa2d62c844d0c6f0ea753e7 -83cd0cf62d63475e7fcad18a2e74108499cdbf28af2113cfe005e3b5887794422da450b1944d0a986eb7e1f4c3b18f25 -b4f8b350a6d88fea5ab2e44715a292efb12eb52df738c9b2393da3f1ddee68d0a75b476733ccf93642154bceb208f2b8 -b3f52aaa4cd4221cb9fc45936cc67fd3864bf6d26bf3dd86aa85aa55ecfc05f5e392ecce5e7cf9406b4b1c4fce0398c8 -b33137084422fb643123f40a6df2b498065e65230fc65dc31791c330e898c51c3a65ff738930f32c63d78f3c9315f85b -91452bfa75019363976bb7337fe3a73f1c10f01637428c135536b0cdc7da5ce558dae3dfc792aa55022292600814a8ef -ad6ba94c787cd4361ca642c20793ea44f1f127d4de0bb4a77c7fbfebae0fcadbf28e2cb6f0c12c12a07324ec8c19761d -890aa6248b17f1501b0f869c556be7bf2b1d31a176f9978bb97ab7a6bd4138eed32467951c5ef1871944b7f620542f43 -82111db2052194ee7dd22ff1eafffac0443cf969d3762cceae046c9a11561c0fdce9c0711f88ac01d1bed165f8a7cee3 -b1527b71df2b42b55832f72e772a466e0fa05743aacc7814f4414e4bcc8d42a4010c9e0fd940e6f254cafedff3cd6543 -922370fa49903679fc565f09c16a5917f8125e72acfeb060fcdbadbd1644eb9f4016229756019c93c6d609cda5d5d174 -aa4c7d98a96cab138d2a53d4aee8ebff6ef903e3b629a92519608d88b3bbd94de5522291a1097e6acf830270e64c8ee1 -b3dc21608a389a72d3a752883a382baaafc61ecc44083b832610a237f6a2363f24195acce529eb4aed4ef0e27a12b66e -94619f5de05e07b32291e1d7ab1d8b7337a2235e49d4fb5f3055f090a65e932e829efa95db886b32b153bdd05a53ec8c -ade1e92722c2ffa85865d2426fb3d1654a16477d3abf580cfc45ea4b92d5668afc9d09275d3b79283e13e6b39e47424d -b7201589de7bed094911dd62fcd25c459a8e327ac447b69f541cdba30233063e5ddffad0b67e9c3e34adcffedfd0e13d -809d325310f862d6549e7cb40f7e5fc9b7544bd751dd28c4f363c724a0378c0e2adcb5e42ec8f912f5f49f18f3365c07 -a79c20aa533de7a5d671c99eb9eb454803ba54dd4f2efa3c8fec1a38f8308e9905c71e9282955225f686146388506ff6 -a85eeacb5e8fc9f3ed06a3fe2dc3108ab9f8c5877b148c73cf26e4e979bf5795edbe2e63a8d452565fd1176ed40402b2 -97ef55662f8a1ec0842b22ee21391227540adf7708f491436044f3a2eb18c471525e78e1e14fa292507c99d74d7437c6 -93110d64ed5886f3d16ce83b11425576a3a7a9bb831cd0de3f9a0b0f2270a730d68136b4ef7ff035ede004358f419b5c -ac9ed0a071517f0ae4f61ce95916a90ba9a77a3f84b0ec50ef7298acdcd44d1b94525d191c39d6bd1bb68f4471428760 -98abd6a02c7690f5a339adf292b8c9368dfc12e0f8069cf26a5e0ce54b4441638f5c66ea735142f3c28e00a0024267e6 -b51efb73ba6d44146f047d69b19c0722227a7748b0e8f644d0fc9551324cf034c041a2378c56ce8b58d06038fb8a78de -8f115af274ef75c1662b588b0896b97d71f8d67986ae846792702c4742ab855952865ce236b27e2321967ce36ff93357 -b3c4548f14d58b3ab03c222da09e4381a0afe47a72d18d50a94e0008797f78e39e99990e5b4757be62310d400746e35a -a9b1883bd5f31f909b8b1b6dcb48c1c60ed20aa7374b3ffa7f5b2ed036599b5bef33289d23c80a5e6420d191723b92f7 -85d38dffd99487ae5bb41ab4a44d80a46157bbbe8ef9497e68f061721f74e4da513ccc3422936b059575975f6787c936 -adf870fcb96e972c033ab7a35d28ae79ee795f82bc49c3bd69138f0e338103118d5529c53f2d72a9c0d947bf7d312af2 -ab4c7a44e2d9446c6ff303eb49aef0e367a58b22cc3bb27b4e69b55d1d9ee639c9234148d2ee95f9ca8079b1457d5a75 -a386420b738aba2d7145eb4cba6d643d96bda3f2ca55bb11980b318d43b289d55a108f4bc23a9606fb0bccdeb3b3bb30 -847020e0a440d9c4109773ecca5d8268b44d523389993b1f5e60e541187f7c597d79ebd6e318871815e26c96b4a4dbb1 -a530aa7e5ca86fcd1bec4b072b55cc793781f38a666c2033b510a69e110eeabb54c7d8cbcb9c61fee531a6f635ffa972 -87364a5ea1d270632a44269d686b2402da737948dac27f51b7a97af80b66728b0256547a5103d2227005541ca4b7ed04 -8816fc6e16ea277de93a6d793d0eb5c15e9e93eb958c5ef30adaf8241805adeb4da8ce19c3c2167f971f61e0b361077d -8836a72d301c42510367181bb091e4be377777aed57b73c29ef2ce1d475feedd7e0f31676284d9a94f6db01cc4de81a2 -b0d9d8b7116156d9dde138d28aa05a33e61f8a85839c1e9071ccd517b46a5b4b53acb32c2edd7150c15bc1b4bd8db9e3 -ae931b6eaeda790ba7f1cd674e53dc87f6306ff44951fa0df88d506316a5da240df9794ccbd7215a6470e6b31c5ea193 -8c6d5bdf87bd7f645419d7c6444e244fe054d437ed1ba0c122fde7800603a5fadc061e5b836cb22a6cfb2b466f20f013 -90d530c6d0cb654999fa771b8d11d723f54b8a8233d1052dc1e839ea6e314fbed3697084601f3e9bbb71d2b4eaa596df -b0d341a1422588c983f767b1ed36c18b141774f67ef6a43cff8e18b73a009da10fc12120938b8bba27f225bdfd3138f9 -a131b56f9537f460d304e9a1dd75702ace8abd68cb45419695cb8dee76998139058336c87b7afd6239dc20d7f8f940cc -aa6c51fa28975f709329adee1bbd35d49c6b878041841a94465e8218338e4371f5cb6c17f44a63ac93644bf28f15d20f -88440fb584a99ebd7f9ea04aaf622f6e44e2b43bbb49fb5de548d24a238dc8f26c8da2ccf03dd43102bda9f16623f609 -9777b8695b790e702159a4a750d5e7ff865425b95fa0a3c15495af385b91c90c00a6bd01d1b77bffe8c47d01baae846f -8b9d764ece7799079e63c7f01690c8eff00896a26a0d095773dea7a35967a8c40db7a6a74692f0118bf0460c26739af4 -85808c65c485520609c9e61fa1bb67b28f4611d3608a9f7a5030ee61c3aa3c7e7dc17fff48af76b4aecee2cb0dbd22ac -ad2783a76f5b3db008ef5f7e67391fda4e7e36abde6b3b089fc4835b5c339370287935af6bd53998bed4e399eda1136d -96f18ec03ae47c205cc4242ca58e2eff185c9dca86d5158817e2e5dc2207ab84aadda78725f8dc080a231efdc093b940 -97de1ab6c6cc646ae60cf7b86df73b9cf56cc0cd1f31b966951ebf79fc153531af55ca643b20b773daa7cab784b832f7 -870ba266a9bfa86ef644b1ef025a0f1b7609a60de170fe9508de8fd53170c0b48adb37f19397ee8019b041ce29a16576 -ad990e888d279ac4e8db90619d663d5ae027f994a3992c2fbc7d262b5990ae8a243e19157f3565671d1cb0de17fe6e55 -8d9d5adcdd94c5ba3be4d9a7428133b42e485f040a28d16ee2384758e87d35528f7f9868de9bd23d1a42a594ce50a567 -85a33ed75d514ece6ad78440e42f7fcdb59b6f4cff821188236d20edae9050b3a042ce9bc7d2054296e133d033e45022 -92afd2f49a124aaba90de59be85ff269457f982b54c91b06650c1b8055f9b4b0640fd378df02a00e4fc91f7d226ab980 -8c0ee09ec64bd831e544785e3d65418fe83ed9c920d9bb4d0bf6dd162c1264eb9d6652d2def0722e223915615931581c -8369bedfa17b24e9ad48ebd9c5afea4b66b3296d5770e09b00446c5b0a8a373d39d300780c01dcc1c6752792bccf5fd0 -8b9e960782576a59b2eb2250d346030daa50bbbec114e95cdb9e4b1ba18c3d34525ae388f859708131984976ca439d94 -b682bface862008fea2b5a07812ca6a28a58fd151a1d54c708fc2f8572916e0d678a9cb8dc1c10c0470025c8a605249e -a38d5e189bea540a824b36815fc41e3750760a52be0862c4cac68214febdc1a754fb194a7415a8fb7f96f6836196d82a -b9e7fbda650f18c7eb8b40e42cc42273a7298e65e8be524292369581861075c55299ce69309710e5b843cb884de171bd -b6657e5e31b3193874a1bace08f42faccbd3c502fb73ad87d15d18a1b6c2a146f1baa929e6f517db390a5a47b66c0acf -ae15487312f84ed6265e4c28327d24a8a0f4d2d17d4a5b7c29b974139cf93223435aaebe3af918f5b4bb20911799715f -8bb4608beb06bc394e1a70739b872ce5a2a3ffc98c7547bf2698c893ca399d6c13686f6663f483894bccaabc3b9c56ad -b58ac36bc6847077584308d952c5f3663e3001af5ecf2e19cb162e1c58bd6c49510205d453cffc876ca1dc6b8e04a578 -924f65ced61266a79a671ffb49b300f0ea44c50a0b4e3b02064faa99fcc3e4f6061ea8f38168ab118c5d47bd7804590e -8d67d43b8a06b0ff4fafd7f0483fa9ed1a9e3e658a03fb49d9d9b74e2e24858dc1bed065c12392037b467f255d4e5643 -b4d4f87813125a6b355e4519a81657fa97c43a6115817b819a6caf4823f1d6a1169683fd68f8d025cdfa40ebf3069acb -a7fd4d2c8e7b59b8eed3d4332ae94b77a89a2616347402f880bc81bde072220131e6dbec8a605be3a1c760b775375879 -8d4a7d8fa6f55a30df37bcf74952e2fa4fd6676a2e4606185cf154bdd84643fd01619f8fb8813a564f72e3f574f8ce30 -8086fb88e6260e9a9c42e9560fde76315ff5e5680ec7140f2a18438f15bc2cc7d7d43bfb5880b180b738c20a834e6134 -916c4c54721de03934fee6f43de50bb04c81f6f8dd4f6781e159e71c40c60408aa54251d457369d133d4ba3ed7c12cb4 -902e5bf468f11ed9954e2a4a595c27e34abe512f1d6dc08bbca1c2441063f9af3dc5a8075ab910a10ff6c05c1c644a35 -a1302953015e164bf4c15f7d4d35e3633425a78294406b861675667eec77765ff88472306531e5d3a4ec0a2ff0dd6a9e -87874461df3c9aa6c0fa91325576c0590f367075f2f0ecfeb34afe162c04c14f8ce9d608c37ac1adc8b9985bc036e366 -84b50a8a61d3cc609bfb0417348133e698fe09a6d37357ce3358de189efcf35773d78c57635c2d26c3542b13cc371752 -acaed2cff8633d12c1d12bb7270c54d65b0b0733ab084fd47f81d0a6e1e9b6f300e615e79538239e6160c566d8bb8d29 -889e6a0e136372ca4bac90d1ab220d4e1cad425a710e8cdd48b400b73bb8137291ceb36a39440fa84305783b1d42c72f -90952e5becec45b2b73719c228429a2c364991cf1d5a9d6845ae5b38018c2626f4308daa322cab1c72e0f6c621bb2b35 -8f5a97a801b6e9dcd66ccb80d337562c96f7914e7169e8ff0fda71534054c64bf2a9493bb830623d612cfe998789be65 -84f3df8b9847dcf1d63ca470dc623154898f83c25a6983e9b78c6d2d90a97bf5e622445be835f32c1e55e6a0a562ea78 -91d12095cd7a88e7f57f254f02fdb1a1ab18984871dead2f107404bcf8069fe68258c4e6f6ebd2477bddf738135400bb -b771a28bc04baef68604d4723791d3712f82b5e4fe316d7adc2fc01b935d8e644c06d59b83bcb542afc40ebafbee0683 -872f6341476e387604a7e93ae6d6117e72d164e38ebc2b825bc6df4fcce815004d7516423c190c1575946b5de438c08d -90d6b4aa7d40a020cdcd04e8b016d041795961a8e532a0e1f4041252131089114a251791bf57794cadb7d636342f5d1c -899023ba6096a181448d927fed7a0fe858be4eac4082a42e30b3050ee065278d72fa9b9d5ce3bc1372d4cbd30a2f2976 -a28f176571e1a9124f95973f414d5bdbf5794d41c3839d8b917100902ac4e2171eb940431236cec93928a60a77ede793 -838dbe5bcd29c4e465d02350270fa0036cd46f8730b13d91e77afb7f5ed16525d0021d3b2ae173a76c378516a903e0cb -8e105d012dd3f5d20f0f1c4a7e7f09f0fdd74ce554c3032e48da8cce0a77260d7d47a454851387770f5c256fa29bcb88 -8f4df0f9feeb7a487e1d138d13ea961459a6402fd8f8cabb226a92249a0d04ded5971f3242b9f90d08da5ff66da28af6 -ad1cfda4f2122a20935aa32fb17c536a3653a18617a65c6836700b5537122af5a8206befe9eaea781c1244c43778e7f1 -832c6f01d6571964ea383292efc8c8fa11e61c0634a25fa180737cc7ab57bc77f25e614aac9a2a03d98f27b3c1c29de2 -903f89cc13ec6685ac7728521898781fecb300e9094ef913d530bf875c18bcc3ceed7ed51e7b482d45619ab4b025c2e9 -a03c474bb915aad94f171e8d96f46abb2a19c9470601f4c915512ec8b9e743c3938450a2a5b077b4618b9df8809e1dc1 -83536c8456f306045a5f38ae4be2e350878fa7e164ea408d467f8c3bc4c2ee396bd5868008c089183868e4dfad7aa50b -88f26b4ea1b236cb326cd7ad7e2517ec8c4919598691474fe15d09cabcfc37a8d8b1b818f4d112432ee3a716b0f37871 -a44324e3fe96e9c12b40ded4f0f3397c8c7ee8ff5e96441118d8a6bfad712d3ac990b2a6a23231a8f691491ac1fd480f -b0de4693b4b9f932191a21ee88629964878680152a82996c0019ffc39f8d9369bbe2fe5844b68d6d9589ace54af947e4 -8e5d8ba948aea5fd26035351a960e87f0d23efddd8e13236cc8e4545a3dda2e9a85e6521efb8577e03772d3637d213d9 -93efc82d2017e9c57834a1246463e64774e56183bb247c8fc9dd98c56817e878d97b05f5c8d900acf1fbbbca6f146556 -8731176363ad7658a2862426ee47a5dce9434216cef60e6045fa57c40bb3ce1e78dac4510ae40f1f31db5967022ced32 -b10c9a96745722c85bdb1a693100104d560433d45b9ac4add54c7646a7310d8e9b3ca9abd1039d473ae768a18e489845 -a2ac374dfbb464bf850b4a2caf15b112634a6428e8395f9c9243baefd2452b4b4c61b0cb2836d8eae2d57d4900bf407e -b69fe3ded0c4f5d44a09a0e0f398221b6d1bf5dbb8bc4e338b93c64f1a3cac1e4b5f73c2b8117158030ec03787f4b452 -8852cdbaf7d0447a8c6f211b4830711b3b5c105c0f316e3a6a18dcfbb9be08bd6f4e5c8ae0c3692da08a2dfa532f9d5c -93bbf6d7432a7d98ade3f94b57bf9f4da9bc221a180a370b113066dd42601bb9e09edd79e2e6e04e00423399339eebda -a80941c391f1eeafc1451c59e4775d6a383946ff22997aeaadf806542ba451d3b0f0c6864eeba954174a296efe2c1550 -a045fe2bb011c2a2f71a0181a8f457a3078470fb74c628eab8b59aef69ffd0d649723bf74d6885af3f028bc5a104fb39 -b9d8c35911009c4c8cad64692139bf3fc16b78f5a19980790cb6a7aea650a25df4231a4437ae0c351676a7e42c16134f -94c79501ded0cfcbab99e1841abe4a00a0252b3870e20774c3da16c982d74c501916ec28304e71194845be6e3113c7ab -900a66418b082a24c6348d8644ddb1817df5b25cb33044a519ef47cc8e1f7f1e38d2465b7b96d32ed472d2d17f8414c6 -b26f45d393b8b2fcb29bdbb16323dc7f4b81c09618519ab3a39f8ee5bd148d0d9f3c0b5dfab55b5ce14a1cb9206d777b -aa1a87735fc493a80a96a9a57ca40a6d9c32702bfcaa9869ce1a116ae65d69cefe2f3e79a12454b4590353e96f8912b4 -a922b188d3d0b69b4e4ea2a2aa076566962844637da12c0832105d7b31dea4a309eee15d12b7a336be3ea36fcbd3e3b7 -8f3841fcf4105131d8c4d9885e6e11a46c448226401cf99356c291fadb864da9fa9d30f3a73c327f23f9fd99a11d633e -9791d1183fae270e226379af6c497e7da803ea854bb20afa74b253239b744c15f670ee808f708ede873e78d79a626c9a -a4cad52e3369491ada61bf28ada9e85de4516d21c882e5f1cd845bea9c06e0b2887b0c5527fcff6fc28acd3c04f0a796 -b9ac86a900899603452bd11a7892a9bfed8054970bfcbeaa8c9d1930db891169e38d6977f5258c25734f96c8462eee3b -a3a154c28e5580656a859f4efc2f5ebfa7eaa84ca40e3f134fa7865e8581586db74992dbfa4036aa252fba103773ddde -95cc2a0c1885a029e094f5d737e3ecf4d26b99036453a8773c77e360101f9f98676ee246f6f732a377a996702d55691f -842651bbe99720438d8d4b0218feb60481280c05beb17750e9ca0d8c0599a60f873b7fbdcc7d8835ba9a6d57b16eec03 -81ee54699da98f5620307893dcea8f64670609fa20e5622265d66283adeac122d458b3308c5898e6c57c298db2c8b24f -b97868b0b2bc98032d68352a535a1b341b9ff3c7af4e3a7f3ebc82d3419daa1b5859d6aedc39994939623c7cd878bd9b -b60325cd5d36461d07ef253d826f37f9ee6474a760f2fff80f9873d01fd2b57711543cdc8d7afa1c350aa753c2e33dea -8c205326c11d25a46717b780c639d89714c7736c974ae71287e3f4b02e6605ac2d9b4928967b1684f12be040b7bf2dd3 -95a392d82db51e26ade6c2ccd3396d7e40aff68fa570b5951466580d6e56dda51775dce5cf3a74a7f28c3cb2eb551c4d -8f2cc8071eb56dffb70bda6dd433b556221dc8bba21c53353c865f00e7d4d86c9e39f119ea9a8a12ef583e9a55d9a6b6 -9449a71af9672aaf8856896d7e3d788b22991a7103f75b08c0abbcc2bfe60fda4ed8ce502cea4511ff0ea52a93e81222 -857090ab9fdb7d59632d068f3cc8cf27e61f0d8322d30e6b38e780a1f05227199b4cd746aac1311c36c659ef20931f28 -98a891f4973e7d9aaf9ac70854608d4f7493dffc7e0987d7be9dd6029f6ea5636d24ef3a83205615ca1ff403750058e1 -a486e1365bbc278dd66a2a25d258dc82f46b911103cb16aab3945b9c95ae87b386313a12b566df5b22322ede0afe25ad -a9a1eb399ed95d396dccd8d1ac718043446f8b979ec62bdce51c617c97a312f01376ab7fb87d27034e5f5570797b3c33 -b7abc3858d7a74bb446218d2f5a037e0fae11871ed9caf44b29b69c500c1fa1dcfad64c9cdccc9d80d5e584f06213deb -8cfb09fe2e202faa4cebad932b1d35f5ca204e1c2a0c740a57812ac9a6792130d1312aabd9e9d4c58ca168bfebd4c177 -a90a305c2cd0f184787c6be596fa67f436afd1f9b93f30e875f817ac2aae8bdd2e6e656f6be809467e6b3ad84adb86b1 -80a9ef993c2b009ae172cc8f7ec036f5734cf4f4dfa06a7db4d54725e7fbfae5e3bc6f22687bdbb6961939d6f0c87537 -848ade1901931e72b955d7db1893f07003e1708ff5d93174bac5930b9a732640f0578839203e9b77eb27965c700032d3 -93fdf4697609c5ae9c33b9ca2f5f1af44abeb2b98dc4fdf732cf7388de086f410730dc384d9b7a7f447bb009653c8381 -89ce3fb805aea618b5715c0d22a9f46da696b6fa86794f56fdf1d44155a33d42daf1920bcbe36cbacf3cf4c92df9cbc7 -829ce2c342cf82aa469c65f724f308f7a750bd1494adc264609cd790c8718b8b25b5cab5858cf4ee2f8f651d569eea67 -af2f0cee7bf413204be8b9df59b9e4991bc9009e0d6dbe6815181df0ec2ca93ab8f4f3135b1c14d8f53d74bff0bd6f27 -b87998cecf7b88cde93d1779f10a521edd5574a2fbd240102978639ec57433ba08cdb53849038a329cebbe74657268d2 -a64542a1261a6ed3d720c2c3a802303aad8c4c110c95d0f12e05c1065e66f42da494792b6bfc5b9272363f3b1d457f58 -86a6fd042e4f282fadf07a4bfee03fc96a3aea49f7a00f52bf249a20f1ec892326855410e61f37fbb27d9305eb2fc713 -967ea5bc403b6db269682f7fd0df90659350d7e1aa66bc4fab4c9dfcd75ed0bba4b52f1cebc5f34dc8ba810793727629 -a52990f9f3b8616ce3cdc2c74cd195029e6a969753dcf2d1630438700e7d6ebde36538532b3525ac516f5f2ce9dd27a3 -a64f7ff870bab4a8bf0d4ef6f5c744e9bf1021ed08b4c80903c7ad318e80ba1817c3180cc45cb5a1cae1170f0241655f -b00f706fa4de1f663f021e8ad3d155e84ce6084a409374b6e6cd0f924a0a0b51bebaaaf1d228c77233a73b0a5a0df0e9 -8b882cc3bff3e42babdb96df95fb780faded84887a0a9bab896bef371cdcf169d909f5658649e93006aa3c6e1146d62e -9332663ef1d1dcf805c3d0e4ce7a07d9863fb1731172e766b3cde030bf81682cc011e26b773fb9c68e0477b4ae2cfb79 -a8aa8151348dbd4ef40aaeb699b71b4c4bfd3218560c120d85036d14f678f6736f0ec68e80ce1459d3d35feccc575164 -a16cd8b729768f51881c213434aa28301fa78fcb554ddd5f9012ee1e4eae7b5cb3dd88d269d53146dea92d10790faf0b -86844f0ef9d37142faf3b1e196e44fbe280a3ba4189aa05c356778cb9e3b388a2bff95eed305ada8769935c9974e4c57 -ae2eec6b328fccf3b47bcdac32901ac2744a51beb410b04c81dea34dee4912b619466a4f5e2780d87ecefaebbe77b46d -915df4c38d301c8a4eb2dc5b1ba0ffaad67cbb177e0a80095614e9c711f4ef24a4cef133f9d982a63d2a943ba6c8669d -ae6a2a4dedfc2d1811711a8946991fede972fdf2a389b282471280737536ffc0ac3a6d885b1f8bda0366eb0b229b9979 -a9b628c63d08b8aba6b1317f6e91c34b2382a6c85376e8ef2410a463c6796740ae936fc4e9e0737cb9455d1daa287bd8 -848e30bf7edf2546670b390d5cf9ab71f98fcb6add3c0b582cb34996c26a446dee5d1bde4fdcde4fc80c10936e117b29 -907d6096c7c8c087d1808dd995d5d2b9169b3768c3f433475b50c2e2bd4b082f4d543afd8b0b0ddffa9c66222a72d51d -a59970a2493b07339124d763ac9d793c60a03354539ecbcf6035bc43d1ea6e35718202ae6d7060b7d388f483d971573c -b9cfef2af9681b2318f119d8611ff6d9485a68d8044581b1959ab1840cbca576dbb53eec17863d2149966e9feb21122f -ad47271806161f61d3afa45cdfe2babceef5e90031a21779f83dc8562e6076680525b4970b2f11fe9b2b23c382768323 -8e425a99b71677b04fe044625d338811fbb8ee32368a424f6ab2381c52e86ee7a6cecedf777dc97181519d41c351bc22 -86b55b54d7adefc12954a9252ee23ae83efe8b5b4b9a7dc307904413e5d69868c7087a818b2833f9b004213d629be8ad -a14fda6b93923dd11e564ae4457a66f397741527166e0b16a8eb91c6701c244fd1c4b63f9dd3515193ec88fa6c266b35 -a9b17c36ae6cd85a0ed7f6cabc5b47dc8f80ced605db327c47826476dc1fb8f8669aa7a7dc679fbd4ee3d8e8b4bd6a6f -82a0829469c1458d959c821148f15dacae9ea94bf56c59a6ab2d4dd8b3d16d73e313b5a3912a6c1f131d73a8f06730c4 -b22d56d549a53eaef549595924bdb621ff807aa4513feedf3fdcbf7ba8b6b9cfa4481c2f67fc642db397a6b794a8b63a -974c59c24392e2cb9294006cbe3c52163e255f3bd0c2b457bdc68a6338e6d5b6f87f716854492f8d880a6b896ccf757c -b70d247ba7cad97c50b57f526c2ba915786e926a94e8f8c3eebc2e1be6f4255411b9670e382060049c8f4184302c40b2 -ad80201fe75ef21c3ddbd98cf23591e0d7a3ba1036dfe77785c32f44755a212c31f0ceb0a0b6f5ee9b6dc81f358d30c3 -8c656e841f9bb90b9a42d425251f3fdbc022a604d75f5845f479ed4be23e02aaf9e6e56cde351dd7449c50574818a199 -8b88dd3fa209d3063b7c5b058f7249ee9900fbc2287d16da61a0704a0a1d71e45d9c96e1cda7fdf9654534ec44558b22 -961da00cc8750bd84d253c08f011970ae1b1158ad6778e8ed943d547bceaf52d6d5a212a7de3bf2706688c4389b827d2 -a5dd379922549a956033e3d51a986a4b1508e575042b8eaa1df007aa77cf0b8c2ab23212f9c075702788fa9c53696133 -ac8fcfde3a349d1e93fc8cf450814e842005c545c4844c0401bc80e6b96cdb77f29285a14455e167c191d4f312e866cd -ac63d79c799783a8466617030c59dd5a8f92ee6c5204676fd8d881ce5f7f8663bdbeb0379e480ea9b6340ab0dc88e574 -805874fde19ce359041ae2bd52a39e2841acabfd31f965792f2737d7137f36d4e4722ede8340d8c95afa6af278af8acb -8d2f323a228aa8ba7b7dc1399138f9e6b41df1a16a7069003ab8104b8b68506a45141bc5fe66acf430e23e13a545190b -a1610c721a2d9af882bb6b39bea97cff1527a3aea041d25934de080214ae77c959e79957164440686d15ab301e897d4d -aba16d29a47fc36f12b654fde513896723e2c700c4190f11b26aa4011da57737ad717daa02794aa3246e4ae5f0b0cc3a -a406db2f15fdd135f346cc4846623c47edd195e80ba8c7cb447332095314d565e4040694ca924696bb5ee7f8996ea0ba -8b30e2cd9b47d75ba57b83630e40f832249af6c058d4f490416562af451993eec46f3e1f90bc4d389e4c06abd1b32a46 -aacf9eb7036e248e209adbfc3dd7ce386569ea9b312caa4b240726549db3c68c4f1c8cbf8ed5ea9ea60c7e57c9df3b8e -b20fcac63bf6f5ee638a42d7f89be847f348c085ddcbec3fa318f4323592d136c230495f188ef2022aa355cc2b0da6f9 -811eff750456a79ec1b1249d76d7c1547065b839d8d4aaad860f6d4528eb5b669473dcceeeea676cddbc3980b68461b7 -b52d14ae33f4ab422f953392ae76a19c618cc31afc96290bd3fe2fb44c954b5c92c4789f3f16e8793f2c0c1691ade444 -a7826dafeeba0db5b66c4dfcf2b17fd7b40507a5a53ac2e42942633a2cb30b95ba1739a6e9f3b7a0e0f1ec729bf274e2 -8acfd83ddf7c60dd7c8b20c706a3b972c65d336b8f9b3d907bdd8926ced271430479448100050b1ef17578a49c8fa616 -af0c69f65184bb06868029ad46f8465d75c36814c621ac20a5c0b06a900d59305584f5a6709683d9c0e4b6cd08d650a6 -b6cc8588191e00680ee6c3339bd0f0a17ad8fd7f4be57d5d7075bede0ea593a19e67f3d7c1a20114894ee5bfcab71063 -a82fd4f58635129dbb6cc3eb9391cf2d28400018b105fc41500fbbd12bd890b918f97d3d359c29dd3b4c4e34391dfab0 -92fc544ed65b4a3625cf03c41ddff7c039bc22d22c0d59dcc00efd5438401f2606adb125a1d5de294cca216ec8ac35a3 -906f67e4a32582b71f15940523c0c7ce370336935e2646bdaea16a06995256d25e99df57297e39d6c39535e180456407 -97510337ea5bbd5977287339197db55c60533b2ec35c94d0a460a416ae9f60e85cee39be82abeeacd5813cf54df05862 -87e6894643815c0ea48cb96c607266c5ee4f1f82ba5fe352fb77f9b6ed14bfc2b8e09e80a99ac9047dfcf62b2ae26795 -b6fd55dd156622ad7d5d51b7dde75e47bd052d4e542dd6449e72411f68275775c846dde301e84613312be8c7bce58b07 -b98461ac71f554b2f03a94e429b255af89eec917e208a8e60edf5fc43b65f1d17a20de3f31d2ce9f0cb573c25f2f4d98 -96f0dea40ca61cefbee41c4e1fe9a7d81fbe1f49bb153d083ab70f5d0488a1f717fd28cedcf6aa18d07cce2c62801898 -8d7c3ab310184f7dc34b6ce4684e4d29a31e77b09940448ea4daac730b7eb308063125d4dd229046cf11bfd521b771e0 -96f0564898fe96687918bbf0a6adead99cf72e3a35ea3347e124af9d006221f8e82e5a9d2fe80094d5e8d48e610f415e -ad50fcb92c2675a398cf07d4c40a579e44bf8d35f27cc330b57e54d5ea59f7d898af0f75dccfe3726e5471133d70f92b -828beed62020361689ae7481dd8f116902b522fb0c6c122678e7f949fdef70ead011e0e6bffd25678e388744e17cdb69 -8349decac1ca16599eee2efc95bcaabf67631107da1d34a2f917884bd70dfec9b4b08ab7bc4379d6c73b19c0b6e54fb8 -b2a6a2e50230c05613ace9e58bb2e98d94127f196f02d9dddc53c43fc68c184549ca12d713cb1b025d8260a41e947155 -94ff52181aadae832aed52fc3b7794536e2a31a21fc8be3ea312ca5c695750d37f08002f286b33f4023dba1e3253ecfa -a21d56153c7e5972ee9a319501be4faff199fdf09bb821ea9ce64aa815289676c00f105e6f00311b3a5b627091b0d0fc -a27a60d219f1f0c971db73a7f563b371b5c9fc3ed1f72883b2eac8a0df6698400c9954f4ca17d7e94e44bd4f95532afb -a2fc56fae99b1f18ba5e4fe838402164ce82f8a7f3193d0bbd360c2bac07c46f9330c4c7681ffb47074c6f81ee6e7ac6 -b748e530cd3afb96d879b83e89c9f1a444f54e55372ab1dcd46a0872f95ce8f49cf2363fc61be82259e04f555937ed16 -8bf8993e81080c7cbba1e14a798504af1e4950b2f186ab3335b771d6acaee4ffe92131ae9c53d74379d957cb6344d9cd -96774d0ef730d22d7ab6d9fb7f90b9ead44285219d076584a901960542756700a2a1603cdf72be4708b267200f6c36a9 -b47703c2ab17be1e823cc7bf3460db1d6760c0e33862c90ca058845b2ff234b0f9834ddba2efb2ee1770eb261e7d8ffd -84319e67c37a9581f8b09b5e4d4ae88d0a7fb4cbb6908971ab5be28070c3830f040b1de83ee663c573e0f2f6198640e4 -96811875fa83133e0b3c0e0290f9e0e28bca6178b77fdf5350eb19344d453dbd0d71e55a0ef749025a5a2ca0ad251e81 -81a423423e9438343879f2bfd7ee9f1c74ebebe7ce3cfffc8a11da6f040cc4145c3b527bd3cf63f9137e714dbcb474ef -b8c3535701ddbeec2db08e17a4fa99ba6752d32ece5331a0b8743676f421fcb14798afc7c783815484f14693d2f70db8 -81aee980c876949bf40782835eec8817d535f6f3f7e00bf402ddd61101fdcd60173961ae90a1cf7c5d060339a18c959d -87e67b928d97b62c49dac321ce6cb680233f3a394d4c9a899ac2e8db8ccd8e00418e66cdfd68691aa3cb8559723b580c -8eac204208d99a2b738648df96353bbb1b1065e33ee4f6bba174b540bbbd37d205855e1f1e69a6b7ff043ca377651126 -848e6e7a54ad64d18009300b93ea6f459ce855971dddb419b101f5ac4c159215626fadc20cc3b9ab1701d8f6dfaddd8b -88aa123d9e0cf309d46dddb6acf634b1ade3b090a2826d6e5e78669fa1220d6df9a6697d7778cd9b627db17eea846126 -9200c2a629b9144d88a61151b661b6c4256cc5dadfd1e59a8ce17a013c2d8f7e754aabe61663c3b30f1bc47784c1f8cf -b6e1a2827c3bdda91715b0e1b1f10dd363cef337e7c80cac1f34165fc0dea7c8b69747e310563db5818390146ce3e231 -92c333e694f89f0d306d54105b2a5dcc912dbe7654d9e733edab12e8537350815be472b063e56cfde5286df8922fdecb -a6fac04b6d86091158ebb286586ccfec2a95c9786e14d91a9c743f5f05546073e5e3cc717635a0c602cad8334e922346 -a581b4af77feebc1fb897d49b5b507c6ad513d8f09b273328efbb24ef0d91eb740d01b4d398f2738125dacfe550330cd -81c4860cccf76a34f8a2bc3f464b7bfd3e909e975cce0d28979f457738a56e60a4af8e68a3992cf273b5946e8d7f76e2 -8d1eaa09a3180d8af1cbaee673db5223363cc7229a69565f592fa38ba0f9d582cedf91e15dabd06ebbf2862fc0feba54 -9832f49b0147f4552402e54593cfa51f99540bffada12759b71fcb86734be8e500eea2d8b3d036710bdf04c901432de9 -8bdb0e8ec93b11e5718e8c13cb4f5de545d24829fd76161216340108098dfe5148ed25e3b57a89a516f09fa79043734d -ab96f06c4b9b0b2c0571740b24fca758e6976315053a7ecb20119150a9fa416db2d3a2e0f8168b390bb063f0c1caf785 -ab777f5c52acd62ecf4d1f168b9cc8e1a9b45d4ec6a8ff52c583e867c2239aba98d7d3af977289b367edce03d9c2dfb1 -a09d3ce5e748da84802436951acc3d3ea5d8ec1d6933505ed724d6b4b0d69973ab0930daec9c6606960f6e541e4a3ce2 -8ef94f7be4d85d5ad3d779a5cf4d7b2fc3e65c52fb8e1c3c112509a4af77a0b5be994f251e5e40fabeeb1f7d5615c22b -a7406a5bf5708d9e10922d3c5c45c03ef891b8d0d74ec9f28328a72be4cdc05b4f2703fa99366426659dfca25d007535 -b7f52709669bf92a2e070bfe740f422f0b7127392c5589c7f0af71bb5a8428697c762d3c0d74532899da24ea7d8695c2 -b9dfb0c8df84104dbf9239ccefa4672ef95ddabb8801b74997935d1b81a78a6a5669a3c553767ec19a1281f6e570f4ff -ae4d5c872156061ce9195ac640190d8d71dd406055ee43ffa6f9893eb24b870075b74c94d65bc1d5a07a6573282b5520 -afe6bd3eb72266d333f1807164900dcfa02a7eb5b1744bb3c86b34b3ee91e3f05e38fa52a50dc64eeb4bdb1dd62874b8 -948043cf1bc2ef3c01105f6a78dc06487f57548a3e6ef30e6ebc51c94b71e4bf3ff6d0058c72b6f3ecc37efd7c7fa8c0 -a22fd17c2f7ffe552bb0f23fa135584e8d2d8d75e3f742d94d04aded2a79e22a00dfe7acbb57d44e1cdb962fb22ae170 -8cd0f4e9e4fb4a37c02c1bde0f69359c43ab012eb662d346487be0c3758293f1ca560122b059b091fddce626383c3a8f -90499e45f5b9c81426f3d735a52a564cafbed72711d9279fdd88de8038e953bc48c57b58cba85c3b2e4ce56f1ddb0e11 -8c30e4c034c02958384564cac4f85022ef36ab5697a3d2feaf6bf105049675bbf23d01b4b6814711d3d9271abff04cac -81f7999e7eeea30f3e1075e6780bbf054f2fb6f27628a2afa4d41872a385b4216dd5f549da7ce6cf39049b2251f27fb7 -b36a7191f82fc39c283ffe53fc1f5a9a00b4c64eee7792a8443475da9a4d226cf257f226ea9d66e329af15d8f04984ec -aad4da528fdbb4db504f3041c747455baff5fcd459a2efd78f15bdf3aea0bdb808343e49df88fe7a7c8620009b7964a3 -99ebd8c6dd5dd299517fb6381cfc2a7f443e6e04a351440260dd7c2aee3f1d8ef06eb6c18820b394366ecdfd2a3ce264 -8873725b81871db72e4ec3643084b1cdce3cbf80b40b834b092767728605825c19b6847ad3dcf328438607e8f88b4410 -b008ee2f895daa6abd35bd39b6f7901ae4611a11a3271194e19da1cdcc7f1e1ea008fe5c5440e50d2c273784541ad9c5 -9036feafb4218d1f576ef89d0e99124e45dacaa6d816988e34d80f454d10e96809791d5b78f7fd65f569e90d4d7238c5 -92073c1d11b168e4fa50988b0288638b4868e48bbc668c5a6dddf5499875d53be23a285acb5e4bad60114f6cf6c556e9 -88c87dfcb8ba6cbfe7e1be081ccfadbd589301db2cb7c99f9ee5d7db90aa297ed1538d5a867678a763f2deede5fd219a -b42a562805c661a50f5dea63108002c0f27c0da113da6a9864c9feb5552225417c0356c4209e8e012d9bcc9d182c7611 -8e6317d00a504e3b79cd47feb4c60f9df186467fe9ca0f35b55c0364db30528f5ff071109dabb2fc80bb9cd4949f0c24 -b7b1ea6a88694f8d2f539e52a47466695e39e43a5eb9c6f23bca15305fe52939d8755cc3ac9d6725e60f82f994a3772f -a3cd55161befe795af93a38d33290fb642b8d80da8b786c6e6fb02d393ea308fbe87f486994039cbd7c7b390414594b6 -b416d2d45b44ead3b1424e92c73c2cf510801897b05d1724ff31cbd741920cd858282fb5d6040fe1f0aa97a65bc49424 -950ee01291754feace97c2e933e4681e7ddfbc4fcd079eb6ff830b0e481d929c93d0c7fb479c9939c28ca1945c40da09 -869bd916aee8d86efe362a49010382674825d49195b413b4b4018e88ce43fe091b475d0b863ff0ba2259400f280c2b23 -9782f38cd9c9d3385ec286ebbc7cba5b718d2e65a5890b0a5906b10a89dc8ed80d417d71d7c213bf52f2af1a1f513ea7 -91cd33bc2628d096269b23faf47ee15e14cb7fdc6a8e3a98b55e1031ea0b68d10ba30d97e660f7e967d24436d40fad73 -8becc978129cc96737034c577ae7225372dd855da8811ae4e46328e020c803833b5bdbc4a20a93270e2b8bd1a2feae52 -a36b1d8076783a9522476ce17f799d78008967728ce920531fdaf88303321bcaf97ecaa08e0c01f77bc32e53c5f09525 -b4720e744943f70467983aa34499e76de6d59aa6fadf86f6b787fdce32a2f5b535b55db38fe2da95825c51002cfe142d -91ad21fc502eda3945f6de874d1b6bf9a9a7711f4d61354f9e5634fc73f9c06ada848de15ab0a75811d3250be862827d -84f78e2ebf5fc077d78635f981712daf17e2475e14c2a96d187913006ad69e234746184a51a06ef510c9455b38acb0d7 -960aa7906e9a2f11db64a26b5892ac45f20d2ccb5480f4888d89973beb6fa0dfdc06d68d241ff5ffc7f1b82b1aac242d -a99365dcd1a00c66c9db6924b97c920f5c723380e823b250db85c07631b320ec4e92e586f7319e67a522a0578f7b6d6c -a25d92d7f70cf6a88ff317cfec071e13774516da664f5fac0d4ecaa65b8bf4eb87a64a4d5ef2bd97dfae98d388dbf5cc -a7af47cd0041295798f9779020a44653007444e8b4ef0712982b06d0dcdd434ec4e1f7c5f7a049326602cb605c9105b7 -aefe172eac5568369a05980931cc476bebd9dea573ba276d59b9d8c4420784299df5a910033b7e324a6c2dfc62e3ef05 -b69bc9d22ffa645baa55e3e02522e9892bb2daa7fff7c15846f13517d0799766883ee09ae0869df4139150c5b843ca8a -95a10856140e493354fdd12722c7fdded21b6a2ffbc78aa2697104af8ad0c8e2206f44b0bfee077ef3949d46bbf7c16b -891f2fcd2c47cbea36b7fa715968540c233313f05333f09d29aba23c193f462ed490dd4d00969656e89c53155fdfe710 -a6c33e18115e64e385c843dde34e8a228222795c7ca90bc2cc085705d609025f3351d9be61822c69035a49fb3e48f2d5 -b87fb12f12c0533b005adad0487f03393ff682e13575e3cb57280c3873b2c38ba96a63c49eef7a442753d26b7005230b -b905c02ba451bfd411c135036d92c27af3b0b1c9c2f1309d6948544a264b125f39dd41afeff4666b12146c545adc168a -8b29c513f43a78951cf742231cf5457a6d9d55edf45df5481a0f299a418d94effef561b15d2c1a01d1b8067e7153fda9 -b9941cccd51dc645920d2781c81a317e5a33cb7cf76427b60396735912cb6d2ca9292bb4d36b6392467d390d2c58d9f3 -a8546b627c76b6ef5c93c6a98538d8593dbe21cb7673fd383d5401b0c935eea0bdeeefeb1af6ad41bad8464fb87bbc48 -aa286b27de2812de63108a1aec29d171775b69538dc6198640ac1e96767c2b83a50391f49259195957d457b493b667c9 -a932fb229f641e9abbd8eb2bd874015d97b6658ab6d29769fc23b7db9e41dd4f850382d4c1f08af8f156c5937d524473 -a1412840fcc86e2aeec175526f2fb36e8b3b8d21a78412b7266daf81e51b3f68584ed8bd42a66a43afdd8c297b320520 -89c78be9efb624c97ebca4fe04c7704fa52311d183ffd87737f76b7dadc187c12c982bd8e9ed7cd8beb48cdaafd2fd01 -a3f5ddec412a5bec0ce15e3bcb41c6214c2b05d4e9135a0d33c8e50a78eaba71e0a5a6ea8b45854dec5c2ed300971fc2 -9721f9cec7a68b7758e3887548790de49fa6a442d0396739efa20c2f50352a7f91d300867556d11a703866def2d5f7b5 -a23764e140a87e5991573521af039630dd28128bf56eed2edbed130fd4278e090b60cf5a1dca9de2910603d44b9f6d45 -a1a6494a994215e48ab55c70efa8ffdddce6e92403c38ae7e8dd2f8288cad460c6c7db526bbdf578e96ca04d9fe12797 -b1705ea4cb7e074efe0405fc7b8ee2ec789af0426142f3ec81241cacd4f7edcd88e39435e4e4d8e7b1df64f3880d6613 -85595d061d677116089a6064418b93eb44ff79e68d12bd9625078d3bbc440a60d0b02944eff6054433ee34710ae6fbb4 -9978d5e30bedb7526734f9a1febd973a70bfa20890490e7cc6f2f9328feab1e24f991285dbc3711d892514e2d7d005ad -af30243c66ea43b9f87a061f947f7bce745f09194f6e95f379c7582b9fead920e5d6957eaf05c12ae1282ada4670652f -a1930efb473f88001e47aa0b2b2a7566848cccf295792e4544096ecd14ee5d7927c173a8576b405bfa2eec551cd67eb5 -b0446d1c590ee5a45f7e22d269c044f3848c97aec1d226b44bfd0e94d9729c28a38bccddc3a1006cc5fe4e3c24f001f2 -b8a8380172df3d84b06176df916cf557966d4f2f716d3e9437e415d75b646810f79f2b2b71d857181b7fc944018883a3 -a563afec25b7817bfa26e19dc9908bc00aa8fc3d19be7d6de23648701659009d10e3e4486c28e9c6b13d48231ae29ac5 -a5a8e80579de886fb7d6408f542791876885947b27ad6fa99a8a26e381f052598d7b4e647b0115d4b5c64297e00ce28e -8f87afcc7ad33c51ac719bade3cd92da671a37a82c14446b0a2073f4a0a23085e2c8d31913ed2d0be928f053297de8f6 -a43c455ce377e0bc434386c53c752880687e017b2f5ae7f8a15c044895b242dffde4c92fb8f8bb50b18470b17351b156 -8368f8b12a5bceb1dba25adb3a2e9c7dc9b1a77a1f328e5a693f5aec195cd1e06b0fe9476b554c1c25dac6c4a5b640a3 -919878b27f3671fc78396f11531c032f3e2bd132d04cc234fa4858676b15fb1db3051c0b1db9b4fc49038216f11321ce -b48cd67fb7f1242696c1f877da4bdf188eac676cd0e561fbac1a537f7b8229aff5a043922441d603a26aae56a15faee4 -a3e0fdfd4d29ea996517a16f0370b54787fefe543c2fe73bfc6f9e560c1fd30dad8409859e2d7fa2d44316f24746c712 -8bb156ade8faf149df7bea02c140c7e392a4742ae6d0394d880a849127943e6f26312033336d3b9fdc0092d71b5efe87 -8845e5d5cc555ca3e0523244300f2c8d7e4d02aaebcb5bd749d791208856c209a6f84dd99fd55968c9f0ab5f82916707 -a3e90bb5c97b07789c2f32dff1aec61d0a2220928202f5ad5355ae71f8249237799d6c8a22602e32e572cb12eabe0c17 -b150bcc391884c996149dc3779ce71f15dda63a759ee9cc05871f5a8379dcb62b047098922c0f26c7bd04deb394c33f9 -95cd4ad88d51f0f2efcfd0c2df802fe252bb9704d1afbf9c26a248df22d55da87bdfaf41d7bc6e5df38bd848f0b13f42 -a05a49a31e91dff6a52ac8b9c2cfdd646a43f0d488253f9e3cfbce52f26667166bbb9b608fc358763a65cbf066cd6d05 -a59c3c1227fdd7c2e81f5e11ef5c406da44662987bac33caed72314081e2eed66055d38137e01b2268e58ec85dd986c0 -b7020ec3bd73a99861f0f1d88cf5a19abab1cbe14b7de77c9868398c84bb8e18dbbe9831838a96b6d6ca06e82451c67b -98d1ff2525e9718ee59a21d8900621636fcd873d9a564b8dceb4be80a194a0148daf1232742730b3341514b2e5a5436c -886d97b635975fc638c1b6afc493e5998ca139edba131b75b65cfe5a8e814f11bb678e0eeee5e6e5cd913ad3f2fefdfc -8fb9fd928d38d5d813b671c924edd56601dd7163b686c13f158645c2f869d9250f3859aa5463a39258c90fef0f41190a -aac35e1cd655c94dec3580bb3800bd9c2946c4a9856f7d725af15fbea6a2d8ca51c8ad2772abed60ee0e3fb9cb24046b -b8d71fa0fa05ac9e443c9b4929df9e7f09a919be679692682e614d24227e04894bfc14a5c73a62fb927fedff4a0e4aa7 -a45a19f11fbbb531a704badbb813ed8088ab827c884ee4e4ebf363fa1132ff7cfa9d28be9c85b143e4f7cdbc94e7cf1a -82b54703a4f295f5471b255ab59dce00f0fe90c9fb6e06b9ee48b15c91d43f4e2ef4a96c3118aeb03b08767be58181bb -8283264c8e6d2a36558f0d145c18576b6600ff45ff99cc93eca54b6c6422993cf392668633e5df396b9331e873d457e5 -8c549c03131ead601bc30eb6b9537b5d3beb7472f5bb1bcbbfd1e9f3704477f7840ab3ab7f7dc13bbbbcdff886a462d4 -afbb0c520ac1b5486513587700ad53e314cb74bfbc12e0b5fbdcfdaac36d342e8b59856196a0d84a25cff6e6e1d17e76 -89e4c22ffb51f2829061b3c7c1983c5c750cad158e3a825d46f7cf875677da5d63f653d8a297022b5db5845c9271b32b -afb27a86c4c2373088c96b9adf4433f2ebfc78ac5c526e9f0510670b6e4e5e0057c0a4f75b185e1a30331b9e805c1c15 -a18e16b57445f88730fc5d3567bf5a176861dc14c7a08ed2996fe80eed27a0e7628501bcb78a1727c5e9ac55f29c12c4 -93d61bf88b192d6825cf4e1120af1c17aa0f994d158b405e25437eaeefae049f7b721a206e7cc8a04fdc29d3c42580a1 -a99f2995a2e3ed2fd1228d64166112038de2f516410aa439f4c507044e2017ea388604e2d0f7121256fadf7fbe7023d1 -914fd91cffc23c32f1c6d0e98bf660925090d873367d543034654389916f65f552e445b0300b71b61b721a72e9a5983c -b42a578a7787b71f924e7def425d849c1c777156b1d4170a8ee7709a4a914e816935131afd9a0412c4cb952957b20828 -82fb30590e84b9e45db1ec475a39971cf554dc01bcc7050bc89265740725c02e2be5a972168c5170c86ae83e5b0ad2c0 -b14f8d8e1e93a84976289e0cf0dfa6f3a1809e98da16ee5c4932d0e1ed6bf8a07697fdd4dd86a3df84fb0003353cdcc0 -85d7a2f4bda31aa2cb208b771fe03291a4ebdaf6f1dc944c27775af5caec412584c1f45bc741fca2a6a85acb3f26ad7d -af02e56ce886ff2253bc0a68faad76f25ead84b2144e5364f3fb9b648f03a50ee9dc0b2c33ebacf7c61e9e43201ef9ef -87e025558c8a0b0abd06dfc350016847ea5ced7af2d135a5c9eec9324a4858c4b21510fb0992ec52a73447f24945058e -80fff0bafcd058118f5e7a4d4f1ae0912efeb281d2cbe4d34ba8945cc3dbe5d8baf47fb077343b90b8d895c90b297aca -b6edcf3a40e7b1c3c0148f47a263cd819e585a51ef31c2e35a29ce6f04c53e413f743034c0d998d9c00a08ba00166f31 -abb87ed86098c0c70a76e557262a494ff51a30fb193f1c1a32f8e35eafa34a43fcc07aa93a3b7a077d9e35afa07b1a3d -a280214cd3bb0fb7ecd2d8bcf518cbd9078417f2b91d2533ec2717563f090fb84f2a5fcfdbbeb2a2a1f8a71cc5aa5941 -a63083ca7238ea2b57d15a475963cf1d4f550d8cd76db290014a0461b90351f1f26a67d674c837b0b773b330c7c3d534 -a8fa39064cb585ece5263e2f42f430206476bf261bd50f18d2b694889bd79d04d56410664cecad62690e5c5a20b3f6ff -85ba52ce9d700a5dcf6c5b00559acbe599d671ce5512467ff4b6179d7fad550567ce2a9c126a50964e3096458ea87920 -b913501e1008f076e5eac6d883105174f88b248e1c9801e568fefaffa1558e4909364fc6d9512aa4d125cbd7cc895f05 -8eb33b5266c8f2ed4725a6ad147a322e44c9264cf261c933cbbe230a43d47fca0f29ec39756b20561dabafadd5796494 -850ebc8b661a04318c9db5a0515066e6454fa73865aa4908767a837857ecd717387f614acb614a88e075d4edc53a2f5a -a08d6b92d866270f29f4ce23a3f5d99b36b1e241a01271ede02817c8ec3f552a5c562db400766c07b104a331835c0c64 -8131804c89bb3e74e9718bfc4afa547c1005ff676bd4db9604335032b203390cfa54478d45c6c78d1fe31a436ed4be9f -9106d94f23cc1eacec8316f16d6f0a1cc160967c886f51981fdb9f3f12ee1182407d2bb24e5b873de58cb1a3ee915a6b -a13806bfc3eae7a7000c9d9f1bd25e10218d4e67f59ae798b145b098bca3edad2b1040e3fc1e6310e612fb8818f459ac -8c69fbca502046cb5f6db99900a47b34117aef3f4b241690cdb3b84ca2a2fc7833e149361995dc41fa78892525bce746 -852c473150c91912d58ecb05769222fa18312800c3f56605ad29eec9e2d8667b0b81c379048d3d29100ed2773bb1f3c5 -b1767f6074426a00e01095dbb1795beb4e4050c6411792cbad6537bc444c3165d1058bafd1487451f9c5ddd209e0ae7e -80c600a5fe99354ce59ff0f84c760923dc8ff66a30bf47dc0a086181785ceb01f9b951c4e66df800ea6d705e8bc47055 -b5cf19002fbc88a0764865b82afcb4d64a50196ea361e5c71dff7de084f4dcbbc34ec94a45cc9e0247bd51da565981aa -93e67a254ea8ce25e112d93cc927fadaa814152a2c4ec7d9a56eaa1ed47aec99b7e9916b02e64452cc724a6641729bbb -ace70b32491bda18eee4a4d041c3bc9effae9340fe7e6c2f5ad975ee0874c17f1a7da7c96bd85fccff9312c518fac6e9 -ab4cfa02065017dd7f1aadc66f2c92f78f0f11b8597c03a5d69d82cb2eaf95a4476a836ac102908f137662472c8d914b -a40b8cd8deb8ae503d20364d64cab7c2801b7728a9646ed19c65edea6a842756a2f636283494299584ad57f4bb12cd0b -8594e11d5fc2396bcd9dbf5509ce4816dbb2b7305168021c426171fb444d111da5a152d6835ad8034542277011c26c0e -8024de98c26b4c994a66628dc304bb737f4b6859c86ded552c5abb81fd4c6c2e19d5a30beed398a694b9b2fdea1dd06a -8843f5872f33f54df8d0e06166c1857d733995f67bc54abb8dfa94ad92407cf0179bc91b0a50bbb56cdc2b350d950329 -b8bab44c7dd53ef9edf497dcb228e2a41282c90f00ba052fc52d57e87b5c8ab132d227af1fcdff9a12713d1f980bcaae -982b4d7b29aff22d527fd82d2a52601d95549bfb000429bb20789ed45e5abf1f4b7416c7b7c4b79431eb3574b29be658 -8eb1f571b6a1878e11e8c1c757e0bc084bab5e82e897ca9be9b7f4b47b91679a8190bf0fc8f799d9b487da5442415857 -a6e74b588e5af935c8b243e888582ef7718f8714569dd4992920740227518305eb35fab674d21a5551cca44b3e511ef2 -a30fc2f3a4cb4f50566e82307de73cd7bd8fe2c1184e9293c136a9b9e926a018d57c6e4f308c95b9eb8299e94d90a2a1 -a50c5869ca5d2b40722c056a32f918d47e0b65ca9d7863ca7d2fb4a7b64fe523fe9365cf0573733ceaadebf20b48fff8 -83bbdd32c04d17581418cf360749c7a169b55d54f2427390defd9f751f100897b2d800ce6636c5bbc046c47508d60c8c -a82904bdf614de5d8deaff688c8a5e7ac5b3431687acbcda8fa53960b7c417a39c8b2e462d7af91ce6d79260f412db8e -a4362e31ff4b05d278b033cf5eebea20de01714ae16d4115d04c1da4754269873afc8171a6f56c5104bfd7b0db93c3e7 -b5b8daa63a3735581e74a021b684a1038cea77168fdb7fdf83c670c2cfabcfc3ab2fc7359069b5f9048188351aef26b5 -b48d723894b7782d96ac8433c48faca1bdfa5238019c451a7f47d958097cce3ae599b876cf274269236b9d6ff8b6d7ca -98ffff6a61a3a6205c7820a91ca2e7176fab5dba02bc194c4d14942ac421cb254183c705506ab279e4f8db066f941c6c -ae7db24731da2eaa6efc4f7fcba2ecc26940ddd68038dce43acf2cee15b72dc4ef42a7bfdd32946d1ed78786dd7696b3 -a656db14f1de9a7eb84f6301b4acb2fbf78bfe867f48a270e416c974ab92821eb4df1cb881b2d600cfed0034ac784641 -aa315f8ecba85a5535e9a49e558b15f39520fce5d4bf43131bfbf2e2c9dfccc829074f9083e8d49f405fb221d0bc4c3c -90bffba5d9ff40a62f6c8e9fc402d5b95f6077ed58d030c93e321b8081b77d6b8dac3f63a92a7ddc01585cf2c127d66c -abdd733a36e0e0f05a570d0504e73801bf9b5a25ff2c78786f8b805704997acb2e6069af342538c581144d53149fa6d3 -b4a723bb19e8c18a01bd449b1bb3440ddb2017f10bb153da27deb7a6a60e9bb37619d6d5435fbb1ba617687838e01dd0 -870016b4678bab3375516db0187a2108b2e840bae4d264b9f4f27dbbc7cc9cac1d7dc582d7a04d6fd1ed588238e5e513 -80d33d2e20e8fc170aa3cb4f69fffb72aeafb3b5bb4ea0bc79ab55da14142ca19b2d8b617a6b24d537366e3b49cb67c3 -a7ee76aec273aaae03b3b87015789289551969fb175c11557da3ab77e39ab49d24634726f92affae9f4d24003050d974 -8415ea4ab69d779ebd42d0fe0c6aef531d6a465a5739e429b1fcf433ec45aa8296c527e965a20f0ec9f340c9273ea3cf -8c7662520794e8b4405d0b33b5cac839784bc86a5868766c06cbc1fa306dbe334978177417b31baf90ce7b0052a29c56 -902b2abecc053a3dbdea9897ee21e74821f3a1b98b2d560a514a35799f4680322550fd3a728d4f6d64e1de98033c32b8 -a05e84ed9ecab8d508d670c39f2db61ad6e08d2795ec32a3c9d0d3737ef3801618f4fc2a95f90ec2f068606131e076c5 -8b9208ff4d5af0c2e3f53c9375da666773ac57197dfabb0d25b1c8d0588ba7f3c15ee9661bb001297f322ea2fbf6928b -a3c827741b34a03254d4451b5ab74a96f2b9f7fb069e2f5adaf54fd97cc7a4d516d378db5ca07da87d8566d6eef13726 -8509d8a3f4a0ed378e0a1e28ea02f6bf1d7f6c819c6c2f5297c7df54c895b848f841653e32ba2a2c22c2ff739571acb8 -a0ce988b7d3c40b4e496aa83a09e4b5472a2d98679622f32bea23e6d607bc7de1a5374fb162bce0549a67dad948519be -aa8a3dd12bd60e3d2e05f9c683cdcb8eab17fc59134815f8d197681b1bcf65108cba63ac5c58ee632b1e5ed6bba5d474 -8b955f1d894b3aefd883fb4b65f14cd37fc2b9db77db79273f1700bef9973bf3fd123897ea2b7989f50003733f8f7f21 -ac79c00ddac47f5daf8d9418d798d8af89fc6f1682e7e451f71ea3a405b0d36af35388dd2a332af790bc83ca7b819328 -a0d44dd2a4438b809522b130d0938c3fe7c5c46379365dbd1810a170a9aa5818e1c783470dd5d0b6d4ac7edbb7330910 -a30b69e39ad43dd540a43c521f05b51b5f1b9c4eed54b8162374ae11eac25da4f5756e7b70ce9f3c92c2eeceee7431ed -ac43220b762c299c7951222ea19761ab938bf38e4972deef58ed84f4f9c68c230647cf7506d7cbfc08562fcca55f0485 -b28233b46a8fb424cfa386a845a3b5399d8489ceb83c8f3e05c22c934798d639c93718b7b68ab3ce24c5358339e41cbb -ac30d50ee8ce59a10d4b37a3a35e62cdb2273e5e52232e202ca7d7b8d09d28958ee667fae41a7bb6cdc6fe8f6e6c9c85 -b199842d9141ad169f35cc7ff782b274cbaa645fdb727761e0a89edbf0d781a15f8218b4bf4eead326f2903dd88a9cc1 -85e018c7ddcad34bb8285a737c578bf741ccd547e68c734bdb3808380e12c5d4ef60fc896b497a87d443ff9abd063b38 -8c856e6ba4a815bdb891e1276f93545b7072f6cb1a9aa6aa5cf240976f29f4dee01878638500a6bf1daf677b96b54343 -b8a47555fa8710534150e1a3f13eab33666017be6b41005397afa647ea49708565f2b86b77ad4964d140d9ced6b4d585 -8cd1f1db1b2f4c85a3f46211599caf512d5439e2d8e184663d7d50166fd3008f0e9253272f898d81007988435f715881 -b1f34b14612c973a3eceb716dc102b82ab18afef9de7630172c2780776679a7706a4874e1df3eaadf541fb009731807f -b25464af9cff883b55be2ff8daf610052c02df9a5e147a2cf4df6ce63edcdee6dc535c533590084cc177da85c5dc0baa -91c3c4b658b42d8d3448ae1415d4541d02379a40dc51e36a59bd6e7b9ba3ea51533f480c7c6e8405250ee9b96a466c29 -86dc027b95deb74c36a58a1333a03e63cb5ae22d3b29d114cfd2271badb05268c9d0c819a977f5e0c6014b00c1512e3a -ae0e6ff58eb5fa35da5107ebeacf222ab8f52a22bb1e13504247c1dfa65320f40d97b0e6b201cb6613476687cb2f0681 -8f13415d960b9d7a1d93ef28afc2223e926639b63bdefce0f85e945dfc81670a55df288893a0d8b3abe13c5708f82f91 -956f67ca49ad27c1e3a68c1faad5e7baf0160c459094bf6b7baf36b112de935fdfd79fa4a9ea87ea8de0ac07272969f4 -835e45e4a67df9fb51b645d37840b3a15c171d571a10b03a406dd69d3c2f22df3aa9c5cbe1e73f8d767ce01c4914ea9a -919b938e56d4b32e2667469d0bdccb95d9dda3341aa907683ee70a14bbbe623035014511c261f4f59b318b610ac90aa3 -96b48182121ccd9d689bf1dfdc228175564cd68dc904a99c808a7f0053a6f636c9d953e12198bdf2ea49ea92772f2e18 -ac5e5a941d567fa38fdbcfa8cf7f85bb304e3401c52d88752bcd516d1fa9bac4572534ea2205e38423c1df065990790f -ac0bd594fb85a8d4fc26d6df0fa81f11919401f1ecf9168b891ec7f061a2d9368af99f7fd8d9b43b2ce361e7b8482159 -83d92c69ca540d298fe80d8162a1c7af3fa9b49dfb69e85c1d136a3ec39fe419c9fa78e0bb6d96878771fbd37fe92e40 -b35443ae8aa66c763c2db9273f908552fe458e96696b90e41dd509c17a5c04ee178e3490d9c6ba2dc0b8f793c433c134 -923b2d25aa45b2e580ffd94cbb37dc8110f340f0f011217ee1bd81afb0714c0b1d5fb4db86006cdd2457563276f59c59 -96c9125d38fca1a61ac21257b696f8ac3dae78def50285e44d90ea293d591d1c58f703540a7e4e99e070afe4646bbe15 -b57946b2332077fbcdcb406b811779aefd54473b5559a163cd65cb8310679b7e2028aa55c12a1401fdcfcac0e6fae29a -845daedc5cf972883835d7e13c937b63753c2200324a3b8082a6c4abb4be06c5f7c629d4abe4bfaf1d80a1f073eb6ce6 -91a55dfd0efefcd03dc6dacc64ec93b8d296cb83c0ee72400a36f27246e7f2a60e73b7b70ba65819e9cfb73edb7bd297 -8874606b93266455fe8fdd25df9f8d2994e927460af06f2e97dd4d2d90db1e6b06d441b72c2e76504d753badca87fb37 -8ee99e6d231274ff9252c0f4e84549da173041299ad1230929c3e3d32399731c4f20a502b4a307642cac9306ccd49d3c -8836497714a525118e20849d6933bb8535fb6f72b96337d49e3133d936999c90a398a740f42e772353b5f1c63581df6d -a6916945e10628f7497a6cdc5e2de113d25f7ade3e41e74d3de48ccd4fce9f2fa9ab69645275002e6f49399b798c40af -9597706983107eb23883e0812e1a2c58af7f3499d50c6e29b455946cb9812fde1aa323d9ed30d1c0ffd455abe32303cd -a24ee89f7f515cc33bdbdb822e7d5c1877d337f3b2162303cfc2dae028011c3a267c5cb4194afa63a4856a6e1c213448 -8cd25315e4318801c2776824ae6e7d543cb85ed3bc2498ba5752df2e8142b37653cf9e60104d674be3aeb0a66912e97a -b5085ecbe793180b40dbeb879f4c976eaaccaca3a5246807dced5890e0ed24d35f3f86955e2460e14fb44ff5081c07ba -960188cc0b4f908633a6840963a6fa2205fc42c511c6c309685234911c5304ef4c304e3ae9c9c69daa2fb6a73560c256 -a32d0a70bf15d569b4cda5aebe3e41e03c28bf99cdd34ffa6c5d58a097f322772acca904b3a47addb6c7492a7126ebac -977f72d06ad72d4aa4765e0f1f9f4a3231d9f030501f320fe7714cc5d329d08112789fa918c60dd7fdb5837d56bb7fc6 -99fa038bb0470d45852bb871620d8d88520adb701712fcb1f278fed2882722b9e729e6cdce44c82caafad95e37d0e6f7 -b855e8f4fc7634ada07e83b6c719a1e37acb06394bc8c7dcab7747a8c54e5df3943915f021364bd019fdea103864e55f -88bc2cd7458532e98c596ef59ea2cf640d7cc31b4c33cef9ed065c078d1d4eb49677a67de8e6229cc17ea48bace8ee5a -aaa78a3feaa836d944d987d813f9b9741afb076e6aca1ffa42682ab06d46d66e0c07b8f40b9dbd63e75e81efa1ef7b08 -b7b080420cc4d808723b98b2a5b7b59c81e624ab568ecdfdeb8bf3aa151a581b6f56e983ef1b6f909661e25db40b0c69 -abee85c462ac9a2c58e54f06c91b3e5cd8c5f9ab5b5deb602b53763c54826ed6deb0d6db315a8d7ad88733407e8d35e2 -994d075c1527407547590df53e9d72dd31f037c763848d1662eebd4cefec93a24328c986802efa80e038cb760a5300f5 -ab8777640116dfb6678e8c7d5b36d01265dfb16321abbfc277da71556a34bb3be04bc4ae90124ed9c55386d2bfb3bda0 -967e3a828bc59409144463bcf883a3a276b5f24bf3cbfdd7a42343348cba91e00b46ac285835a9b91eef171202974204 -875a9f0c4ffe5bb1d8da5e3c8e41d0397aa6248422a628bd60bfae536a651417d4e8a7d2fb98e13f2dad3680f7bd86d3 -acaa330c3e8f95d46b1880126572b238dbb6d04484d2cd4f257ab9642d8c9fc7b212188b9c7ac9e0fd135c520d46b1bf -aceb762edbb0f0c43dfcdb01ea7a1ac5918ca3882b1e7ebc4373521742f1ed5250d8966b498c00b2b0f4d13212e6dd0b -81d072b4ad258b3646f52f399bced97c613b22e7ad76373453d80b1650c0ca87edb291a041f8253b649b6e5429bb4cff -980a47d27416ac39c7c3a0ebe50c492f8c776ea1de44d5159ac7d889b6d554357f0a77f0e5d9d0ff41aae4369eba1fc2 -8b4dfd5ef5573db1476d5e43aacfb5941e45d6297794508f29c454fe50ea622e6f068b28b3debe8635cf6036007de2e3 -a60831559d6305839515b68f8c3bc7abbd8212cc4083502e19dd682d56ca37c9780fc3ce4ec2eae81ab23b221452dc57 -951f6b2c1848ced9e8a2339c65918e00d3d22d3e59a0a660b1eca667d18f8430d737884e9805865ef3ed0fe1638a22d9 -b02e38fe790b492aa5e89257c4986c9033a8b67010fa2add9787de857d53759170fdd67715ca658220b4e14b0ca48124 -a51007e4346060746e6b0e4797fc08ef17f04a34fe24f307f6b6817edbb8ce2b176f40771d4ae8a60d6152cbebe62653 -a510005b05c0b305075b27b243c9d64bcdce85146b6ed0e75a3178b5ff9608213f08c8c9246f2ca6035a0c3e31619860 -aaff4ef27a7a23be3419d22197e13676d6e3810ceb06a9e920d38125745dc68a930f1741c9c2d9d5c875968e30f34ab5 -864522a9af9857de9814e61383bebad1ba9a881696925a0ea6bfc6eff520d42c506bbe5685a9946ed710e889765be4a0 -b63258c080d13f3b7d5b9f3ca9929f8982a6960bdb1b0f8676f4dca823971601672f15e653917bf5d3746bb220504913 -b51ce0cb10869121ae310c7159ee1f3e3a9f8ad498827f72c3d56864808c1f21fa2881788f19ece884d3f705cd7bd0c5 -95d9cecfc018c6ed510e441cf84c712d9909c778c16734706c93222257f64dcd2a9f1bd0b400ca271e22c9c487014274 -8beff4d7d0140b86380ff4842a9bda94c2d2be638e20ac68a4912cb47dbe01a261857536375208040c0554929ced1ddc -891ff49258749e2b57c1e9b8e04b12c77d79c3308b1fb615a081f2aacdfb4b39e32d53e069ed136fdbd43c53b87418fa -9625cad224e163d387738825982d1e40eeff35fe816d10d7541d15fdc4d3eee48009090f3faef4024b249205b0b28f72 -8f3947433d9bd01aa335895484b540a9025a19481a1c40b4f72dd676bfcf332713714fd4010bde936eaf9470fd239ed0 -a00ec2d67789a7054b53f0e858a8a232706ccc29a9f3e389df7455f1a51a2e75801fd78469a13dbc25d28399ae4c6182 -a3f65884506d4a62b8775a0ea0e3d78f5f46bc07910a93cd604022154eabdf1d73591e304d61edc869e91462951975e1 -a14eef4fd5dfac311713f0faa9a60415e3d30b95a4590cbf95f2033dffb4d16c02e7ceff3dcd42148a4e3bc49cce2dd4 -8afa11c0eef3c540e1e3460bc759bb2b6ea90743623f88e62950c94e370fe4fd01c22b6729beba4dcd4d581198d9358f -afb05548a69f0845ffcc5f5dc63e3cdb93cd270f5655173b9a950394b0583663f2b7164ba6df8d60c2e775c1d9f120af -97f179e01a947a906e1cbeafa083960bc9f1bade45742a3afee488dfb6011c1c6e2db09a355d77f5228a42ccaa7bdf8e -8447fca4d35f74b3efcbd96774f41874ca376bf85b79b6e66c92fa3f14bdd6e743a051f12a7fbfd87f319d1c6a5ce217 -a57ca39c23617cd2cf32ff93b02161bd7baf52c4effb4679d9d5166406e103bc8f3c6b5209e17c37dbb02deb8bc72ddd -9667c7300ff80f0140be002b0e36caab07aaee7cce72679197c64d355e20d96196acaf54e06e1382167d081fe6f739c1 -828126bb0559ce748809b622677267ca896fa2ee76360fd2c02990e6477e06a667241379ca7e65d61a5b64b96d7867de -8b8835dea6ba8cf61c91f01a4b3d2f8150b687a4ee09b45f2e5fc8f80f208ae5d142d8e3a18153f0722b90214e60c5a7 -a98e8ff02049b4da386e3ee93db23bbb13dfeb72f1cfde72587c7e6d962780b7671c63e8ac3fbaeb1a6605e8d79e2f29 -87a4892a0026d7e39ef3af632172b88337cb03669dea564bcdb70653b52d744730ebb5d642e20cb627acc9dbb547a26b -877352a22fc8052878a57effc159dac4d75fe08c84d3d5324c0bab6d564cdf868f33ceee515eee747e5856b62cfa0cc7 -8b801ba8e2ff019ee62f64b8cb8a5f601fc35423eb0f9494b401050103e1307dc584e4e4b21249cd2c686e32475e96c3 -a9e7338d6d4d9bfec91b2af28a8ed13b09415f57a3a00e5e777c93d768fdb3f8e4456ae48a2c6626b264226e911a0e28 -99c05fedf40ac4726ed585d7c1544c6e79619a0d3fb6bda75a08c7f3c0008e8d5e19ed4da48de3216135f34a15eba17c -a61cce8a1a8b13a4a650fdbec0eeea8297c352a8238fb7cac95a0df18ed16ee02a3daa2de108fa122aca733bd8ad7855 -b97f37da9005b440b4cb05870dd881bf8491fe735844f2d5c8281818583b38e02286e653d9f2e7fa5e74c3c3eb616540 -a72164a8554da8e103f692ac5ebb4aece55d5194302b9f74b6f2a05335b6e39beede0bf7bf8c5bfd4d324a784c5fb08c -b87e8221c5341cd9cc8bb99c10fe730bc105550f25ed4b96c0d45e6142193a1b2e72f1b3857373a659b8c09be17b3d91 -a41fb1f327ef91dcb7ac0787918376584890dd9a9675c297c45796e32d6e5985b12f9b80be47fc3a8596c245f419d395 -90dafa3592bdbb3465c92e2a54c2531822ba0459d45d3e7a7092fa6b823f55af28357cb51896d4ec2d66029c82f08e26 -a0a9adc872ebc396557f484f1dd21954d4f4a21c4aa5eec543f5fa386fe590839735c01f236574f7ff95407cd12de103 -b8c5c940d58be7538acf8672852b5da3af34f82405ef2ce8e4c923f1362f97fc50921568d0fd2fe846edfb0823e62979 -85aaf06a8b2d0dac89dafd00c28533f35dbd074978c2aaa5bef75db44a7b12aeb222e724f395513b9a535809a275e30b -81f3cbe82fbc7028c26a6c1808c604c63ba023a30c9f78a4c581340008dbda5ec07497ee849a2183fcd9124f7936af32 -a11ac738de75fd60f15a34209d3825d5e23385796a4c7fc5931822f3f380af977dd0f7b59fbd58eed7777a071e21b680 -85a279c493de03db6fa6c3e3c1b1b29adc9a8c4effc12400ae1128da8421954fa8b75ad19e5388fe4543b76fb0812813 -83a217b395d59ab20db6c4adb1e9713fc9267f5f31a6c936042fe051ce8b541f579442f3dcf0fa16b9e6de9fd3518191 -83a0b86e7d4ed8f9ccdc6dfc8ff1484509a6378fa6f09ed908e6ab9d1073f03011dc497e14304e4e3d181b57de06a5ab -a63ad69c9d25704ce1cc8e74f67818e5ed985f8f851afa8412248b2df5f833f83b95b27180e9e7273833ed0d07113d3b -99b1bc2021e63b561fe44ddd0af81fcc8627a91bfeecbbc989b642bc859abc0c8d636399701aad7bbaf6a385d5f27d61 -b53434adb66f4a807a6ad917c6e856321753e559b1add70824e5c1e88191bf6993fccb9b8b911fc0f473fb11743acacd -97ed3b9e6fb99bf5f945d4a41f198161294866aa23f2327818cdd55cb5dc4c1a8eff29dd8b8d04902d6cd43a71835c82 -b1e808260e368a18d9d10bdea5d60223ba1713b948c782285a27a99ae50cc5fc2c53d407de07155ecc16fb8a36d744a0 -a3eb4665f18f71833fec43802730e56b3ee5a357ea30a888ad482725b169d6f1f6ade6e208ee081b2e2633079b82ba7d -ab8beb2c8353fc9f571c18fdd02bdb977fc883313469e1277b0372fbbb33b80dcff354ca41de436d98d2ed710faa467e -aa9071cfa971e4a335a91ad634c98f2be51544cb21f040f2471d01bb97e1df2277ae1646e1ea8f55b7ba9f5c8c599b39 -80b7dbfdcaf40f0678012acc634eba44ea51181475180d9deb2050dc4f2de395289edd0223018c81057ec79b04b04c49 -89623d7f6cb17aa877af14de842c2d4ab7fd576d61ddd7518b5878620a01ded40b6010de0da3cdf31d837eecf30e9847 -a773bb024ae74dd24761f266d4fb27d6fd366a8634febe8235376b1ae9065c2fe12c769f1d0407867dfbe9f5272c352f -8455a561c3aaa6ba64c881a5e13921c592b3a02e968f4fb24a2243c36202795d0366d9cc1a24e916f84d6e158b7aeac7 -81d8bfc4b283cf702a40b87a2b96b275bdbf0def17e67d04842598610b67ea08c804d400c3e69fa09ea001eaf345b276 -b8f8f82cb11fea1c99467013d7e167ff03deb0c65a677fab76ded58826d1ba29aa7cf9fcd7763615735ea3ad38e28719 -89a6a04baf9cccc1db55179e1650b1a195dd91fb0aebc197a25143f0f393524d2589975e3fbfc2547126f0bced7fd6f2 -b81b2162df045390f04df07cbd0962e6b6ca94275a63edded58001a2f28b2ae2af2c7a6cba4ecd753869684e77e7e799 -a3757f722776e50de45c62d9c4a2ee0f5655a512344c4cbec542d8045332806568dd626a719ef21a4eb06792ca70f204 -8c5590df96ec22179a4e8786de41beb44f987a1dcc508eb341eecbc0b39236fdfad47f108f852e87179ccf4e10091e59 -87502f026ed4e10167419130b88c3737635c5b9074c364e1dd247cef5ef0fc064b4ae99b187e33301e438bbd2fe7d032 -af925a2165e980ced620ff12289129fe17670a90ae0f4db9d4b39bd887ccb1f5d2514ac9ecf910f6390a8fc66bd5be17 -857fca899828cf5c65d26e3e8a6e658542782fc72762b3b9c73514919f83259e0f849a9d4838b40dc905fe43024d0d23 -87ffebdbfb69a9e1007ebac4ffcb4090ff13705967b73937063719aa97908986effcb7262fdadc1ae0f95c3690e3245d -a9ff6c347ac6f4c6ab993b748802e96982eaf489dc69032269568412fc9a79e7c2850dfc991b28211b3522ee4454344b -a65b3159df4ec48bebb67cb3663cd744027ad98d970d620e05bf6c48f230fa45bf17527fe726fdf705419bb7a1bb913e -84b97b1e6408b6791831997b03cd91f027e7660fd492a93d95daafe61f02427371c0e237c75706412f442991dfdff989 -ab761c26527439b209af0ae6afccd9340bbed5fbe098734c3145b76c5d2cd7115d9227b2eb523882b7317fbb09180498 -a0479a8da06d7a69c0b0fee60df4e691c19c551f5e7da286dab430bfbcabf31726508e20d26ea48c53365a7f00a3ad34 -a732dfc9baa0f4f40b5756d2e8d8937742999623477458e0bc81431a7b633eefc6f53b3b7939fe0a020018549c954054 -901502436a1169ba51dc479a5abe7c8d84e0943b16bc3c6a627b49b92cd46263c0005bc324c67509edd693f28e612af1 -b627aee83474e7f84d1bab9b7f6b605e33b26297ac6bbf52d110d38ba10749032bd551641e73a383a303882367af429b -95108866745760baef4a46ef56f82da6de7e81c58b10126ebd2ba2cd13d339f91303bf2fb4dd104a6956aa3b13739503 -899ed2ade37236cec90056f3569bc50f984f2247792defafcceb49ad0ca5f6f8a2f06573705300e07f0de0c759289ff5 -a9f5eee196d608efe4bcef9bf71c646d27feb615e21252cf839a44a49fd89da8d26a758419e0085a05b1d59600e2dc42 -b36c6f68fed6e6c85f1f4a162485f24817f2843ec5cbee45a1ebfa367d44892e464949c6669f7972dc7167af08d55d25 -aaaede243a9a1b6162afbc8f571a52671a5a4519b4062e3f26777664e245ba873ed13b0492c5dbf0258c788c397a0e9e -972b4fb39c31cbe127bf9a32a5cc10d621ebdd9411df5e5da3d457f03b2ab2cd1f6372d8284a4a9400f0b06ecdbfd38e -8f6ca1e110e959a4b1d9a5ce5f212893cec21db40d64d5ac4d524f352d72198f923416a850bf845bc5a22a79c0ea2619 -a0f3c93b22134f66f04b2553a53b738644d1665ceb196b8494b315a4c28236fb492017e4a0de4224827c78e42f9908b7 -807fb5ee74f6c8735b0b5ca07e28506214fe4047dbeb00045d7c24f7849e98706aea79771241224939cb749cf1366c7d -915eb1ff034224c0b645442cdb7d669303fdc00ca464f91aaf0b6fde0b220a3a74ff0cb043c26c9f3a5667b3fdaa9420 -8fda6cef56ed33fefffa9e6ac8e6f76b1af379f89761945c63dd448801f7bb8ca970504a7105fac2f74f652ccff32327 -87380cffdcffb1d0820fa36b63cc081e72187f86d487315177d4d04da4533eb19a0e2ff6115ceab528887819c44a5164 -8cd89e03411a18e7f16f968b89fb500c36d47d229f6487b99e62403a980058db5925ce249206743333538adfad168330 -974451b1df33522ce7056de9f03e10c70bf302c44b0741a59df3d6877d53d61a7394dcee1dd46e013d7cb9d73419c092 -98c35ddf645940260c490f384a49496a7352bb8e3f686feed815b1d38f59ded17b1ad6e84a209e773ed08f7b8ff1e4c2 -963f386cf944bb9b2ddebb97171b64253ea0a2894ac40049bdd86cda392292315f3a3d490ca5d9628c890cfb669f0acb -8d507712152babd6d142ee682638da8495a6f3838136088df9424ef50d5ec28d815a198c9a4963610b22e49b4cdf95e9 -83d4bc6b0be87c8a4f1e9c53f257719de0c73d85b490a41f7420e777311640937320557ff2f1d9bafd1daaa54f932356 -82f5381c965b7a0718441131c4d13999f4cdce637698989a17ed97c8ea2e5bdb5d07719c5f7be8688edb081b23ede0f4 -a6ebecab0b72a49dfd01d69fa37a7f74d34fb1d4fef0aa10e3d6fceb9eccd671225c230af89f6eb514250e41a5f91f52 -846d185bdad6e11e604df7f753b7a08a28b643674221f0e750ebdb6b86ec584a29c869e131bca868972a507e61403f6a -85a98332292acb744bd1c0fd6fdcf1f889a78a2c9624d79413ffa194cc8dfa7821a4b60cde8081d4b5f71f51168dd67f -8f7d97c3b4597880d73200d074eb813d95432306e82dafc70b580b8e08cb8098b70f2d07b4b3ac6a4d77e92d57035031 -8185439c8751e595825d7053518cbe121f191846a38d4dbcb558c3f9d7a3104f3153401adaaaf27843bbe2edb504bfe3 -b3c00d8ece1518fca6b1215a139b0a0e26d9cba1b3a424f7ee59f30ce800a5db967279ed60958dd1f3ee69cf4dd1b204 -a2e6cb6978e883f9719c3c0d44cfe8de0cc6f644b98f98858433bea8bbe7b612c8aca5952fccce4f195f9d54f9722dc2 -99663087e3d5000abbec0fbda4e7342ec38846cc6a1505191fb3f1a337cb369455b7f8531a6eb8b0f7b2c4baf83cbe2b -ab0836c6377a4dbc7ca6a4d6cf021d4cd60013877314dd05f351706b128d4af6337711ed3443cb6ca976f40d74070a9a -87abfd5126152fd3bac3c56230579b489436755ea89e0566aa349490b36a5d7b85028e9fb0710907042bcde6a6f5d7e3 -974ba1033f75f60e0cf7c718a57ae1da3721cf9d0fb925714c46f027632bdd84cd9e6de4cf4d00bc55465b1c5ebb7384 -a607b49d73689ac64f25cec71221d30d53e781e1100d19a2114a21da6507a60166166369d860bd314acb226596525670 -a7c2b0b915d7beba94954f2aa7dd08ec075813661e2a3ecca5d28a0733e59583247fed9528eb28aba55b972cdbaf06eb -b8b3123e44128cc8efbe3270f2f94e50ca214a4294c71c3b851f8cbb70cb67fe9536cf07d04bf7fe380e5e3a29dd3c15 -a59a07e343b62ad6445a0859a32b58c21a593f9ddbfe52049650f59628c93715aa1f4e1f45b109321756d0eeec8a5429 -94f51f8a4ed18a6030d0aaa8899056744bd0e9dc9ac68f62b00355cddab11da5da16798db75f0bfbce0e5bdfe750c0b6 -97460a97ca1e1fa5ce243b81425edc0ec19b7448e93f0b55bc9785eedeeafe194a3c8b33a61a5c72990edf375f122777 -8fa859a089bc17d698a7ee381f37ce9beadf4e5b44fce5f6f29762bc04f96faff5d58c48c73631290325f05e9a1ecf49 -abdf38f3b20fc95eff31de5aa9ef1031abfa48f1305ee57e4d507594570401503476d3bcc493838fc24d6967a3082c7f -b8914bfb82815abb86da35c64d39ab838581bc0bf08967192697d9663877825f2b9d6fbdcf9b410463482b3731361aef -a8187f9d22b193a5f578999954d6ec9aa9b32338ccadb8a3e1ce5bad5ea361d69016e1cdfac44e9d6c54e49dd88561b9 -aac262cb7cba7fd62c14daa7b39677cabc1ef0947dd06dd89cac8570006a200f90d5f0353e84f5ff03179e3bebe14231 -a630ef5ece9733b8c46c0a2df14a0f37647a85e69c63148e79ffdcc145707053f9f9d305c3f1cf3c7915cb46d33abd07 -b102c237cb2e254588b6d53350dfda6901bd99493a3fbddb4121d45e0b475cf2663a40d7b9a75325eda83e4ba1e68cb3 -86a930dd1ddcc16d1dfa00aa292cb6c2607d42c367e470aa920964b7c17ab6232a7108d1c2c11fc40fb7496547d0bbf8 -a832fdc4500683e72a96cce61e62ac9ee812c37fe03527ad4cf893915ca1962cee80e72d4f82b20c8fc0b764376635a1 -88ad985f448dabb04f8808efd90f273f11f5e6d0468b5489a1a6a3d77de342992a73eb842d419034968d733f101ff683 -98a8538145f0d86f7fbf9a81c9140f6095c5bdd8960b1c6f3a1716428cd9cca1bf8322e6d0af24e6169abcf7df2b0ff6 -9048c6eba5e062519011e177e955a200b2c00b3a0b8615bdecdebc217559d41058d3315f6d05617be531ef0f6aef0e51 -833bf225ab6fc68cdcacf1ec1b50f9d05f5410e6cdcd8d56a3081dc2be8a8d07b81534d1ec93a25c2e270313dfb99e3b -a84bcd24c3da5e537e64a811b93c91bfc84d7729b9ead7f79078989a6eb76717d620c1fad17466a0519208651e92f5ff -b7cdd0a3fbd79aed93e1b5a44ca44a94e7af5ed911e4492f332e3a5ed146c7286bde01b52276a2fcc02780d2109874dd -8a19a09854e627cb95750d83c20c67442b66b35896a476358f993ba9ac114d32c59c1b3d0b8787ee3224cf3888b56c64 -a9abd5afb8659ee52ada8fa5d57e7dd355f0a7350276f6160bec5fbf70d5f99234dd179eb221c913e22a49ec6d267846 -8c13c4274c0d30d184e73eaf812200094bbbd57293780bdadbceb262e34dee5b453991e7f37c7333a654fc71c69d6445 -a4320d73296ff8176ce0127ca1921c450e2a9c06eff936681ebaffb5a0b05b17fded24e548454de89aca2dcf6d7a9de4 -b2b8b3e15c1f645f07783e5628aba614e60157889db41d8161d977606788842b67f83f361eae91815dc0abd84e09abd5 -ad26c3aa35ddfddc15719b8bb6c264aaec7065e88ac29ba820eb61f220fef451609a7bb037f3722d022e6c86e4f1dc88 -b8615bf43e13ae5d7b8dd903ce37190800cd490f441c09b22aa29d7a29ed2c0417b7a08ead417868f1de2589deaadd80 -8d3425e1482cd1e76750a76239d33c06b3554c3c3c87c15cb7ab58b1cee86a4c5c4178b44e23f36928365a1b484bde02 -806893a62e38c941a7dd6f249c83af16596f69877cc737d8f73f6b8cd93cbc01177a7a276b2b8c6b0e5f2ad864db5994 -86618f17fa4b0d65496b661bbb5ba3bc3a87129d30a4b7d4f515b904f4206ca5253a41f49fd52095861e5e065ec54f21 -9551915da1304051e55717f4c31db761dcdcf3a1366c89a4af800a9e99aca93a357bf928307f098e62b44a02cb689a46 -8f79c4ec0ec1146cb2a523b52fe33def90d7b5652a0cb9c2d1c8808a32293e00aec6969f5b1538e3a94cd1efa3937f86 -a0c03e329a707300081780f1e310671315b4c6a4cedcb29697aedfabb07a9d5df83f27b20e9c44cf6b16e39d9ded5b98 -86a7cfa7c8e7ce2c01dd0baec2139e97e8e090ad4e7b5f51518f83d564765003c65968f85481bbb97cb18f005ccc7d9f -a33811770c6dfda3f7f74e6ad0107a187fe622d61b444bbd84fd7ef6e03302e693b093df76f6ab39bb4e02afd84a575a -85480f5c10d4162a8e6702b5e04f801874d572a62a130be94b0c02b58c3c59bdcd48cd05f0a1c2839f88f06b6e3cd337 -8e181011564b17f7d787fe0e7f3c87f6b62da9083c54c74fd6c357a1f464c123c1d3d8ade3cf72475000b464b14e2be3 -8ee178937294b8c991337e0621ab37e9ffa4ca2bdb3284065c5e9c08aad6785d50cf156270ff9daf9a9127289710f55b -8bd1e8e2d37379d4b172f1aec96f2e41a6e1393158d7a3dbd9a95c8dd4f8e0b05336a42efc11a732e5f22b47fc5c271d -8f3da353cd487c13136a85677de8cedf306faae0edec733cf4f0046f82fa4639db4745b0095ff33a9766aba50de0cbcf -8d187c1e97638df0e4792b78e8c23967dac43d98ea268ca4aabea4e0fa06cb93183fd92d4c9df74118d7cc27bf54415e -a4c992f08c2f8bac0b74b3702fb0c75c9838d2ce90b28812019553d47613c14d8ce514d15443159d700b218c5a312c49 -a6fd1874034a34c3ea962a316c018d9493d2b3719bb0ec4edbc7c56b240802b2228ab49bee6f04c8a3e9f6f24a48c1c2 -b2efed8e799f8a15999020900dc2c58ece5a3641c90811b86a5198e593d7318b9d53b167818ccdfbe7df2414c9c34011 -995ff7de6181ddf95e3ead746089c6148da3508e4e7a2323c81785718b754d356789b902e7e78e2edc6b0cbd4ff22c78 -944073d24750a9068cbd020b834afc72d2dde87efac04482b3287b40678ad07588519a4176b10f2172a2c463d063a5cd -99db4b1bb76475a6fd75289986ef40367960279524378cc917525fb6ba02a145a218c1e9caeb99332332ab486a125ac0 -89fce4ecd420f8e477af4353b16faabb39e063f3f3c98fde2858b1f2d1ef6eed46f0975a7c08f233b97899bf60ccd60a -8c09a4f07a02b80654798bc63aada39fd638d3e3c4236ccd8a5ca280350c31e4a89e5f4c9aafb34116e71da18c1226b8 -85325cfa7ded346cc51a2894257eab56e7488dbff504f10f99f4cd2b630d913003761a50f175ed167e8073f1b6b63fb0 -b678b4fbec09a8cc794dcbca185f133578f29e354e99c05f6d07ac323be20aecb11f781d12898168e86f2e0f09aca15e -a249cfcbca4d9ba0a13b5f6aac72bf9b899adf582f9746bb2ad043742b28915607467eb794fca3704278f9136f7642be -9438e036c836a990c5e17af3d78367a75b23c37f807228362b4d13e3ddcb9e431348a7b552d09d11a2e9680704a4514f -925ab70450af28c21a488bfb5d38ac994f784cf249d7fd9ad251bb7fd897a23e23d2528308c03415074d43330dc37ef4 -a290563904d5a8c0058fc8330120365bdd2ba1fdbaef7a14bc65d4961bb4217acfaed11ab82669e359531f8bf589b8db -a7e07a7801b871fc9b981a71e195a3b4ba6b6313bc132b04796a125157e78fe5c11a3a46cf731a255ac2d78a4ae78cd0 -b26cd2501ee72718b0eebab6fb24d955a71f363f36e0f6dff0ab1d2d7836dab88474c0cef43a2cc32701fca7e82f7df3 -a1dc3b6c968f3de00f11275092290afab65b2200afbcfa8ddc70e751fa19dbbc300445d6d479a81bda3880729007e496 -a9bc213e28b630889476a095947d323b9ac6461dea726f2dc9084473ae8e196d66fb792a21905ad4ec52a6d757863e7d -b25d178df8c2df8051e7c888e9fa677fde5922e602a95e966db9e4a3d6b23ce043d7dc48a5b375c6b7c78e966893e8c3 -a1c8d88d72303692eaa7adf68ea41de4febec40cc14ae551bb4012afd786d7b6444a3196b5d9d5040655a3366d96b7cd -b22bd44f9235a47118a9bbe2ba5a2ba9ec62476061be2e8e57806c1a17a02f9a51403e849e2e589520b759abd0117683 -b8add766050c0d69fe81d8d9ea73e1ed05f0135d093ff01debd7247e42dbb86ad950aceb3b50b9af6cdc14ab443b238f -af2cf95f30ef478f018cf81d70d47d742120b09193d8bb77f0d41a5d2e1a80bfb467793d9e2471b4e0ad0cb2c3b42271 -8af5ef2107ad284e246bb56e20fef2a255954f72de791cbdfd3be09f825298d8466064f3c98a50496c7277af32b5c0bc -85dc19558572844c2849e729395a0c125096476388bd1b14fa7f54a7c38008fc93e578da3aac6a52ff1504d6ca82db05 -ae8c9b43c49572e2e166d704caf5b4b621a3b47827bb2a3bcd71cdc599bba90396fd9a405261b13e831bb5d44c0827d7 -a7ba7efede25f02e88f6f4cbf70643e76784a03d97e0fbd5d9437c2485283ad7ca3abb638a5f826cd9f6193e5dec0b6c -94a9d122f2f06ef709fd8016fd4b712d88052245a65a301f5f177ce22992f74ad05552b1f1af4e70d1eac62cef309752 -82d999b3e7cf563833b8bc028ff63a6b26eb357dfdb3fd5f10e33a1f80a9b2cfa7814d871b32a7ebfbaa09e753e37c02 -aec6edcde234df502a3268dd2c26f4a36a2e0db730afa83173f9c78fcb2b2f75510a02b80194327b792811caefda2725 -94c0bfa66c9f91d462e9194144fdd12d96f9bbe745737e73bab8130607ee6ea9d740e2cfcbbd00a195746edb6369ee61 -ab7573dab8c9d46d339e3f491cb2826cabe8b49f85f1ede78d845fc3995537d1b4ab85140b7d0238d9c24daf0e5e2a7e -87e8b16832843251fe952dadfd01d41890ed4bb4b8fa0254550d92c8cced44368225eca83a6c3ad47a7f81ff8a80c984 -9189d2d9a7c64791b19c0773ad4f0564ce6bea94aa275a917f78ad987f150fdb3e5e26e7fef9982ac184897ecc04683f -b3661bf19e2da41415396ae4dd051a9272e8a2580b06f1a1118f57b901fa237616a9f8075af1129af4eabfefedbe2f1c -af43c86661fb15daf5d910a4e06837225e100fb5680bd3e4b10f79a2144c6ec48b1f8d6e6b98e067d36609a5d038889a -82ac0c7acaa83ddc86c5b4249aae12f28155989c7c6b91e5137a4ce05113c6cbc16f6c44948b0efd8665362d3162f16a -8f268d1195ab465beeeb112cd7ffd5d5548559a8bc01261106d3555533fc1971081b25558d884d552df0db1cddda89d8 -8ef7caa5521f3e037586ce8ac872a4182ee20c7921c0065ed9986c047e3dda08294da1165f385d008b40d500f07d895f -8c2f98f6880550573fad46075d3eba26634b5b025ce25a0b4d6e0193352c8a1f0661064027a70fe8190b522405f9f4e3 -b7653f353564feb164f0f89ec7949da475b8dad4a4d396d252fc2a884f6932d027b7eb2dc4d280702c74569319ed701a -a026904f4066333befd9b87a8fad791d014096af60cdd668ef919c24dbe295ff31f7a790e1e721ba40cf5105abca67f4 -988f982004ada07a22dd345f2412a228d7a96b9cae2c487de42e392afe1e35c2655f829ce07a14629148ce7079a1f142 -9616add009067ed135295fb74d5b223b006b312bf14663e547a0d306694ff3a8a7bb9cfc466986707192a26c0bce599f -ad4c425de9855f6968a17ee9ae5b15e0a5b596411388cf976df62ecc6c847a6e2ddb2cea792a5f6e9113c2445dba3e5c -b698ac9d86afa3dc69ff8375061f88e3b0cff92ff6dfe747cebaf142e813c011851e7a2830c10993b715e7fd594604a9 -a386fa189847bb3b798efca917461e38ead61a08b101948def0f82cd258b945ed4d45b53774b400af500670149e601b7 -905c95abda2c68a6559d8a39b6db081c68cef1e1b4be63498004e1b2f408409be9350b5b5d86a30fd443e2b3e445640a -9116dade969e7ce8954afcdd43e5cab64dc15f6c1b8da9d2d69de3f02ba79e6c4f6c7f54d6bf586d30256ae405cd1e41 -a3084d173eacd08c9b5084a196719b57e47a0179826fda73466758235d7ecdb87cbcf097bd6b510517d163a85a7c7edd -85bb00415ad3c9be99ff9ba83672cc59fdd24356b661ab93713a3c8eab34e125d8867f628a3c3891b8dc056e69cd0e83 -8d58541f9f39ed2ee4478acce5d58d124031338ec11b0d55551f00a5a9a6351faa903a5d7c132dc5e4bb026e9cbd18e4 -a622adf72dc250e54f672e14e128c700166168dbe0474cecb340da175346e89917c400677b1bc1c11fcc4cc26591d9db -b3f865014754b688ca8372e8448114fff87bf3ca99856ab9168894d0c4679782c1ced703f5b74e851b370630f5e6ee86 -a7e490b2c40c2446fcd91861c020da9742c326a81180e38110558bb5d9f2341f1c1885e79b364e6419023d1cbdc47380 -b3748d472b1062e54572badbb8e87ac36534407f74932e7fc5b8392d008e8e89758f1671d1e4d30ab0fa40551b13bb5e -89898a5c5ec4313aabc607b0049fd1ebad0e0c074920cf503c9275b564d91916c2c446d3096491c950b7af3ac5e4b0ed -8eb8c83fef2c9dd30ea44e286e9599ec5c20aba983f702e5438afe2e5b921884327ad8d1566c72395587efac79ca7d56 -b92479599e806516ce21fb0bd422a1d1d925335ebe2b4a0a7e044dd275f30985a72b97292477053ac5f00e081430da80 -a34ae450a324fe8a3c25a4d653a654f9580ed56bbea213b8096987bbad0f5701d809a17076435e18017fea4d69f414bc -81381afe6433d62faf62ea488f39675e0091835892ecc238e02acf1662669c6d3962a71a3db652f6fe3bc5f42a0e5dc5 -a430d475bf8580c59111103316fe1aa79c523ea12f1d47a976bbfae76894717c20220e31cf259f08e84a693da6688d70 -b842814c359754ece614deb7d184d679d05d16f18a14b288a401cef5dad2cf0d5ee90bad487b80923fc5573779d4e4e8 -971d9a2627ff2a6d0dcf2af3d895dfbafca28b1c09610c466e4e2bff2746f8369de7f40d65b70aed135fe1d72564aa88 -8f4ce1c59e22b1ce7a0664caaa7e53735b154cfba8d2c5cc4159f2385843de82ab58ed901be876c6f7fce69cb4130950 -86cc9dc321b6264297987000d344fa297ef45bcc2a4df04e458fe2d907ad304c0ea2318e32c3179af639a9a56f3263cf -8229e0876dfe8f665c3fb19b250bd89d40f039bbf1b331468b403655be7be2e104c2fd07b9983580c742d5462ca39a43 -99299d73066e8eb128f698e56a9f8506dfe4bd014931e86b6b487d6195d2198c6c5bf15cccb40ccf1f8ddb57e9da44a2 -a3a3be37ac554c574b393b2f33d0a32a116c1a7cfeaf88c54299a4da2267149a5ecca71f94e6c0ef6e2f472b802f5189 -a91700d1a00387502cdba98c90f75fbc4066fefe7cc221c8f0e660994c936badd7d2695893fde2260c8c11d5bdcdd951 -8e03cae725b7f9562c5c5ab6361644b976a68bada3d7ca508abca8dfc80a469975689af1fba1abcf21bc2a190dab397d -b01461ad23b2a8fa8a6d241e1675855d23bc977dbf4714add8c4b4b7469ccf2375cec20e80cedfe49361d1a30414ac5b -a2673bf9bc621e3892c3d7dd4f1a9497f369add8cbaa3472409f4f86bd21ac67cfac357604828adfee6ada1835365029 -a042dff4bf0dfc33c178ba1b335e798e6308915128de91b12e5dbbab7c4ac8d60a01f6aea028c3a6d87b9b01e4e74c01 -86339e8a75293e4b3ae66b5630d375736b6e6b6b05c5cda5e73fbf7b2f2bd34c18a1d6cefede08625ce3046e77905cb8 -af2ebe1b7d073d03e3d98bc61af83bf26f7a8c130fd607aa92b75db22d14d016481b8aa231e2c9757695f55b7224a27f -a00ee882c9685e978041fd74a2c465f06e2a42ffd3db659053519925be5b454d6f401e3c12c746e49d910e4c5c9c5e8c -978a781c0e4e264e0dad57e438f1097d447d891a1e2aa0d5928f79a9d5c3faae6f258bc94fdc530b7b2fa6a9932bb193 -aa4b7ce2e0c2c9e9655bf21e3e5651c8503bce27483017b0bf476be743ba06db10228b3a4c721219c0779747f11ca282 -b003d1c459dacbcf1a715551311e45d7dbca83a185a65748ac74d1800bbeaba37765d9f5a1a221805c571910b34ebca8 -95b6e531b38648049f0d19de09b881baa1f7ea3b2130816b006ad5703901a05da57467d1a3d9d2e7c73fb3f2e409363c -a6cf9c06593432d8eba23a4f131bb7f72b9bd51ab6b4b772a749fe03ed72b5ced835a349c6d9920dba2a39669cb7c684 -aa3d59f6e2e96fbb66195bc58c8704e139fa76cd15e4d61035470bd6e305db9f98bcbf61ac1b95e95b69ba330454c1b3 -b57f97959c208361de6d7e86dff2b873068adb0f158066e646f42ae90e650079798f165b5cd713141cd3a2a90a961d9a -a76ee8ed9052f6a7a8c69774bb2597be182942f08115baba03bf8faaeaee526feba86120039fe8ca7b9354c3b6e0a8e6 -95689d78c867724823f564627d22d25010f278674c6d2d0cdb10329169a47580818995d1d727ce46c38a1e47943ebb89 -ab676d2256c6288a88e044b3d9ffd43eb9d5aaee00e8fc60ac921395fb835044c71a26ca948e557fed770f52d711e057 -96351c72785c32e5d004b6f4a1259fb8153d631f0c93fed172f18e8ba438fbc5585c1618deeabd0d6d0b82173c2e6170 -93dd8d3db576418e22536eba45ab7f56967c6c97c64260d6cddf38fb19c88f2ec5cd0e0156f50e70855eee8a2b879ffd -ad6ff16f40f6de3d7a737f8e6cebd8416920c4ff89dbdcd75eabab414af9a6087f83ceb9aff7680aa86bff98bd09c8cc -84de53b11671abc9c38710e19540c5c403817562aeb22a88404cdaff792c1180f717dbdfe8f54940c062c4d032897429 -872231b9efa1cdd447b312099a5c164c560440a9441d904e70f5abfc3b2a0d16be9a01aca5e0a2599a61e19407587e3d -88f44ac27094a2aa14e9dc40b099ee6d68f97385950f303969d889ee93d4635e34dff9239103bdf66a4b7cbba3e7eb7a -a59afebadf0260e832f6f44468443562f53fbaf7bcb5e46e1462d3f328ac437ce56edbca617659ac9883f9e13261fad7 -b1990e42743a88de4deeacfd55fafeab3bc380cb95de43ed623d021a4f2353530bcab9594389c1844b1c5ea6634c4555 -85051e841149a10e83f56764e042182208591396d0ce78c762c4a413e6836906df67f38c69793e158d64fef111407ba3 -9778172bbd9b1f2ec6bbdd61829d7b39a7df494a818e31c654bf7f6a30139899c4822c1bf418dd4f923243067759ce63 -9355005b4878c87804fc966e7d24f3e4b02bed35b4a77369d01f25a3dcbff7621b08306b1ac85b76fe7b4a3eb5f839b1 -8f9dc6a54fac052e236f8f0e1f571ac4b5308a43acbe4cc8183bce26262ddaf7994e41cf3034a4cbeca2c505a151e3b1 -8cc59c17307111723fe313046a09e0e32ea0cce62c13814ab7c6408c142d6a0311d801be4af53fc9240523f12045f9ef -8e6057975ed40a1932e47dd3ac778f72ee2a868d8540271301b1aa6858de1a5450f596466494a3e0488be4fbeb41c840 -812145efbd6559ae13325d56a15940ca4253b17e72a9728986b563bb5acc13ec86453796506ac1a8f12bd6f9e4a288c3 -911da0a6d6489eb3dab2ec4a16e36127e8a291ae68a6c2c9de33e97f3a9b1f00da57a94e270a0de79ecc5ecb45d19e83 -b72ea85973f4b2a7e6e71962b0502024e979a73c18a9111130e158541fa47bbaaf53940c8f846913a517dc69982ba9e1 -a7a56ad1dbdc55f177a7ad1d0af78447dc2673291e34e8ab74b26e2e2e7d8c5fe5dc89e7ef60f04a9508847b5b3a8188 -b52503f6e5411db5d1e70f5fb72ccd6463fa0f197b3e51ca79c7b5a8ab2e894f0030476ada72534fa4eb4e06c3880f90 -b51c7957a3d18c4e38f6358f2237b3904618d58b1de5dec53387d25a63772e675a5b714ad35a38185409931157d4b529 -b86b4266e719d29c043d7ec091547aa6f65bbf2d8d831d1515957c5c06513b72aa82113e9645ad38a7bc3f5383504fa6 -b95b547357e6601667b0f5f61f261800a44c2879cf94e879def6a105b1ad2bbf1795c3b98a90d588388e81789bd02681 -a58fd4c5ae4673fa350da6777e13313d5d37ed1dafeeb8f4f171549765b84c895875d9d3ae6a9741f3d51006ef81d962 -9398dc348d078a604aadc154e6eef2c0be1a93bb93ba7fe8976edc2840a3a318941338cc4d5f743310e539d9b46613d2 -902c9f0095014c4a2f0dccaaab543debba6f4cc82c345a10aaf4e72511725dbed7a34cd393a5f4e48a3e5142b7be84ed -a7c0447849bb44d04a0393a680f6cd390093484a79a147dd238f5d878030d1c26646d88211108e59fe08b58ad20c6fbd -80db045535d6e67a422519f5c89699e37098449d249698a7cc173a26ccd06f60238ae6cc7242eb780a340705c906790c -8e52b451a299f30124505de2e74d5341e1b5597bdd13301cc39b05536c96e4380e7f1b5c7ef076f5b3005a868657f17c -824499e89701036037571761e977654d2760b8ce21f184f2879fda55d3cda1e7a95306b8abacf1caa79d3cc075b9d27f -9049b956b77f8453d2070607610b79db795588c0cec12943a0f5fe76f358dea81e4f57a4692112afda0e2c05c142b26f -81911647d818a4b5f4990bfd4bc13bf7be7b0059afcf1b6839333e8569cdb0172fd2945410d88879349f677abaed5eb3 -ad4048f19b8194ed45b6317d9492b71a89a66928353072659f5ce6c816d8f21e69b9d1817d793effe49ca1874daa1096 -8d22f7b2ddb31458661abd34b65819a374a1f68c01fc6c9887edeba8b80c65bceadb8f57a3eb686374004b836261ef67 -92637280c259bc6842884db3d6e32602a62252811ae9b019b3c1df664e8809ffe86db88cfdeb8af9f46435c9ee790267 -a2f416379e52e3f5edc21641ea73dc76c99f7e29ea75b487e18bd233856f4c0183429f378d2bfc6cd736d29d6cadfa49 -882cb6b76dbdc188615dcf1a8439eba05ffca637dd25197508156e03c930b17b9fed2938506fdd7b77567cb488f96222 -b68b621bb198a763fb0634eddb93ed4b5156e59b96c88ca2246fd1aea3e6b77ed651e112ac41b30cd361fadc011d385e -a3cb22f6b675a29b2d1f827cacd30df14d463c93c3502ef965166f20d046af7f9ab7b2586a9c64f4eae4fad2d808a164 -8302d9ce4403f48ca217079762ce42cee8bc30168686bb8d3a945fbd5acd53b39f028dce757b825eb63af2d5ae41169d -b2eef1fbd1a176f1f4cd10f2988c7329abe4eb16c7405099fb92baa724ab397bc98734ef7d4b24c0f53dd90f57520d04 -a1bbef0bd684a3f0364a66bde9b29326bac7aa3dde4caed67f14fb84fed3de45c55e406702f1495a3e2864d4ee975030 -976acdb0efb73e3a3b65633197692dedc2adaed674291ae3df76b827fc866d214e9cac9ca46baefc4405ff13f953d936 -b9fbf71cc7b6690f601f0b1c74a19b7d14254183a2daaafec7dc3830cba5ae173d854bbfebeca985d1d908abe5ef0cda -90591d7b483598c94e38969c4dbb92710a1a894bcf147807f1bcbd8aa3ac210b9f2be65519aa829f8e1ccdc83ad9b8cf -a30568577c91866b9c40f0719d46b7b3b2e0b4a95e56196ac80898a2d89cc67880e1229933f2cd28ee3286f8d03414d7 -97589a88c3850556b359ec5e891f0937f922a751ac7c95949d3bbc7058c172c387611c0f4cb06351ef02e5178b3dd9e4 -98e7bbe27a1711f4545df742f17e3233fbcc63659d7419e1ca633f104cb02a32c84f2fac23ca2b84145c2672f68077ab -a7ddb91636e4506d8b7e92aa9f4720491bb71a72dadc47c7f4410e15f93e43d07d2b371951a0e6a18d1bd087aa96a5c4 -a7c006692227a06db40bceac3d5b1daae60b5692dd9b54772bedb5fea0bcc91cbcdb530cac31900ffc70c5b3ffadc969 -8d3ec6032778420dfa8be52066ba0e623467df33e4e1901dbadd586c5d750f4ccde499b5197e26b9ea43931214060f69 -8d9a8410518ea64f89df319bfd1fc97a0971cdb9ad9b11d1f8fe834042ea7f8dce4db56eeaf179ff8dda93b6db93e5ce -a3c533e9b3aa04df20b9ff635cb1154ce303e045278fcf3f10f609064a5445552a1f93989c52ce852fd0bbd6e2b6c22e -81934f3a7f8c1ae60ec6e4f212986bcc316118c760a74155d06ce0a8c00a9b9669ec4e143ca214e1b995e41271774fd9 -ab8e2d01a71192093ef8fafa7485e795567cc9db95a93fb7cc4cf63a391ef89af5e2bfad4b827fffe02b89271300407f -83064a1eaa937a84e392226f1a60b7cfad4efaa802f66de5df7498962f7b2649924f63cd9962d47906380b97b9fe80e1 -b4f5e64a15c6672e4b55417ee5dc292dcf93d7ea99965a888b1cc4f5474a11e5b6520eacbcf066840b343f4ceeb6bf33 -a63d278b842456ef15c278b37a6ea0f27c7b3ffffefca77c7a66d2ea06c33c4631eb242bbb064d730e70a8262a7b848a -83a41a83dbcdf0d22dc049de082296204e848c453c5ab1ba75aa4067984e053acf6f8b6909a2e1f0009ed051a828a73b -819485b036b7958508f15f3c19436da069cbe635b0318ebe8c014cf1ef9ab2df038c81161b7027475bcfa6fff8dd9faf -aa40e38172806e1e045e167f3d1677ef12d5dcdc89b43639a170f68054bd196c4fae34c675c1644d198907a03f76ba57 -969bae484883a9ed1fbed53b26b3d4ee4b0e39a6c93ece5b3a49daa01444a1c25727dabe62518546f36b047b311b177c -80a9e73a65da99664988b238096a090d313a0ee8e4235bc102fa79bb337b51bb08c4507814eb5baec22103ec512eaab0 -86604379aec5bddda6cbe3ef99c0ac3a3c285b0b1a15b50451c7242cd42ae6b6c8acb717dcca7917838432df93a28502 -a23407ee02a495bed06aa7e15f94cfb05c83e6d6fba64456a9bbabfa76b2b68c5c47de00ba169e710681f6a29bb41a22 -98cff5ecc73b366c6a01b34ac9066cb34f7eeaf4f38a5429bad2d07e84a237047e2a065c7e8a0a6581017dadb4695deb -8de9f68a938f441f3b7ab84bb1f473c5f9e5c9e139e42b7ccee1d254bd57d0e99c2ccda0f3198f1fc5737f6023dd204e -b0ce48d815c2768fb472a315cad86aa033d0e9ca506f146656e2941829e0acb735590b4fbc713c2d18d3676db0a954ac -82f485cdefd5642a6af58ac6817991c49fac9c10ace60f90b27f1788cc026c2fe8afc83cf499b3444118f9f0103598a8 -82c24550ed512a0d53fc56f64cc36b553823ae8766d75d772dacf038c460f16f108f87a39ceef7c66389790f799dbab3 -859ffcf1fe9166388316149b9acc35694c0ea534d43f09dae9b86f4aa00a23b27144dda6a352e74b9516e8c8d6fc809c -b8f7f353eec45da77fb27742405e5ad08d95ec0f5b6842025be9def3d9892f85eb5dd0921b41e6eff373618dba215bca -8ccca4436f9017e426229290f5cd05eac3f16571a4713141a7461acfe8ae99cd5a95bf5b6df129148693c533966145da -a2c67ecc19c0178b2994846fea4c34c327a5d786ac4b09d1d13549d5be5996d8a89021d63d65cb814923388f47cc3a03 -aa0ff87d676b418ec08f5cbf577ac7e744d1d0e9ebd14615b550eb86931eafd2a36d4732cc5d6fab1713fd7ab2f6f7c0 -8aef4730bb65e44efd6bb9441c0ae897363a2f3054867590a2c2ecf4f0224e578c7a67f10b40f8453d9f492ac15a9b2d -86a187e13d8fba5addcfdd5b0410cedd352016c930f913addd769ee09faa6be5ca3e4b1bdb417a965c643a99bd92be42 -a0a4e9632a7a094b14b29b78cd9c894218cdf6783e61671e0203865dc2a835350f465fbaf86168f28af7c478ca17bc89 -a8c7b02d8deff2cd657d8447689a9c5e2cd74ef57c1314ac4d69084ac24a7471954d9ff43fe0907d875dcb65fd0d3ce5 -97ded38760aa7be6b6960b5b50e83b618fe413cbf2bcc1da64c05140bcc32f5e0e709cd05bf8007949953fac5716bad9 -b0d293835a24d64c2ae48ce26e550b71a8c94a0883103757fb6b07e30747f1a871707d23389ba2b2065fa6bafe220095 -8f9e291bf849feaa575592e28e3c8d4b7283f733d41827262367ea1c40f298c7bcc16505255a906b62bf15d9f1ba85fb -998f4e2d12708b4fd85a61597ca2eddd750f73c9e0c9b3cf0825d8f8e01f1628fd19797dcaed3b16dc50331fc6b8b821 -b30d1f8c115d0e63bf48f595dd10908416774c78b3bbb3194192995154d80ea042d2e94d858de5f8aa0261b093c401fd -b5d9c75bb41f964cbff3f00e96d9f1480c91df8913f139f0d385d27a19f57a820f838eb728e46823cbff00e21c660996 -a6edec90b5d25350e2f5f0518777634f9e661ec9d30674cf5b156c4801746d62517751d90074830ac0f4b09911c262f1 -82f98da1264b6b75b8fbeb6a4d96d6a05b25c24db0d57ba3a38efe3a82d0d4e331b9fc4237d6494ccfe4727206457519 -b89511843453cf4ecd24669572d6371b1e529c8e284300c43e0d5bb6b3aaf35aeb634b3cb5c0a2868f0d5e959c1d0772 -a82bf065676583e5c1d3b81987aaae5542f522ba39538263a944bb33ea5b514c649344a96c0205a3b197a3f930fcda6c -a37b47ea527b7e06c460776aa662d9a49ff4149d3993f1a974b0dd165f7171770d189b0e2ea54fd5fccb6a14b116e68a -a1017677f97dda818274d47556d09d0e4ccacb23a252f82a6cfe78c630ad46fb9806307445a59fb61262182de3a2b29c -b01e9fcac239ba270e6877b79273ddd768bf8a51d2ed8a051b1c11e18eff3de5920e2fcbfbd26f06d381eddd3b1f1e1b -82fcd53d803b1c8e4ed76adc339b7f3a5962d37042b9683aabac7513ac68775d4a566a9460183926a6a95dbe7d551a1f -a763e78995d55cd21cdb7ef75d9642d6e1c72453945e346ab6690c20a4e1eeec61bb848ef830ae4b56182535e3c71d8f -b769f4db602251d4b0a1186782799bdcef66de33c110999a5775c50b349666ffd83d4c89714c4e376f2efe021a5cfdb2 -a59cbd1b785efcfa6e83fc3b1d8cf638820bc0c119726b5368f3fba9dce8e3414204fb1f1a88f6c1ff52e87961252f97 -95c8c458fd01aa23ecf120481a9c6332ebec2e8bb70a308d0576926a858457021c277958cf79017ddd86a56cacc2d7db -82eb41390800287ae56e77f2e87709de5b871c8bdb67c10a80fc65f3acb9f7c29e8fa43047436e8933f27449ea61d94d -b3ec25e3545eb83aed2a1f3558d1a31c7edde4be145ecc13b33802654b77dc049b4f0065069dd9047b051e52ab11dcdd -b78a0c715738f56f0dc459ab99e252e3b579b208142836b3c416b704ca1de640ca082f29ebbcee648c8c127df06f6b1e -a4083149432eaaf9520188ebf4607d09cf664acd1f471d4fb654476e77a9eaae2251424ffda78d09b6cb880df35c1219 -8c52857d68d6e9672df3db2df2dbf46b516a21a0e8a18eec09a6ae13c1ef8f369d03233320dd1c2c0bbe00abfc1ea18b -8c856089488803066bff3f8d8e09afb9baf20cecc33c8823c1c0836c3d45498c3de37e87c016b705207f60d2b00f8609 -831a3df39be959047b2aead06b4dcd3012d7b29417f642b83c9e8ce8de24a3dbbd29c6fdf55e2db3f7ea04636c94e403 -aed84d009f66544addabe404bf6d65af7779ce140dc561ff0c86a4078557b96b2053b7b8a43432ffb18cd814f143b9da -93282e4d72b0aa85212a77b336007d8ba071eea17492da19860f1ad16c1ea8867ccc27ef5c37c74b052465cc11ea4f52 -a7b78b8c8d057194e8d68767f1488363f77c77bddd56c3da2bc70b6354c7aa76247c86d51f7371aa38a4aa7f7e3c0bb7 -b1c77283d01dcd1bde649b5b044eac26befc98ff57cbee379fb5b8e420134a88f2fc7f0bf04d15e1fbd45d29e7590fe6 -a4aa8de70330a73b2c6458f20a1067eed4b3474829b36970a8df125d53bbdda4f4a2c60063b7cccb0c80fc155527652f -948a6c79ba1b8ad7e0bed2fae2f0481c4e41b4d9bbdd9b58164e28e9065700e83f210c8d5351d0212e0b0b68b345b3a5 -86a48c31dcbbf7b082c92d28e1f613a2378a910677d7db3a349dc089e4a1e24b12eee8e8206777a3a8c64748840b7387 -976adb1af21e0fc34148917cf43d933d7bfd3fd12ed6c37039dcd5a4520e3c6cf5868539ba5bf082326430deb8a4458d -b93e1a4476f2c51864bb4037e7145f0635eb2827ab91732b98d49b6c07f6ac443111aa1f1da76d1888665cb897c3834e -8afd46fb23bf869999fa19784b18a432a1f252d09506b8dbb756af900518d3f5f244989b3d7c823d9029218c655d3dc6 -83f1e59e3abeed18cdc632921672673f1cb6e330326e11c4e600e13e0d5bc11bdc970ae12952e15103a706fe720bf4d6 -90ce4cc660714b0b673d48010641c09c00fc92a2c596208f65c46073d7f349dd8e6e077ba7dcef9403084971c3295b76 -8b09b0f431a7c796561ecf1549b85048564de428dac0474522e9558b6065fede231886bc108539c104ce88ebd9b5d1b0 -85d6e742e2fb16a7b0ba0df64bc2c0dbff9549be691f46a6669bca05e89c884af16822b85faefefb604ec48c8705a309 -a87989ee231e468a712c66513746fcf03c14f103aadca0eac28e9732487deb56d7532e407953ab87a4bf8961588ef7b0 -b00da10efe1c29ee03c9d37d5918e391ae30e48304e294696b81b434f65cf8c8b95b9d1758c64c25e534d045ba28696f -91c0e1fb49afe46c7056400baa06dbb5f6e479db78ee37e2d76c1f4e88994357e257b83b78624c4ef6091a6c0eb8254d -883fb797c498297ccbf9411a3e727c3614af4eccde41619b773dc7f3259950835ee79453debf178e11dec4d3ada687a0 -a14703347e44eb5059070b2759297fcfcfc60e6893c0373eea069388eba3950aa06f1c57cd2c30984a2d6f9e9c92c79e -afebc7585b304ceba9a769634adff35940e89cd32682c78002822aab25eec3edc29342b7f5a42a56a1fec67821172ad5 -aea3ff3822d09dba1425084ca95fd359718d856f6c133c5fabe2b2eed8303b6e0ba0d8698b48b93136a673baac174fd9 -af2456a09aa777d9e67aa6c7c49a1845ea5cdda2e39f4c935c34a5f8280d69d4eec570446998cbbe31ede69a91e90b06 -82cada19fed16b891ef3442bafd49e1f07c00c2f57b2492dd4ee36af2bd6fd877d6cb41188a4d6ce9ec8d48e8133d697 -82a21034c832287f616619a37c122cee265cc34ae75e881fcaea4ea7f689f3c2bc8150bbf7dbcfd123522bfb7f7b1d68 -86877217105f5d0ec3eeff0289fc2a70d505c9fdf7862e8159553ef60908fb1a27bdaf899381356a4ef4649072a9796c -82b196e49c6e861089a427c0b4671d464e9d15555ffb90954cd0d630d7ae02eb3d98ceb529d00719c2526cd96481355a -a29b41d0d43d26ce76d4358e0db2b77df11f56e389f3b084d8af70a636218bd3ac86b36a9fe46ec9058c26a490f887f7 -a4311c4c20c4d7dd943765099c50f2fd423e203ccfe98ff00087d205467a7873762510cac5fdce7a308913ed07991ed7 -b1f040fc5cc51550cb2c25cf1fd418ecdd961635a11f365515f0cb4ffb31da71f48128c233e9cc7c0cf3978d757ec84e -a9ebae46f86d3bd543c5f207ed0d1aed94b8375dc991161d7a271f01592912072e083e2daf30c146430894e37325a1b9 -826418c8e17ad902b5fe88736323a47e0ca7a44bce4cbe27846ec8fe81de1e8942455dda6d30e192cdcc73e11df31256 -85199db563427c5edcbac21f3d39fec2357be91fb571982ddcdc4646b446ad5ced84410de008cb47b3477ee0d532daf8 -b7eed9cd400b2ca12bf1d9ae008214b8561fb09c8ad9ff959e626ffde00fee5ff2f5b6612e231f2a1a9b1646fcc575e3 -8b40bf12501dcbac78f5a314941326bfcddf7907c83d8d887d0bb149207f85d80cd4dfbd7935439ea7b14ea39a3fded7 -83e3041af302485399ba6cd5120e17af61043977083887e8d26b15feec4a6b11171ac5c06e6ad0971d4b58a81ff12af3 -8f5b9a0eecc589dbf8c35a65d5e996a659277ef6ea509739c0cb7b3e2da9895e8c8012de662e5b23c5fa85d4a8f48904 -835d71ed5e919d89d8e6455f234f3ff215462c4e3720c371ac8c75e83b19dfe3ae15a81547e4dc1138e5f5997f413cc9 -8b7d2e4614716b1db18e9370176ea483e6abe8acdcc3dcdf5fb1f4d22ca55d652feebdccc171c6de38398d9f7bfdec7a -93eace72036fe57d019676a02acf3d224cf376f166658c1bf705db4f24295881d477d6fdd7916efcfceff8c7a063deda -b1ac460b3d516879a84bc886c54f020a9d799e7c49af3e4d7de5bf0d2793c852254c5d8fe5616147e6659512e5ccb012 -acd0947a35cb167a48bcd9667620464b54ac0e78f9316b4aa92dcaab5422d7a732087e52e1c827faa847c6b2fe6e7766 -94ac33d21c3d12ff762d32557860e911cd94d666609ddcc42161b9c16f28d24a526e8b10bb03137257a92cec25ae637d -832e02058b6b994eadd8702921486241f9a19e68ed1406dad545e000a491ae510f525ccf9d10a4bba91c68f2c53a0f58 -9471035d14f78ff8f463b9901dd476b587bb07225c351161915c2e9c6114c3c78a501379ab6fb4eb03194c457cbd22bf -ab64593e034c6241d357fcbc32d8ea5593445a5e7c24cac81ad12bd2ef01843d477a36dc1ba21dbe63b440750d72096a -9850f3b30045e927ad3ec4123a32ed2eb4c911f572b6abb79121873f91016f0d80268de8b12e2093a4904f6e6cab7642 -987212c36b4722fe2e54fa30c52b1e54474439f9f35ca6ad33c5130cd305b8b54b532dd80ffd2c274105f20ce6d79f6e -8b4d0c6abcb239b5ed47bef63bc17efe558a27462c8208fa652b056e9eae9665787cd1aee34fbb55beb045c8bfdb882b -a9f3483c6fee2fe41312d89dd4355d5b2193ac413258993805c5cbbf0a59221f879386d3e7a28e73014f10e65dd503d9 -a2225da3119b9b7c83d514b9f3aeb9a6d9e32d9cbf9309cbb971fd53c4b2c001d10d880a8ad8a7c281b21d85ceca0b7c -a050be52e54e676c151f7a54453bbb707232f849beab4f3bf504b4d620f59ed214409d7c2bd3000f3ff13184ccda1c35 -adbccf681e15b3edb6455a68d292b0a1d0f5a4cb135613f5e6db9943f02181341d5755875db6ee474e19ace1c0634a28 -8b6eff675632a6fad0111ec72aacc61c7387380eb87933fd1d098856387d418bd38e77d897e65d6fe35951d0627c550b -aabe2328ddf90989b15e409b91ef055cb02757d34987849ae6d60bef2c902bf8251ed21ab30acf39e500d1d511e90845 -92ba4eb1f796bc3d8b03515f65c045b66e2734c2da3fc507fdd9d6b5d1e19ab3893726816a32141db7a31099ca817d96 -8a98b3cf353138a1810beb60e946183803ef1d39ac4ea92f5a1e03060d35a4774a6e52b14ead54f6794d5f4022b8685c -909f8a5c13ec4a59b649ed3bee9f5d13b21d7f3e2636fd2bb3413c0646573fdf9243d63083356f12f5147545339fcd55 -9359d914d1267633141328ed0790d81c695fea3ddd2d406c0df3d81d0c64931cf316fe4d92f4353c99ff63e2aefc4e34 -b88302031681b54415fe8fbfa161c032ea345c6af63d2fb8ad97615103fd4d4281c5a9cae5b0794c4657b97571a81d3b -992c80192a519038082446b1fb947323005b275e25f2c14c33cc7269e0ec038581cc43705894f94bad62ae33a8b7f965 -a78253e3e3eece124bef84a0a8807ce76573509f6861d0b6f70d0aa35a30a123a9da5e01e84969708c40b0669eb70aa6 -8d5724de45270ca91c94792e8584e676547d7ac1ac816a6bb9982ee854eb5df071d20545cdfd3771cd40f90e5ba04c8e -825a6f586726c68d45f00ad0f5a4436523317939a47713f78fd4fe81cd74236fdac1b04ecd97c2d0267d6f4981d7beb1 -93e02b6052719f607dacd3a088274f65596bd0d09920b61ab5da61bbdc7f5049334cf11213945d57e5ac7d055d042b7e024aa2b2f08f0a91260805272dc51051c6e47ad4fa403b02b4510b647ae3d1770bac0326a805bbefd48056c8c121bdb8 -b5bfd7dd8cdeb128843bc287230af38926187075cbfbefa81009a2ce615ac53d2914e5870cb452d2afaaab24f3499f72185cbfee53492714734429b7b38608e23926c911cceceac9a36851477ba4c60b087041de621000edc98edada20c1def2 -b5337ba0ce5d37224290916e268e2060e5c14f3f9fc9e1ec3af5a958e7a0303122500ce18f1a4640bf66525bd10e763501fe986d86649d8d45143c08c3209db3411802c226e9fe9a55716ac4a0c14f9dcef9e70b2bb309553880dc5025eab3cc -b3c1dcdc1f62046c786f0b82242ef283e7ed8f5626f72542aa2c7a40f14d9094dd1ebdbd7457ffdcdac45fd7da7e16c51200b06d791e5e43e257e45efdf0bd5b06cd2333beca2a3a84354eb48662d83aef5ecf4e67658c851c10b13d8d87c874 -954d91c7688983382609fca9e211e461f488a5971fd4e40d7e2892037268eacdfd495cfa0a7ed6eb0eb11ac3ae6f651716757e7526abe1e06c64649d80996fd3105c20c4c94bc2b22d97045356fe9d791f21ea6428ac48db6f9e68e30d875280 -88a6b6bb26c51cf9812260795523973bb90ce80f6820b6c9048ab366f0fb96e48437a7f7cb62aedf64b11eb4dfefebb0147608793133d32003cb1f2dc47b13b5ff45f1bb1b2408ea45770a08dbfaec60961acb8119c47b139a13b8641e2c9487 -85cd7be9728bd925d12f47fb04b32d9fad7cab88788b559f053e69ca18e463113ecc8bbb6dbfb024835f901b3a957d3108d6770fb26d4c8be0a9a619f6e3a4bf15cbfd48e61593490885f6cee30e4300c5f9cf5e1c08e60a2d5b023ee94fcad0 -80477dba360f04399821a48ca388c0fa81102dd15687fea792ee8c1114e00d1bc4839ad37ac58900a118d863723acfbe08126ea883be87f50e4eabe3b5e72f5d9e041db8d9b186409fd4df4a7dde38c0e0a3b1ae29b098e5697e7f110b6b27e4 -b7a6aec08715a9f8672a2b8c367e407be37e59514ac19dd4f0942a68007bba3923df22da48702c63c0d6b3efd3c2d04e0fe042d8b5a54d562f9f33afc4865dcbcc16e99029e25925580e87920c399e710d438ac1ce3a6dc9b0d76c064a01f6f7 -ac1b001edcea02c8258aeffbf9203114c1c874ad88dae1184fadd7d94cd09053649efd0ca413400e6e9b5fa4eac33261000af88b6bd0d2abf877a4f0355d2fb4d6007adb181695201c5432e50b850b51b3969f893bddf82126c5a71b042b7686 -90043fda4de53fb364fab2c04be5296c215599105ecff0c12e4917c549257125775c29f2507124d15f56e30447f367db0596c33237242c02d83dfd058735f1e3c1ff99069af55773b6d51d32a68bf75763f59ec4ee7267932ae426522b8aaab6 -a8660ce853e9dc08271bf882e29cd53397d63b739584dda5263da4c7cc1878d0cf6f3e403557885f557e184700575fee016ee8542dec22c97befe1d10f414d22e84560741cdb3e74c30dda9b42eeaaf53e27822de2ee06e24e912bf764a9a533 -8fe3921a96d0d065e8aa8fce9aa42c8e1461ca0470688c137be89396dd05103606dab6cdd2a4591efd6addf72026c12e065da7be276dee27a7e30afa2bd81c18f1516e7f068f324d0bad9570b95f6bd02c727cd2343e26db0887c3e4e26dceda -8ae1ad97dcb9c192c9a3933541b40447d1dc4eebf380151440bbaae1e120cc5cdf1bcea55180b128d8e180e3af623815191d063cc0d7a47d55fb7687b9d87040bf7bc1a7546b07c61db5ccf1841372d7c2fe4a5431ffff829f3c2eb590b0b710 -8c2fa96870a88150f7876c931e2d3cc2adeaaaf5c73ef5fa1cf9dfa0991ae4819f9321af7e916e5057d87338e630a2f21242c29d76963cf26035b548d2a63d8ad7bd6efefa01c1df502cbdfdfe0334fb21ceb9f686887440f713bf17a89b8081 -b9aa98e2f02bb616e22ee5dd74c7d1049321ac9214d093a738159850a1dbcc7138cb8d26ce09d8296368fd5b291d74fa17ac7cc1b80840fdd4ee35e111501e3fa8485b508baecda7c1ab7bd703872b7d64a2a40b3210b6a70e8a6ffe0e5127e3 -9292db67f8771cdc86854a3f614a73805bf3012b48f1541e704ea4015d2b6b9c9aaed36419769c87c49f9e3165f03edb159c23b3a49c4390951f78e1d9b0ad997129b17cdb57ea1a6638794c0cca7d239f229e589c5ae4f9fe6979f7f8cba1d7 -91cd9e86550f230d128664f7312591fee6a84c34f5fc7aed557bcf986a409a6de722c4330453a305f06911d2728626e611acfdf81284f77f60a3a1595053a9479964fd713117e27c0222cc679674b03bc8001501aaf9b506196c56de29429b46 -a9516b73f605cc31b89c68b7675dc451e6364595243d235339437f556cf22d745d4250c1376182273be2d99e02c10eee047410a43eff634d051aeb784e76cb3605d8e079b9eb6ad1957dfdf77e1cd32ce4a573c9dfcc207ca65af6eb187f6c3d -a9667271f7d191935cc8ad59ef3ec50229945faea85bfdfb0d582090f524436b348aaa0183b16a6231c00332fdac2826125b8c857a2ed9ec66821cfe02b3a2279be2412441bc2e369b255eb98614e4be8490799c4df22f18d47d24ec70bba5f7 -a4371144d2aa44d70d3cb9789096d3aa411149a6f800cb46f506461ee8363c8724667974252f28aea61b6030c05930ac039c1ee64bb4bd56532a685cae182bf2ab935eee34718cffcb46cae214c77aaca11dbb1320faf23c47247db1da04d8dc -89a7eb441892260b7e81168c386899cd84ffc4a2c5cad2eae0d1ab9e8b5524662e6f660fe3f8bfe4c92f60b060811bc605b14c5631d16709266886d7885a5eb5930097127ec6fb2ebbaf2df65909cf48f253b3d5e22ae48d3e9a2fd2b01f447e -9648c42ca97665b5eccb49580d8532df05eb5a68db07f391a2340769b55119eaf4c52fe4f650c09250fa78a76c3a1e271799b8333cc2628e3d4b4a6a3e03da1f771ecf6516dd63236574a7864ff07e319a6f11f153406280d63af9e2b5713283 -9663bf6dd446ea7a90658ee458578d4196dc0b175ef7fcfa75f44d41670850774c2e46c5a6be132a2c072a3c0180a24f0305d1acac49d2d79878e5cda80c57feda3d01a6af12e78b5874e2a4b3717f11c97503b41a4474e2e95b179113726199 -b212aeb4814e0915b432711b317923ed2b09e076aaf558c3ae8ef83f9e15a83f9ea3f47805b2750ab9e8106cb4dc6ad003522c84b03dc02829978a097899c773f6fb31f7fe6b8f2d836d96580f216fec20158f1590c3e0d7850622e15194db05 -925f005059bf07e9ceccbe66c711b048e236ade775720d0fe479aebe6e23e8af281225ad18e62458dc1b03b42ad4ca290d4aa176260604a7aad0d9791337006fbdebe23746f8060d42876f45e4c83c3643931392fde1cd13ff8bddf8111ef974 -9553edb22b4330c568e156a59ef03b26f5c326424f830fe3e8c0b602f08c124730ffc40bc745bec1a22417adb22a1a960243a10565c2be3066bfdb841d1cd14c624cd06e0008f4beb83f972ce6182a303bee3fcbcabc6cfe48ec5ae4b7941bfc -935f5a404f0a78bdcce709899eda0631169b366a669e9b58eacbbd86d7b5016d044b8dfc59ce7ed8de743ae16c2343b50e2f925e88ba6319e33c3fc76b314043abad7813677b4615c8a97eb83cc79de4fedf6ccbcfa4d4cbf759a5a84e4d9742 -a5b014ab936eb4be113204490e8b61cd38d71da0dec7215125bcd131bf3ab22d0a32ce645bca93e7b3637cf0c2db3d6601a0ddd330dc46f9fae82abe864ffc12d656c88eb50c20782e5bb6f75d18760666f43943abb644b881639083e122f557 -935b7298ae52862fa22bf03bfc1795b34c70b181679ae27de08a9f5b4b884f824ef1b276b7600efa0d2f1d79e4a470d51692fd565c5cf8343dd80e5d3336968fc21c09ba9348590f6206d4424eb229e767547daefa98bc3aa9f421158dee3f2a -9830f92446e708a8f6b091cc3c38b653505414f8b6507504010a96ffda3bcf763d5331eb749301e2a1437f00e2415efb01b799ad4c03f4b02de077569626255ac1165f96ea408915d4cf7955047620da573e5c439671d1fa5c833fb11de7afe6 -840dcc44f673fff3e387af2bb41e89640f2a70bcd2b92544876daa92143f67c7512faf5f90a04b7191de01f3e2b1bde00622a20dc62ca23bbbfaa6ad220613deff43908382642d4d6a86999f662efd64b1df448b68c847cfa87630a3ffd2ec76 -92950c895ed54f7f876b2fda17ecc9c41b7accfbdd42c210cc5b475e0737a7279f558148531b5c916e310604a1de25a80940c94fe5389ae5d6a5e9c371be67bceea1877f5401725a6595bcf77ece60905151b6dfcb68b75ed2e708c73632f4fd -8010246bf8e94c25fd029b346b5fbadb404ef6f44a58fd9dd75acf62433d8cc6db66974f139a76e0c26dddc1f329a88214dbb63276516cf325c7869e855d07e0852d622c332ac55609ba1ec9258c45746a2aeb1af0800141ee011da80af175d4 -b0f1bad257ebd187bdc3f37b23f33c6a5d6a8e1f2de586080d6ada19087b0e2bf23b79c1b6da1ee82271323f5bdf3e1b018586b54a5b92ab6a1a16bb3315190a3584a05e6c37d5ca1e05d702b9869e27f513472bcdd00f4d0502a107773097da -9636d24f1ede773ce919f309448dd7ce023f424afd6b4b69cb98c2a988d849a283646dc3e469879daa1b1edae91ae41f009887518e7eb5578f88469321117303cd3ac2d7aee4d9cb5f82ab9ae3458e796dfe7c24284b05815acfcaa270ff22e2 -b373feb5d7012fd60578d7d00834c5c81df2a23d42794fed91aa9535a4771fde0341c4da882261785e0caca40bf83405143085e7f17e55b64f6c5c809680c20b050409bf3702c574769127c854d27388b144b05624a0e24a1cbcc4d08467005b -b15680648949ce69f82526e9b67d9b55ce5c537dc6ab7f3089091a9a19a6b90df7656794f6edc87fb387d21573ffc847062623685931c2790a508cbc8c6b231dd2c34f4d37d4706237b1407673605a604bcf6a50cc0b1a2db20485e22b02c17e -8817e46672d40c8f748081567b038a3165f87994788ec77ee8daea8587f5540df3422f9e120e94339be67f186f50952504cb44f61e30a5241f1827e501b2de53c4c64473bcc79ab887dd277f282fbfe47997a930dd140ac08b03efac88d81075 -a6e4ef6c1d1098f95aae119905f87eb49b909d17f9c41bcfe51127aa25fee20782ea884a7fdf7d5e9c245b5a5b32230b07e0dbf7c6743bf52ee20e2acc0b269422bd6cf3c07115df4aa85b11b2c16630a07c974492d9cdd0ec325a3fabd95044 -8634aa7c3d00e7f17150009698ce440d8e1b0f13042b624a722ace68ead870c3d2212fbee549a2c190e384d7d6ac37ce14ab962c299ea1218ef1b1489c98906c91323b94c587f1d205a6edd5e9d05b42d591c26494a6f6a029a2aadb5f8b6f67 -821a58092900bdb73decf48e13e7a5012a3f88b06288a97b855ef51306406e7d867d613d9ec738ebacfa6db344b677d21509d93f3b55c2ebf3a2f2a6356f875150554c6fff52e62e3e46f7859be971bf7dd9d5b3e1d799749c8a97c2e04325df -8dba356577a3a388f782e90edb1a7f3619759f4de314ad5d95c7cc6e197211446819c4955f99c5fc67f79450d2934e3c09adefc91b724887e005c5190362245eec48ce117d0a94d6fa6db12eda4ba8dde608fbbd0051f54dcf3bb057adfb2493 -a32a690dc95c23ed9fb46443d9b7d4c2e27053a7fcc216d2b0020a8cf279729c46114d2cda5772fd60a97016a07d6c5a0a7eb085a18307d34194596f5b541cdf01b2ceb31d62d6b55515acfd2b9eec92b27d082fbc4dc59fc63b551eccdb8468 -a040f7f4be67eaf0a1d658a3175d65df21a7dbde99bfa893469b9b43b9d150fc2e333148b1cb88cfd0447d88fa1a501d126987e9fdccb2852ecf1ba907c2ca3d6f97b055e354a9789854a64ecc8c2e928382cf09dda9abde42bbdf92280cdd96 -864baff97fa60164f91f334e0c9be00a152a416556b462f96d7c43b59fe1ebaff42f0471d0bf264976f8aa6431176eb905bd875024cf4f76c13a70bede51dc3e47e10b9d5652d30d2663b3af3f08d5d11b9709a0321aba371d2ef13174dcfcaf -95a46f32c994133ecc22db49bad2c36a281d6b574c83cfee6680b8c8100466ca034b815cfaedfbf54f4e75188e661df901abd089524e1e0eb0bf48d48caa9dd97482d2e8c1253e7e8ac250a32fd066d5b5cb08a8641bdd64ecfa48289dca83a3 -a2cce2be4d12144138cb91066e0cd0542c80b478bf467867ebef9ddaf3bd64e918294043500bf5a9f45ee089a8d6ace917108d9ce9e4f41e7e860cbce19ac52e791db3b6dde1c4b0367377b581f999f340e1d6814d724edc94cb07f9c4730774 -b145f203eee1ac0a1a1731113ffa7a8b0b694ef2312dabc4d431660f5e0645ef5838e3e624cfe1228cfa248d48b5760501f93e6ab13d3159fc241427116c4b90359599a4cb0a86d0bb9190aa7fabff482c812db966fd2ce0a1b48cb8ac8b3bca -adabe5d215c608696e03861cbd5f7401869c756b3a5aadc55f41745ad9478145d44393fec8bb6dfc4ad9236dc62b9ada0f7ca57fe2bae1b71565dbf9536d33a68b8e2090b233422313cc96afc7f1f7e0907dc7787806671541d6de8ce47c4cd0 -ae7845fa6b06db53201c1080e01e629781817f421f28956589c6df3091ec33754f8a4bd4647a6bb1c141ac22731e3c1014865d13f3ed538dcb0f7b7576435133d9d03be655f8fbb4c9f7d83e06d1210aedd45128c2b0c9bab45a9ddde1c862a5 -9159eaa826a24adfa7adf6e8d2832120ebb6eccbeb3d0459ffdc338548813a2d239d22b26451fda98cc0c204d8e1ac69150b5498e0be3045300e789bcb4e210d5cd431da4bdd915a21f407ea296c20c96608ded0b70d07188e96e6c1a7b9b86b -a9fc6281e2d54b46458ef564ffaed6944bff71e389d0acc11fa35d3fcd8e10c1066e0dde5b9b6516f691bb478e81c6b20865281104dcb640e29dc116daae2e884f1fe6730d639dbe0e19a532be4fb337bf52ae8408446deb393d224eee7cfa50 -84291a42f991bfb36358eedead3699d9176a38f6f63757742fdbb7f631f2c70178b1aedef4912fed7b6cf27e88ddc7eb0e2a6aa4b999f3eb4b662b93f386c8d78e9ac9929e21f4c5e63b12991fcde93aa64a735b75b535e730ff8dd2abb16e04 -a1b7fcacae181495d91765dfddf26581e8e39421579c9cbd0dd27a40ea4c54af3444a36bf85a11dda2114246eaddbdd619397424bb1eb41b5a15004b902a590ede5742cd850cf312555be24d2df8becf48f5afba5a8cd087cb7be0a521728386 -92feaaf540dbd84719a4889a87cdd125b7e995a6782911931fef26da9afcfbe6f86aaf5328fe1f77631491ce6239c5470f44c7791506c6ef1626803a5794e76d2be0af92f7052c29ac6264b7b9b51f267ad820afc6f881460521428496c6a5f1 -a525c925bfae1b89320a5054acc1fa11820f73d0cf28d273092b305467b2831fab53b6daf75fb926f332782d50e2522a19edcd85be5eb72f1497193c952d8cd0bcc5d43b39363b206eae4cb1e61668bde28a3fb2fc1e0d3d113f6dfadb799717 -98752bb6f5a44213f40eda6aa4ff124057c1b13b6529ab42fe575b9afa66e59b9c0ed563fb20dff62130c436c3e905ee17dd8433ba02c445b1d67182ab6504a90bbe12c26a754bbf734665c622f76c62fe2e11dd43ce04fd2b91a8463679058b -a9aa9a84729f7c44219ff9e00e651e50ddea3735ef2a73fdf8ed8cd271961d8ed7af5cd724b713a89a097a3fe65a3c0202f69458a8b4c157c62a85668b12fc0d3957774bc9b35f86c184dd03bfefd5c325da717d74192cc9751c2073fe9d170e -b221c1fd335a4362eff504cd95145f122bf93ea02ae162a3fb39c75583fc13a932d26050e164da97cff3e91f9a7f6ff80302c19dd1916f24acf6b93b62f36e9665a8785413b0c7d930c7f1668549910f849bca319b00e59dd01e5dec8d2edacc -a71e2b1e0b16d754b848f05eda90f67bedab37709550171551050c94efba0bfc282f72aeaaa1f0330041461f5e6aa4d11537237e955e1609a469d38ed17f5c2a35a1752f546db89bfeff9eab78ec944266f1cb94c1db3334ab48df716ce408ef -b990ae72768779ba0b2e66df4dd29b3dbd00f901c23b2b4a53419226ef9232acedeb498b0d0687c463e3f1eead58b20b09efcefa566fbfdfe1c6e48d32367936142d0a734143e5e63cdf86be7457723535b787a9cfcfa32fe1d61ad5a2617220 -8d27e7fbff77d5b9b9bbc864d5231fecf817238a6433db668d5a62a2c1ee1e5694fdd90c3293c06cc0cb15f7cbeab44d0d42be632cb9ff41fc3f6628b4b62897797d7b56126d65b694dcf3e298e3561ac8813fbd7296593ced33850426df42db -a92039a08b5502d5b211a7744099c9f93fa8c90cedcb1d05e92f01886219dd464eb5fb0337496ad96ed09c987da4e5f019035c5b01cc09b2a18b8a8dd419bc5895388a07e26958f6bd26751929c25f89b8eb4a299d822e2d26fec9ef350e0d3c -92dcc5a1c8c3e1b28b1524e3dd6dbecd63017c9201da9dbe077f1b82adc08c50169f56fc7b5a3b28ec6b89254de3e2fd12838a761053437883c3e01ba616670cea843754548ef84bcc397de2369adcca2ab54cd73c55dc68d87aec3fc2fe4f10 diff --git a/core/lib/merkle_tree/src/domain.rs b/core/lib/merkle_tree/src/domain.rs index ecd9b4c1fbe4..9a59943f3376 100644 --- a/core/lib/merkle_tree/src/domain.rs +++ b/core/lib/merkle_tree/src/domain.rs @@ -47,6 +47,11 @@ pub struct ZkSyncTree { } impl ZkSyncTree { + /// Returns a hash of an empty tree. This is a constant value. + pub fn empty_tree_hash() -> ValueHash { + Blake2Hasher.empty_tree_hash() + } + fn create_thread_pool(thread_count: usize) -> ThreadPool { ThreadPoolBuilder::new() .thread_name(|idx| format!("new-merkle-tree-{idx}")) @@ -375,9 +380,10 @@ impl ZkSyncTreeReader { &self.0.db } - /// Returns the current root hash of this tree. - pub fn root_hash(&self) -> ValueHash { - self.0.latest_root_hash() + /// Returns the root hash and leaf count at the specified L1 batch. + pub fn root_info(&self, l1_batch_number: L1BatchNumber) -> Option<(ValueHash, u64)> { + let root = self.0.root(l1_batch_number.0.into())?; + Some((root.hash(&Blake2Hasher), root.leaf_count())) } /// Returns the next L1 batch number that should be processed by the tree. @@ -397,11 +403,6 @@ impl ZkSyncTreeReader { }) } - /// Returns the number of leaves in the tree. - pub fn leaf_count(&self) -> u64 { - self.0.latest_root().leaf_count() - } - /// Reads entries together with Merkle proofs with the specified keys from the tree. The entries are returned /// in the same order as requested. /// diff --git a/core/lib/merkle_tree/src/hasher/nodes.rs b/core/lib/merkle_tree/src/hasher/nodes.rs index 6e1c007bc423..6172d9088126 100644 --- a/core/lib/merkle_tree/src/hasher/nodes.rs +++ b/core/lib/merkle_tree/src/hasher/nodes.rs @@ -4,7 +4,8 @@ use std::slice; use crate::{ hasher::HasherWithStats, - types::{ChildRef, InternalNode, LeafNode, Node, ValueHash, TREE_DEPTH}, + types::{ChildRef, InternalNode, LeafNode, Node, Root, ValueHash, TREE_DEPTH}, + HashTree, }; impl LeafNode { @@ -256,6 +257,15 @@ impl Node { } } +impl Root { + pub(crate) fn hash(&self, hasher: &dyn HashTree) -> ValueHash { + let Self::Filled { node, .. } = self else { + return hasher.empty_tree_hash(); + }; + node.hash(&mut HasherWithStats::new(&hasher), 0) + } +} + #[cfg(test)] mod tests { use zksync_crypto::hasher::{blake2::Blake2Hasher, Hasher}; diff --git a/core/lib/merkle_tree/src/lib.rs b/core/lib/merkle_tree/src/lib.rs index caa965751578..09bd1bf91a22 100644 --- a/core/lib/merkle_tree/src/lib.rs +++ b/core/lib/merkle_tree/src/lib.rs @@ -61,7 +61,7 @@ pub use crate::{ TreeLogEntry, TreeLogEntryWithProof, ValueHash, }, }; -use crate::{hasher::HasherWithStats, storage::Storage, types::Root}; +use crate::{storage::Storage, types::Root}; mod consistency; pub mod domain; @@ -166,10 +166,7 @@ impl MerkleTree { /// was not written yet. pub fn root_hash(&self, version: u64) -> Option { let root = self.root(version)?; - let Root::Filled { node, .. } = root else { - return Some(self.hasher.empty_tree_hash()); - }; - Some(node.hash(&mut HasherWithStats::new(&self.hasher), 0)) + Some(root.hash(&self.hasher)) } pub(crate) fn root(&self, version: u64) -> Option { @@ -256,6 +253,8 @@ impl MerkleTree { #[cfg(test)] mod tests { + use std::collections::HashMap; + use super::*; use crate::types::TreeTags; @@ -268,6 +267,7 @@ mod tests { depth: 256, hasher: "blake2s256".to_string(), is_recovering: false, + custom: HashMap::new(), }); MerkleTree::new(db); @@ -282,6 +282,7 @@ mod tests { depth: 128, hasher: "blake2s256".to_string(), is_recovering: false, + custom: HashMap::new(), }); MerkleTree::new(db); @@ -296,6 +297,7 @@ mod tests { depth: 256, hasher: "sha256".to_string(), is_recovering: false, + custom: HashMap::new(), }); MerkleTree::new(db); diff --git a/core/lib/merkle_tree/src/metrics.rs b/core/lib/merkle_tree/src/metrics.rs index 8c8fdc4aeaa4..2190b9acaa07 100644 --- a/core/lib/merkle_tree/src/metrics.rs +++ b/core/lib/merkle_tree/src/metrics.rs @@ -67,37 +67,37 @@ const LEAF_LEVEL_BUCKETS: Buckets = Buckets::linear(20.0..=40.0, 4.0); #[metrics(prefix = "merkle_tree_extend_patch")] struct TreeUpdateMetrics { // Metrics related to the AR16MT tree architecture - /// Number of new leaves inserted during tree traversal while processing a single block. + /// Number of new leaves inserted during tree traversal while processing a single batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] new_leaves: Histogram, - /// Number of new internal nodes inserted during tree traversal while processing a single block. + /// Number of new internal nodes inserted during tree traversal while processing a single batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] new_internal_nodes: Histogram, - /// Number of existing leaves moved to a new location while processing a single block. + /// Number of existing leaves moved to a new location while processing a single batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] moved_leaves: Histogram, - /// Number of existing leaves updated while processing a single block. + /// Number of existing leaves updated while processing a single batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] updated_leaves: Histogram, - /// Average level of leaves moved or created while processing a single block. + /// Average level of leaves moved or created while processing a single batch. #[metrics(buckets = LEAF_LEVEL_BUCKETS)] avg_leaf_level: Histogram, - /// Maximum level of leaves moved or created while processing a single block. + /// Maximum level of leaves moved or created while processing a single batch. #[metrics(buckets = LEAF_LEVEL_BUCKETS)] max_leaf_level: Histogram, // Metrics related to input instructions - /// Number of keys read while processing a single block (only applicable to the full operation mode). + /// Number of keys read while processing a single batch (only applicable to the full operation mode). #[metrics(buckets = NODE_COUNT_BUCKETS)] key_reads: Histogram, - /// Number of missing keys read while processing a single block (only applicable to the full + /// Number of missing keys read while processing a single batch (only applicable to the full /// operation mode). #[metrics(buckets = NODE_COUNT_BUCKETS)] missing_key_reads: Histogram, - /// Number of nodes of previous versions read from the DB while processing a single block. + /// Number of nodes of previous versions read from the DB while processing a single batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] db_reads: Histogram, - /// Number of nodes of the current version re-read from the patch set while processing a single block. + /// Number of nodes of the current version re-read from the patch set while processing a single batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] patch_reads: Histogram, } @@ -194,13 +194,13 @@ impl ops::AddAssign for TreeUpdaterStats { #[derive(Debug, Metrics)] #[metrics(prefix = "merkle_tree")] pub(crate) struct BlockTimings { - /// Time spent loading tree nodes from DB per block. + /// Time spent loading tree nodes from DB per batch. #[metrics(buckets = Buckets::LATENCIES)] pub load_nodes: Histogram, - /// Time spent traversing the tree and creating new nodes per block. + /// Time spent traversing the tree and creating new nodes per batch. #[metrics(buckets = Buckets::LATENCIES)] pub extend_patch: Histogram, - /// Time spent finalizing the block (mainly hash computations). + /// Time spent finalizing a batch (mainly hash computations). #[metrics(buckets = Buckets::LATENCIES)] pub finalize_patch: Histogram, } @@ -233,13 +233,13 @@ impl fmt::Display for NibbleCount { #[derive(Debug, Metrics)] #[metrics(prefix = "merkle_tree_apply_patch")] struct ApplyPatchMetrics { - /// Total number of nodes included into a RocksDB patch per block. + /// Total number of nodes included into a RocksDB patch per batch. #[metrics(buckets = NODE_COUNT_BUCKETS)] nodes: Histogram, - /// Number of nodes included into a RocksDB patch per block, grouped by the key nibble count. + /// Number of nodes included into a RocksDB patch per batch, grouped by the key nibble count. #[metrics(buckets = NODE_COUNT_BUCKETS)] nodes_by_nibble_count: Family>, - /// Total byte size of nodes included into a RocksDB patch per block, grouped by the key nibble count. + /// Total byte size of nodes included into a RocksDB patch per batch, grouped by the key nibble count. #[metrics(buckets = BYTE_SIZE_BUCKETS)] node_bytes: Family>, /// Number of hashes in child references copied from previous tree versions. Allows to estimate @@ -295,7 +295,7 @@ impl ApplyPatchStats { for (nibble_count, stats) in node_bytes { let label = NibbleCount::new(nibble_count); metrics.nodes_by_nibble_count[&label].observe(stats.count); - metrics.nodes_by_nibble_count[&label].observe(stats.bytes); + metrics.node_bytes[&label].observe(stats.bytes); } metrics.copied_hashes.observe(self.copied_hashes); @@ -359,3 +359,39 @@ pub(crate) struct PruningTimings { #[vise::register] pub(crate) static PRUNING_TIMINGS: Global = Global::new(); + +#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, EncodeLabelValue, EncodeLabelSet)] +#[metrics(label = "stage", rename_all = "snake_case")] +pub(crate) enum RecoveryStage { + Extend, + ApplyPatch, +} + +const CHUNK_SIZE_BUCKETS: Buckets = Buckets::values(&[ + 1_000.0, + 2_000.0, + 5_000.0, + 10_000.0, + 20_000.0, + 50_000.0, + 100_000.0, + 200_000.0, + 500_000.0, + 1_000_000.0, + 2_000_000.0, + 5_000_000.0, +]); + +#[derive(Debug, Metrics)] +#[metrics(prefix = "merkle_tree_recovery")] +pub(crate) struct RecoveryMetrics { + /// Number of entries in a recovered chunk. + #[metrics(buckets = CHUNK_SIZE_BUCKETS)] + pub chunk_size: Histogram, + /// Latency of a specific stage of recovery for a single chunk. + #[metrics(buckets = Buckets::LATENCIES, unit = Unit::Seconds)] + pub stage_latency: Family>, +} + +#[vise::register] +pub(crate) static RECOVERY_METRICS: Global = Global::new(); diff --git a/core/lib/merkle_tree/src/recovery.rs b/core/lib/merkle_tree/src/recovery.rs index aecda593a254..bc9e6cc486fa 100644 --- a/core/lib/merkle_tree/src/recovery.rs +++ b/core/lib/merkle_tree/src/recovery.rs @@ -35,12 +35,13 @@ //! before extending the tree; these nodes are guaranteed to be the *only* DB reads necessary //! to insert new entries. -use std::time::Instant; +use std::{collections::HashMap, time::Instant}; use zksync_crypto::hasher::blake2::Blake2Hasher; use crate::{ hasher::{HashTree, HasherWithStats}, + metrics::{RecoveryStage, RECOVERY_METRICS}, storage::{PatchSet, PruneDatabase, PrunePatchSet, Storage}, types::{Key, Manifest, Root, TreeEntry, TreeTags, ValueHash}, }; @@ -110,6 +111,24 @@ impl MerkleTreeRecovery { } } + /// Updates custom tags for the tree using the provided closure. The update is atomic and unconditional. + #[allow(clippy::missing_panics_doc)] // should never be triggered; the manifest is added in the constructor + pub fn update_custom_tags( + &mut self, + update: impl FnOnce(&mut HashMap) -> R, + ) -> R { + let mut manifest = self + .db + .manifest() + .expect("Merkle tree manifest disappeared"); + let tags = manifest + .tags + .get_or_insert_with(|| TreeTags::new(&self.hasher)); + let output = update(&mut tags.custom); + self.db.apply_patch(PatchSet::from_manifest(manifest)); + output + } + /// Returns the version of the tree being recovered. pub fn recovered_version(&self) -> u64 { self.recovered_version @@ -149,15 +168,18 @@ impl MerkleTreeRecovery { )] pub fn extend_linear(&mut self, entries: Vec) { tracing::debug!("Started extending tree"); + RECOVERY_METRICS.chunk_size.observe(entries.len()); - let started_at = Instant::now(); + let stage_latency = RECOVERY_METRICS.stage_latency[&RecoveryStage::Extend].start(); let storage = Storage::new(&self.db, &self.hasher, self.recovered_version, false); let patch = storage.extend_during_linear_recovery(entries); - tracing::debug!("Finished processing keys; took {:?}", started_at.elapsed()); + let stage_latency = stage_latency.observe(); + tracing::debug!("Finished processing keys; took {stage_latency:?}"); - let started_at = Instant::now(); + let stage_latency = RECOVERY_METRICS.stage_latency[&RecoveryStage::ApplyPatch].start(); self.db.apply_patch(patch); - tracing::debug!("Finished persisting to DB; took {:?}", started_at.elapsed()); + let stage_latency = stage_latency.observe(); + tracing::debug!("Finished persisting to DB; took {stage_latency:?}"); } /// Extends a tree with a chunk of entries. Unlike [`Self::extend_linear()`], entries may be @@ -172,15 +194,18 @@ impl MerkleTreeRecovery { )] pub fn extend_random(&mut self, entries: Vec) { tracing::debug!("Started extending tree"); + RECOVERY_METRICS.chunk_size.observe(entries.len()); - let started_at = Instant::now(); + let stage_latency = RECOVERY_METRICS.stage_latency[&RecoveryStage::Extend].start(); let storage = Storage::new(&self.db, &self.hasher, self.recovered_version, false); let patch = storage.extend_during_random_recovery(entries); - tracing::debug!("Finished processing keys; took {:?}", started_at.elapsed()); + let stage_latency = stage_latency.observe(); + tracing::debug!("Finished processing keys; took {stage_latency:?}"); - let started_at = Instant::now(); + let stage_latency = RECOVERY_METRICS.stage_latency[&RecoveryStage::ApplyPatch].start(); self.db.apply_patch(patch); - tracing::debug!("Finished persisting to DB; took {:?}", started_at.elapsed()); + let stage_latency = stage_latency.observe(); + tracing::debug!("Finished persisting to DB; took {stage_latency:?}"); } /// Finalizes the recovery process marking it as complete in the tree manifest. diff --git a/core/lib/merkle_tree/src/storage/serialization.rs b/core/lib/merkle_tree/src/storage/serialization.rs index 6ad6e1ff0b2f..f21fece94e09 100644 --- a/core/lib/merkle_tree/src/storage/serialization.rs +++ b/core/lib/merkle_tree/src/storage/serialization.rs @@ -1,6 +1,6 @@ //! Serialization of node types in the database. -use std::str; +use std::{collections::HashMap, str}; use crate::{ errors::{DeserializeError, DeserializeErrorKind, ErrorContext}, @@ -206,12 +206,14 @@ impl Node { impl TreeTags { /// Tags are serialized as a length-prefixed list of `(&str, &str)` tuples, where each /// `&str` is length-prefixed as well. All lengths are encoded using LEB128. + /// Custom tag keys are prefixed with `custom.` to ensure they don't intersect with standard tags. fn deserialize(bytes: &mut &[u8]) -> Result { let tag_count = leb128::read::unsigned(bytes).map_err(DeserializeErrorKind::Leb128)?; let mut architecture = None; let mut hasher = None; let mut depth = None; let mut is_recovering = false; + let mut custom = HashMap::new(); for _ in 0..tag_count { let key = Self::deserialize_str(bytes)?; @@ -237,7 +239,13 @@ impl TreeTags { })?; is_recovering = parsed; } - _ => return Err(DeserializeErrorKind::UnknownTag(key.to_owned()).into()), + key => { + if let Some(custom_key) = key.strip_prefix("custom.") { + custom.insert(custom_key.to_owned(), value.to_owned()); + } else { + return Err(DeserializeErrorKind::UnknownTag(key.to_owned()).into()); + } + } } } Ok(Self { @@ -245,6 +253,7 @@ impl TreeTags { hasher: hasher.ok_or(DeserializeErrorKind::MissingTag("hasher"))?, depth: depth.ok_or(DeserializeErrorKind::MissingTag("depth"))?, is_recovering, + custom, }) } @@ -266,8 +275,9 @@ impl TreeTags { } fn serialize(&self, buffer: &mut Vec) { - let entry_count = 3 + u64::from(self.is_recovering); + let entry_count = 3 + u64::from(self.is_recovering) + self.custom.len() as u64; leb128::write::unsigned(buffer, entry_count).unwrap(); + Self::serialize_str(buffer, "architecture"); Self::serialize_str(buffer, &self.architecture); Self::serialize_str(buffer, "depth"); @@ -278,6 +288,11 @@ impl TreeTags { Self::serialize_str(buffer, "is_recovering"); Self::serialize_str(buffer, "true"); } + + for (custom_key, value) in &self.custom { + Self::serialize_str(buffer, &format!("custom.{custom_key}")); + Self::serialize_str(buffer, value); + } } } @@ -347,6 +362,40 @@ mod tests { assert_eq!(manifest_copy, manifest); } + #[test] + fn serializing_manifest_with_custom_tags() { + let mut manifest = Manifest::new(42, &()); + // Test a single custom tag first to not deal with non-determinism when enumerating tags. + manifest.tags.as_mut().unwrap().custom = + HashMap::from([("test".to_owned(), "1".to_owned())]); + let mut buffer = vec![]; + manifest.serialize(&mut buffer); + assert_eq!(buffer[0], 42); // version count + assert_eq!(buffer[1], 4); // number of tags (3 standard + 1 custom) + assert_eq!( + buffer[2..], + *b"\x0Carchitecture\x06AR16MT\x05depth\x03256\x06hasher\x08no_op256\x0Bcustom.test\x011" + ); + + let manifest_copy = Manifest::deserialize(&buffer).unwrap(); + assert_eq!(manifest_copy, manifest); + + // Test multiple tags. + let tags = manifest.tags.as_mut().unwrap(); + tags.is_recovering = true; + tags.custom = HashMap::from([ + ("test".to_owned(), "1".to_owned()), + ("other.long.tag".to_owned(), "123456!!!".to_owned()), + ]); + let mut buffer = vec![]; + manifest.serialize(&mut buffer); + assert_eq!(buffer[0], 42); // version count + assert_eq!(buffer[1], 6); // number of tags (4 standard + 2 custom) + + let manifest_copy = Manifest::deserialize(&buffer).unwrap(); + assert_eq!(manifest_copy, manifest); + } + #[test] fn manifest_serialization_errors() { let manifest = Manifest::new(42, &()); diff --git a/core/lib/merkle_tree/src/types/internal.rs b/core/lib/merkle_tree/src/types/internal.rs index e8d307517363..e71465aa06db 100644 --- a/core/lib/merkle_tree/src/types/internal.rs +++ b/core/lib/merkle_tree/src/types/internal.rs @@ -2,7 +2,7 @@ //! some of these types are declared as public and can be even exported using the `unstable` module. //! Still, logically these types are private, so adding them to new public APIs etc. is a logical error. -use std::{fmt, num::NonZeroU64}; +use std::{collections::HashMap, fmt, num::NonZeroU64}; use crate::{ hasher::{HashTree, InternalNodeCache}, @@ -25,6 +25,8 @@ pub(crate) struct TreeTags { pub depth: usize, pub hasher: String, pub is_recovering: bool, + /// Custom / user-defined tags. + pub custom: HashMap, } impl TreeTags { @@ -36,6 +38,7 @@ impl TreeTags { hasher: hasher.name().to_owned(), depth: TREE_DEPTH, is_recovering: false, + custom: HashMap::new(), } } diff --git a/core/lib/multivm/src/glue/types/vm/vm_block_result.rs b/core/lib/multivm/src/glue/types/vm/vm_block_result.rs index 3f94157b7c73..824acc1ddfd2 100644 --- a/core/lib/multivm/src/glue/types/vm/vm_block_result.rs +++ b/core/lib/multivm/src/glue/types/vm/vm_block_result.rs @@ -71,7 +71,7 @@ impl GlueFrom for crate::interface::Fi }, final_bootloader_memory: None, pubdata_input: None, - initially_written_slots: None, + state_diffs: None, } } } @@ -131,7 +131,7 @@ impl GlueFrom for crate::interface::Fi }, final_bootloader_memory: None, pubdata_input: None, - initially_written_slots: None, + state_diffs: None, } } } @@ -189,7 +189,7 @@ impl GlueFrom for crate::interface: }, final_bootloader_memory: None, pubdata_input: None, - initially_written_slots: None, + state_diffs: None, } } } diff --git a/core/lib/multivm/src/interface/traits/vm.rs b/core/lib/multivm/src/interface/traits/vm.rs index 14047b4381d3..0e90a42e4888 100644 --- a/core/lib/multivm/src/interface/traits/vm.rs +++ b/core/lib/multivm/src/interface/traits/vm.rs @@ -143,7 +143,7 @@ pub trait VmInterface { final_execution_state: execution_state, final_bootloader_memory: Some(bootloader_memory), pubdata_input: None, - initially_written_slots: None, + state_diffs: None, } } } diff --git a/core/lib/multivm/src/interface/types/outputs/finished_l1batch.rs b/core/lib/multivm/src/interface/types/outputs/finished_l1batch.rs index 90cd0d195620..9c0afc6659f0 100644 --- a/core/lib/multivm/src/interface/types/outputs/finished_l1batch.rs +++ b/core/lib/multivm/src/interface/types/outputs/finished_l1batch.rs @@ -1,4 +1,4 @@ -use zksync_types::H256; +use zksync_types::writes::StateDiffRecord; use super::{BootloaderMemory, CurrentExecutionState, VmExecutionResultAndLogs}; @@ -13,7 +13,6 @@ pub struct FinishedL1Batch { pub final_bootloader_memory: Option, /// Pubdata to be published on L1. Could be none for old versions of the VM. pub pubdata_input: Option>, - /// List of hashed keys of slots that were initially written in the batch. - /// Could be none for old versions of the VM. - pub initially_written_slots: Option>, + /// List of state diffs. Could be none for old versions of the VM. + pub state_diffs: Option>, } diff --git a/core/lib/multivm/src/versions/vm_1_4_1/vm.rs b/core/lib/multivm/src/versions/vm_1_4_1/vm.rs index 07ff757f3efa..6f0c8e75745c 100644 --- a/core/lib/multivm/src/versions/vm_1_4_1/vm.rs +++ b/core/lib/multivm/src/versions/vm_1_4_1/vm.rs @@ -179,7 +179,7 @@ impl VmInterface for Vm { .clone() .build_pubdata(false), ), - initially_written_slots: None, + state_diffs: None, } } } diff --git a/core/lib/multivm/src/versions/vm_1_4_2/vm.rs b/core/lib/multivm/src/versions/vm_1_4_2/vm.rs index daa29d4059d8..917abcfe8aa0 100644 --- a/core/lib/multivm/src/versions/vm_1_4_2/vm.rs +++ b/core/lib/multivm/src/versions/vm_1_4_2/vm.rs @@ -3,7 +3,7 @@ use zksync_state::{StoragePtr, WriteStorage}; use zksync_types::{ event::extract_l2tol1logs_from_l1_messenger, l2_to_l1_log::{SystemL2ToL1Log, UserL2ToL1Log}, - Transaction, H256, + Transaction, }; use zksync_utils::bytecode::CompressedBytecodeInfo; @@ -179,17 +179,11 @@ impl VmInterface for Vm { .clone() .build_pubdata(false), ), - initially_written_slots: Some( + state_diffs: Some( self.bootloader_state .get_pubdata_information() .state_diffs - .iter() - .filter_map(|record| { - record - .is_write_initial() - .then_some(H256(record.derived_key)) - }) - .collect(), + .clone(), ), } } diff --git a/core/lib/multivm/src/versions/vm_boojum_integration/vm.rs b/core/lib/multivm/src/versions/vm_boojum_integration/vm.rs index db8528f58f3f..0d99b4d97b9d 100644 --- a/core/lib/multivm/src/versions/vm_boojum_integration/vm.rs +++ b/core/lib/multivm/src/versions/vm_boojum_integration/vm.rs @@ -179,7 +179,7 @@ impl VmInterface for Vm { .clone() .build_pubdata(false), ), - initially_written_slots: None, + state_diffs: None, } } } diff --git a/core/lib/multivm/src/versions/vm_latest/vm.rs b/core/lib/multivm/src/versions/vm_latest/vm.rs index 83805bdd18fc..fb0f3fb8d595 100644 --- a/core/lib/multivm/src/versions/vm_latest/vm.rs +++ b/core/lib/multivm/src/versions/vm_latest/vm.rs @@ -3,7 +3,7 @@ use zksync_state::{StoragePtr, WriteStorage}; use zksync_types::{ event::extract_l2tol1logs_from_l1_messenger, l2_to_l1_log::{SystemL2ToL1Log, UserL2ToL1Log}, - Transaction, VmVersion, H256, + Transaction, VmVersion, }; use zksync_utils::bytecode::CompressedBytecodeInfo; @@ -209,17 +209,11 @@ impl VmInterface for Vm { .clone() .build_pubdata(false), ), - initially_written_slots: Some( + state_diffs: Some( self.bootloader_state .get_pubdata_information() .state_diffs - .iter() - .filter_map(|record| { - record - .is_write_initial() - .then_some(H256(record.derived_key)) - }) - .collect(), + .clone(), ), } } diff --git a/core/lib/protobuf_config/src/eth.rs b/core/lib/protobuf_config/src/eth.rs index 8e7a9a6a880f..4ed5a8841436 100644 --- a/core/lib/protobuf_config/src/eth.rs +++ b/core/lib/protobuf_config/src/eth.rs @@ -24,24 +24,6 @@ impl proto::ProofSendingMode { } } -impl proto::ProofLoadingMode { - fn new(x: &configs::eth_sender::ProofLoadingMode) -> Self { - use configs::eth_sender::ProofLoadingMode as From; - match x { - From::OldProofFromDb => Self::OldProofFromDb, - From::FriProofFromGcs => Self::FriProofFromGcs, - } - } - - fn parse(&self) -> configs::eth_sender::ProofLoadingMode { - use configs::eth_sender::ProofLoadingMode as To; - match self { - Self::OldProofFromDb => To::OldProofFromDb, - Self::FriProofFromGcs => To::FriProofFromGcs, - } - } -} - impl proto::PubdataSendingMode { fn new(x: &configs::eth_sender::PubdataSendingMode) -> Self { use configs::eth_sender::PubdataSendingMode as From; @@ -127,10 +109,6 @@ impl ProtoRepr for proto::Sender { .and_then(|x| Ok(proto::PubdataSendingMode::try_from(*x)?)) .context("pubdata_sending_mode")? .parse(), - proof_loading_mode: required(&self.proof_loading_mode) - .and_then(|x| Ok(proto::ProofLoadingMode::try_from(*x)?)) - .context("proof_loading_mode")? - .parse(), }) } @@ -161,7 +139,6 @@ impl ProtoRepr for proto::Sender { pubdata_sending_mode: Some( proto::PubdataSendingMode::new(&this.pubdata_sending_mode).into(), ), - proof_loading_mode: Some(proto::ProofLoadingMode::new(&this.proof_loading_mode).into()), } } } diff --git a/core/lib/protobuf_config/src/proto/config/eth_sender.proto b/core/lib/protobuf_config/src/proto/config/eth_sender.proto index f6b3f4231e4c..1eb15f0679a4 100644 --- a/core/lib/protobuf_config/src/proto/config/eth_sender.proto +++ b/core/lib/protobuf_config/src/proto/config/eth_sender.proto @@ -43,7 +43,7 @@ message Sender { optional uint64 l1_batch_min_age_before_execute_seconds = 15; // optional; s optional uint64 max_acceptable_priority_fee_in_gwei = 16; // required; gwei optional PubdataSendingMode pubdata_sending_mode = 18; // required - optional ProofLoadingMode proof_loading_mode = 19; + reserved 19; reserved "proof_loading_mode"; } message GasAdjuster { diff --git a/core/lib/types/src/storage/writes/mod.rs b/core/lib/types/src/storage/writes/mod.rs index 83e8120268c6..ef19eeffed02 100644 --- a/core/lib/types/src/storage/writes/mod.rs +++ b/core/lib/types/src/storage/writes/mod.rs @@ -1,6 +1,6 @@ -use std::convert::TryInto; +use std::{convert::TryInto, fmt}; -use serde::{Deserialize, Serialize}; +use serde::{de, ser::SerializeTuple, Deserialize, Deserializer, Serialize, Serializer}; use zksync_basic_types::{Address, U256}; pub(crate) use self::compression::{compress_with_best_strategy, COMPRESSION_VERSION_NUMBER}; @@ -188,6 +188,77 @@ fn prepend_header(compressed_state_diffs: Vec) -> Vec { res.to_vec() } +/// Struct for storing tree writes in DB. +#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] +pub struct TreeWrite { + /// `address` part of storage key. + pub address: Address, + /// `key` part of storage key. + pub key: H256, + /// Value written. + pub value: H256, + /// Leaf index of the slot. + pub leaf_index: u64, +} + +impl Serialize for TreeWrite { + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + let mut tup = serializer.serialize_tuple(4)?; + tup.serialize_element(&self.address.0)?; + tup.serialize_element(&self.key.0)?; + tup.serialize_element(&self.value.0)?; + tup.serialize_element(&self.leaf_index)?; + tup.end() + } +} + +impl<'de> Deserialize<'de> for TreeWrite { + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + struct TreeWriteVisitor; + + impl<'de> de::Visitor<'de> for TreeWriteVisitor { + type Value = TreeWrite; + + fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { + formatter.write_str("a tuple of 4 elements") + } + + fn visit_seq(self, mut seq: V) -> Result + where + V: de::SeqAccess<'de>, + { + let address: [u8; 20] = seq + .next_element()? + .ok_or_else(|| de::Error::invalid_length(0, &self))?; + let key: [u8; 32] = seq + .next_element()? + .ok_or_else(|| de::Error::invalid_length(1, &self))?; + let value: [u8; 32] = seq + .next_element()? + .ok_or_else(|| de::Error::invalid_length(2, &self))?; + let leaf_index = seq + .next_element()? + .ok_or_else(|| de::Error::invalid_length(3, &self))?; + + Ok(TreeWrite { + address: Address::from_slice(&address), + key: H256::from_slice(&key), + value: H256::from_slice(&value), + leaf_index, + }) + } + } + + deserializer.deserialize_tuple(4, TreeWriteVisitor) + } +} + #[cfg(test)] mod tests { use std::{ @@ -515,4 +586,26 @@ mod tests { panic!("invalid operation id"); } } + + #[test] + fn check_tree_write_serde() { + let tree_write = TreeWrite { + address: Address::repeat_byte(0x11), + key: H256::repeat_byte(0x22), + value: H256::repeat_byte(0x33), + leaf_index: 1, + }; + + let serialized = bincode::serialize(&tree_write).unwrap(); + let expected: Vec<_> = vec![0x11u8; 20] + .into_iter() + .chain(vec![0x22u8; 32]) + .chain(vec![0x33u8; 32]) + .chain(1u64.to_le_bytes()) + .collect(); + assert_eq!(serialized, expected); + + let deserialized: TreeWrite = bincode::deserialize(&serialized).unwrap(); + assert_eq!(tree_write, deserialized); + } } diff --git a/core/lib/types/src/system_contracts.rs b/core/lib/types/src/system_contracts.rs index c802246da1d2..a28c45b8feae 100644 --- a/core/lib/types/src/system_contracts.rs +++ b/core/lib/types/src/system_contracts.rs @@ -180,7 +180,7 @@ static SYSTEM_CONTRACTS: Lazy> = Lazy::new(|| { .collect::>() }); -/// Gets default set of system contracts, based on ZKSYNC_HOME environment variable. +/// Gets default set of system contracts, based on Cargo workspace location. pub fn get_system_smart_contracts() -> Vec { SYSTEM_CONTRACTS.clone() } diff --git a/core/lib/utils/Cargo.toml b/core/lib/utils/Cargo.toml index 1fe736094e9d..4eea7d1398d1 100644 --- a/core/lib/utils/Cargo.toml +++ b/core/lib/utils/Cargo.toml @@ -25,9 +25,10 @@ futures.workspace = true hex.workspace = true reqwest = { workspace = true, features = ["blocking"] } itertools.workspace = true +serde_json.workspace = true +once_cell.workspace = true [dev-dependencies] -serde_json.workspace = true rand.workspace = true tokio = { workspace = true, features = ["macros", "rt"] } bincode.workspace = true diff --git a/core/lib/utils/src/env.rs b/core/lib/utils/src/env.rs new file mode 100644 index 000000000000..fec413927929 --- /dev/null +++ b/core/lib/utils/src/env.rs @@ -0,0 +1,68 @@ +use std::{ + path::{Path, PathBuf}, + str, +}; + +use anyhow::Context as _; +use once_cell::sync::OnceCell; + +static WORKSPACE: OnceCell> = OnceCell::new(); + +fn locate_workspace_inner() -> anyhow::Result { + let output = std::process::Command::new( + std::env::var("CARGO") + .ok() + .unwrap_or_else(|| "cargo".to_string()), + ) + .arg("locate-project") + .arg("--workspace") + .output() + .context("Can't find Cargo workspace location")?; + + let output = + serde_json::from_slice::(&output.stdout).with_context(|| { + format!( + "Error parsing `cargo locate-project` output {}", + str::from_utf8(&output.stdout).unwrap_or("(non-utf8 output)") + ) + })?; + let root = output.get("root").with_context(|| { + format!("root doesn't exist in output from `cargo locate-project` {output:?}") + })?; + + let serde_json::Value::String(root) = root else { + return Err(anyhow::anyhow!("`root` is not a string: {root:?}")); + }; + let root_path = PathBuf::from(root); + Ok(root_path + .parent() + .with_context(|| format!("`root` path doesn't have a parent: {}", root_path.display()))? + .to_path_buf()) +} + +/// Find the location of the current workspace, if this code works in workspace +/// then it will return the correct folder if, it's binary e.g. in docker container +/// you have to use fallback to another directory +/// The code has been inspired by `insta` +/// `https://github.com/mitsuhiko/insta/blob/master/insta/src/env.rs` +pub fn locate_workspace() -> Option<&'static Path> { + // Since `locate_workspace_inner()` should be deterministic, it makes little sense to call + // `OnceCell::get_or_try_init()` here; the repeated calls are just as unlikely to succeed as the initial call. + // Instead, we store `None` in the `OnceCell` if initialization failed. + WORKSPACE + .get_or_init(|| { + let result = locate_workspace_inner(); + if let Err(err) = &result { + // `get_or_init()` is guaranteed to call the provided closure once per `OnceCell`; + // i.e., we won't spam logs here. + tracing::warn!("locate_workspace() failed: {err:?}"); + } + result.ok() + }) + .as_deref() +} + +/// Returns [`locate_workspace()`] output with the "." fallback. +pub fn workspace_dir_or_current_dir() -> &'static Path { + locate_workspace().unwrap_or_else(|| Path::new(".")) +} diff --git a/core/lib/utils/src/lib.rs b/core/lib/utils/src/lib.rs index df26dbf6ab88..1c17d4efe264 100644 --- a/core/lib/utils/src/lib.rs +++ b/core/lib/utils/src/lib.rs @@ -2,6 +2,7 @@ pub mod bytecode; mod convert; +mod env; pub mod http_with_retries; pub mod misc; pub mod panic_extractor; @@ -9,6 +10,4 @@ mod serde_wrappers; pub mod time; pub mod wait_for_tasks; -pub use convert::*; -pub use misc::*; -pub use serde_wrappers::*; +pub use self::{convert::*, env::*, misc::*, serde_wrappers::*}; diff --git a/core/lib/zksync_core_leftovers/src/lib.rs b/core/lib/zksync_core_leftovers/src/lib.rs index 01358e05a8cf..49d1109e934f 100644 --- a/core/lib/zksync_core_leftovers/src/lib.rs +++ b/core/lib/zksync_core_leftovers/src/lib.rs @@ -42,16 +42,14 @@ use zksync_eth_sender::{Aggregator, EthTxAggregator, EthTxManager}; use zksync_eth_watch::{EthHttpQueryClient, EthWatch}; use zksync_health_check::{AppHealthCheck, HealthStatus, ReactiveHealthCheck}; use zksync_house_keeper::{ - blocks_state_reporter::L1BatchMetricsReporter, fri_gpu_prover_archiver::FriGpuProverArchiver, - fri_proof_compressor_job_retry_manager::FriProofCompressorJobRetryManager, - fri_proof_compressor_queue_monitor::FriProofCompressorStatsReporter, - fri_prover_job_retry_manager::FriProverJobRetryManager, - fri_prover_jobs_archiver::FriProverJobArchiver, - fri_prover_queue_monitor::FriProverStatsReporter, - fri_witness_generator_jobs_retry_manager::FriWitnessGeneratorJobRetryManager, - fri_witness_generator_queue_monitor::FriWitnessGeneratorStatsReporter, + blocks_state_reporter::L1BatchMetricsReporter, periodic_job::PeriodicJob, - waiting_to_queued_fri_witness_job_mover::WaitingToQueuedFriWitnessJobMover, + prover::{ + FriGpuProverArchiver, FriProofCompressorJobRetryManager, FriProofCompressorQueueReporter, + FriProverJobRetryManager, FriProverJobsArchiver, FriProverQueueReporter, + FriWitnessGeneratorJobRetryManager, FriWitnessGeneratorQueueReporter, + WaitingToQueuedFriWitnessJobMover, + }, }; use zksync_metadata_calculator::{ api_server::TreeApiHttpClient, MetadataCalculator, MetadataCalculatorConfig, @@ -72,6 +70,7 @@ use zksync_state::{PostgresStorageCaches, RocksdbStorageOptions}; use zksync_state_keeper::{ create_state_keeper, io::seal_logic::l2_block_seal_subtasks::L2BlockSealProcess, AsyncRocksdbCache, MempoolFetcher, MempoolGuard, OutputHandler, StateKeeperPersistence, + TreeWritesPersistence, }; use zksync_tee_verifier_input_producer::TeeVerifierInputProducer; use zksync_types::{ethabi::Contract, fee_model::FeeModelConfig, Address, L2ChainId}; @@ -765,8 +764,9 @@ pub async fn initialize_components( } if components.contains(&Component::CommitmentGenerator) { + let pool_size = CommitmentGenerator::default_parallelism().get(); let commitment_generator_pool = - ConnectionPool::::singleton(database_secrets.master_url()?) + ConnectionPool::::builder(database_secrets.master_url()?, pool_size) .build() .await .context("failed to build commitment_generator_pool")?; @@ -821,7 +821,7 @@ async fn add_state_keeper_to_task_futures( }; // L2 Block sealing process is parallelized, so we have to provide enough pooled connections. - let l2_block_sealer_pool = ConnectionPool::::builder( + let persistence_pool = ConnectionPool::::builder( database_secrets.master_url()?, L2BlockSealProcess::subtasks_len(), ) @@ -829,7 +829,7 @@ async fn add_state_keeper_to_task_futures( .await .context("failed to build l2_block_sealer_pool")?; let (persistence, l2_block_sealer) = StateKeeperPersistence::new( - l2_block_sealer_pool, + persistence_pool.clone(), contracts_config .l2_shared_bridge_addr .context("`l2_shared_bridge_addr` config is missing")?, @@ -854,6 +854,10 @@ async fn add_state_keeper_to_task_futures( db_config.state_keeper_db_path.clone(), cache_options, ); + + let tree_writes_persistence = TreeWritesPersistence::new(persistence_pool); + let output_handler = + OutputHandler::new(Box::new(persistence)).with_handler(Box::new(tree_writes_persistence)); let state_keeper = create_state_keeper( state_keeper_config, state_keeper_wallets, @@ -863,7 +867,7 @@ async fn add_state_keeper_to_task_futures( state_keeper_pool, mempool.clone(), batch_fee_input_provider.clone(), - OutputHandler::new(Box::new(persistence)), + output_handler, stop_receiver.clone(), ) .await; @@ -1041,8 +1045,12 @@ async fn add_tee_verifier_input_producer_to_task_futures( ) -> anyhow::Result<()> { let started_at = Instant::now(); tracing::info!("initializing TeeVerifierInputProducer"); - let producer = - TeeVerifierInputProducer::new(connection_pool.clone(), store_factory, l2_chain_id).await?; + let producer = TeeVerifierInputProducer::new( + connection_pool.clone(), + store_factory.create_store().await, + l2_chain_id, + ) + .await?; task_futures.push(tokio::spawn(producer.run(stop_receiver, None))); tracing::info!( "Initialized TeeVerifierInputProducer in {:?}", @@ -1131,7 +1139,7 @@ async fn add_house_keeper_to_task_futures( let task = waiting_to_queued_fri_witness_job_mover.run(stop_receiver.clone()); task_futures.push(tokio::spawn(task)); - let fri_witness_generator_stats_reporter = FriWitnessGeneratorStatsReporter::new( + let fri_witness_generator_stats_reporter = FriWitnessGeneratorQueueReporter::new( prover_connection_pool.clone(), house_keeper_config.witness_generator_stats_reporting_interval_ms, ); @@ -1142,7 +1150,7 @@ async fn add_house_keeper_to_task_futures( if let Some((archiving_interval, archive_after)) = house_keeper_config.prover_job_archiver_params() { - let fri_prover_jobs_archiver = FriProverJobArchiver::new( + let fri_prover_jobs_archiver = FriProverJobsArchiver::new( prover_connection_pool.clone(), archiving_interval, archive_after, @@ -1167,7 +1175,7 @@ async fn add_house_keeper_to_task_futures( .prover_group_config .clone() .context("fri_prover_group_config")?; - let fri_prover_stats_reporter = FriProverStatsReporter::new( + let fri_prover_stats_reporter = FriProverQueueReporter::new( house_keeper_config.prover_stats_reporting_interval_ms, prover_connection_pool.clone(), connection_pool.clone(), @@ -1180,7 +1188,7 @@ async fn add_house_keeper_to_task_futures( .proof_compressor_config .clone() .context("fri_proof_compressor_config")?; - let fri_proof_compressor_stats_reporter = FriProofCompressorStatsReporter::new( + let fri_proof_compressor_stats_reporter = FriProofCompressorQueueReporter::new( house_keeper_config.proof_compressor_stats_reporting_interval_ms, prover_connection_pool.clone(), ); diff --git a/core/node/api_server/src/web3/tests/ws.rs b/core/node/api_server/src/web3/tests/ws.rs index 93f6b536c34b..91a7c2595aeb 100644 --- a/core/node/api_server/src/web3/tests/ws.rs +++ b/core/node/api_server/src/web3/tests/ws.rs @@ -1,13 +1,14 @@ //! WS-related tests. -use std::collections::HashSet; +use std::{collections::HashSet, str::FromStr}; +use assert_matches::assert_matches; use async_trait::async_trait; use http::StatusCode; use tokio::sync::watch; use zksync_config::configs::chain::NetworkConfig; use zksync_dal::ConnectionPool; -use zksync_types::{api, Address, L1BatchNumber, H256, U64}; +use zksync_types::{api, Address, L1BatchNumber, H160, H2048, H256, U64}; use zksync_web3_decl::{ client::{WsClient, L2}, jsonrpsee::{ @@ -19,7 +20,7 @@ use zksync_web3_decl::{ rpc_params, }, namespaces::{EthNamespaceClient, ZksNamespaceClient}, - types::{BlockHeader, PubSubFilter}, + types::{BlockHeader, Bytes, PubSubFilter}, }; use super::*; @@ -290,15 +291,45 @@ impl WsTest for BasicSubscriptionsTest { .await .context("Timed out waiting for new block header")? .context("New blocks subscription terminated")??; + + let sha3_uncles_hash = + H256::from_str("0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347") + .unwrap(); + assert_eq!( received_block_header.number, Some(new_l2_block.number.0.into()) ); assert_eq!(received_block_header.hash, Some(new_l2_block.hash)); + assert_matches!(received_block_header.parent_hash, H256(_)); + assert_eq!(received_block_header.uncles_hash, sha3_uncles_hash); + assert_eq!(received_block_header.author, H160::zero()); + assert_eq!(received_block_header.state_root, H256::zero()); + assert_eq!(received_block_header.transactions_root, H256::zero()); + assert_eq!(received_block_header.receipts_root, H256::zero()); + assert_eq!( + received_block_header.number, + Some(U64::from(new_l2_block.number.0)) + ); + assert_matches!(received_block_header.gas_used, U256(_)); + assert_eq!( + received_block_header.gas_limit, + new_l2_block.gas_limit.into() + ); + assert_eq!( + received_block_header.base_fee_per_gas, + Some(new_l2_block.base_fee_per_gas.into()) + ); + assert_eq!(received_block_header.extra_data, Bytes::default()); + assert_eq!(received_block_header.logs_bloom, H2048::default()); assert_eq!( received_block_header.timestamp, new_l2_block.timestamp.into() ); + assert_eq!(received_block_header.difficulty, U256::zero()); + assert_eq!(received_block_header.mix_hash, None); + assert_eq!(received_block_header.nonce, None); + blocks_subscription.unsubscribe().await?; Ok(()) } diff --git a/core/node/commitment_generator/Cargo.toml b/core/node/commitment_generator/Cargo.toml index 45c62161e3f2..24752691348b 100644 --- a/core/node/commitment_generator/Cargo.toml +++ b/core/node/commitment_generator/Cargo.toml @@ -28,11 +28,16 @@ zk_evm_1_4_1.workspace = true zk_evm_1_3_3.workspace = true tokio = { workspace = true, features = ["time"] } +futures.workspace = true +num_cpus.workspace = true anyhow.workspace = true tracing.workspace = true itertools.workspace = true serde_json.workspace = true [dev-dependencies] -jsonrpsee.workspace = true zksync_web3_decl.workspace = true +zksync_node_genesis.workspace = true +zksync_node_test_utils.workspace = true + +rand.workspace = true diff --git a/core/node/commitment_generator/src/lib.rs b/core/node/commitment_generator/src/lib.rs index 866ef572b065..cbb6279481ca 100644 --- a/core/node/commitment_generator/src/lib.rs +++ b/core/node/commitment_generator/src/lib.rs @@ -1,4 +1,4 @@ -use std::time::Duration; +use std::{num::NonZeroU32, ops, sync::Arc, time::Duration}; use anyhow::Context; use itertools::Itertools; @@ -11,7 +11,7 @@ use zksync_types::{ blob::num_blobs_required, commitment::{ AuxCommitments, CommitmentCommonInput, CommitmentInput, L1BatchAuxiliaryOutput, - L1BatchCommitment, L1BatchCommitmentMode, + L1BatchCommitment, L1BatchCommitmentArtifacts, L1BatchCommitmentMode, }, event::convert_vm_events_to_log_queries, writes::{InitialStorageWrite, RepeatedStorageWrite, StateDiffRecord}, @@ -21,34 +21,56 @@ use zksync_utils::h256_to_u256; use crate::{ metrics::{CommitmentStage, METRICS}, - utils::{bootloader_initial_content_commitment, events_queue_commitment}, + utils::{CommitmentComputer, RealCommitmentComputer}, }; mod metrics; +#[cfg(test)] +mod tests; mod utils; pub mod validation_task; const SLEEP_INTERVAL: Duration = Duration::from_millis(100); +/// Component responsible for generating commitments for L1 batches. #[derive(Debug)] pub struct CommitmentGenerator { + computer: Arc, connection_pool: ConnectionPool, health_updater: HealthUpdater, commitment_mode: L1BatchCommitmentMode, + parallelism: NonZeroU32, } impl CommitmentGenerator { + /// Creates a commitment generator with the provided mode. pub fn new( connection_pool: ConnectionPool, commitment_mode: L1BatchCommitmentMode, ) -> Self { Self { + computer: Arc::new(RealCommitmentComputer), connection_pool, health_updater: ReactiveHealthCheck::new("commitment_generator").1, commitment_mode, + parallelism: Self::default_parallelism(), } } + /// Returns default parallelism for commitment generation based on the number of CPU cores available. + pub fn default_parallelism() -> NonZeroU32 { + // Leave at least one core free to handle other blocking tasks. `unwrap()`s are safe by design. + let cpus = u32::try_from(num_cpus::get().saturating_sub(1).clamp(1, 16)).unwrap(); + NonZeroU32::new(cpus).unwrap() + } + + /// Sets the degree of parallelism to be used by this generator. A reasonable value can be obtained + /// using [`Self::default_parallelism()`]. + pub fn set_max_parallelism(&mut self, parallelism: NonZeroU32) { + self.parallelism = parallelism; + } + + /// Returns a health check for this generator. pub fn health_check(&self) -> ReactiveHealthCheck { self.health_updater.subscribe() } @@ -82,25 +104,26 @@ impl CommitmentGenerator { })?; drop(connection); + let computer = self.computer.clone(); let events_commitment_task: JoinHandle> = tokio::task::spawn_blocking(move || { let latency = METRICS.events_queue_commitment_latency.start(); let events_queue_commitment = - events_queue_commitment(&events_queue, protocol_version) - .context("Events queue commitment is required for post-boojum batch")?; + computer.events_queue_commitment(&events_queue, protocol_version)?; latency.observe(); Ok(events_queue_commitment) }); + let computer = self.computer.clone(); let bootloader_memory_commitment_task: JoinHandle> = tokio::task::spawn_blocking(move || { let latency = METRICS.bootloader_content_commitment_latency.start(); - let bootloader_initial_content_commitment = bootloader_initial_content_commitment( - &initial_bootloader_contents, - protocol_version, - ) - .context("Bootloader content commitment is required for post-boojum batch")?; + let bootloader_initial_content_commitment = computer + .bootloader_initial_content_commitment( + &initial_bootloader_contents, + protocol_version, + )?; latency.observe(); Ok(bootloader_initial_content_commitment) @@ -262,7 +285,10 @@ impl CommitmentGenerator { Ok(input) } - async fn step(&self, l1_batch_number: L1BatchNumber) -> anyhow::Result<()> { + async fn process_batch( + &self, + l1_batch_number: L1BatchNumber, + ) -> anyhow::Result { let latency = METRICS.generate_commitment_latency_stage[&CommitmentStage::PrepareInput].start(); let input = self.prepare_input(l1_batch_number).await?; @@ -278,22 +304,45 @@ impl CommitmentGenerator { tracing::debug!( "Generated commitment artifacts for L1 batch #{l1_batch_number} in {latency:?}" ); + Ok(artifacts) + } - let latency = - METRICS.generate_commitment_latency_stage[&CommitmentStage::SaveResults].start(); - self.connection_pool + async fn step( + &self, + l1_batch_numbers: ops::RangeInclusive, + ) -> anyhow::Result<()> { + let iterable_numbers = + (l1_batch_numbers.start().0..=l1_batch_numbers.end().0).map(L1BatchNumber); + let batch_futures = iterable_numbers.map(|number| async move { + let artifacts = self + .process_batch(number) + .await + .with_context(|| format!("failed processing L1 batch #{number}"))?; + anyhow::Ok((number, artifacts)) + }); + let artifacts = futures::future::try_join_all(batch_futures).await?; + + let mut connection = self + .connection_pool .connection_tagged("commitment_generator") - .await? - .blocks_dal() - .save_l1_batch_commitment_artifacts(l1_batch_number, &artifacts) .await?; - let latency = latency.observe(); - tracing::debug!( - "Stored commitment artifacts for L1 batch #{l1_batch_number} in {latency:?}" - ); + // Saving changes atomically is not required here; since we save batches in order, if we encounter a DB error, + // the commitment generator will be able to recover gracefully. + for (l1_batch_number, artifacts) in artifacts { + let latency = + METRICS.generate_commitment_latency_stage[&CommitmentStage::SaveResults].start(); + connection + .blocks_dal() + .save_l1_batch_commitment_artifacts(l1_batch_number, &artifacts) + .await?; + let latency = latency.observe(); + tracing::debug!( + "Stored commitment artifacts for L1 batch #{l1_batch_number} in {latency:?}" + ); + } let health_details = serde_json::json!({ - "l1_batch_number": l1_batch_number, + "l1_batch_number": *l1_batch_numbers.end(), }); self.health_updater .update(Health::from(HealthStatus::Ready).with_details(health_details)); @@ -335,29 +384,72 @@ impl CommitmentGenerator { } } + async fn next_batch_range(&self) -> anyhow::Result>> { + let mut connection = self + .connection_pool + .connection_tagged("commitment_generator") + .await?; + let Some(next_batch_number) = connection + .blocks_dal() + .get_next_l1_batch_ready_for_commitment_generation() + .await? + else { + return Ok(None); + }; + + let Some(last_batch_number) = connection + .blocks_dal() + .get_last_l1_batch_ready_for_commitment_generation() + .await? + else { + return Ok(None); + }; + anyhow::ensure!( + next_batch_number <= last_batch_number, + "Unexpected node state: next L1 batch ready for commitment generation (#{next_batch_number}) is greater than \ + the last L1 batch ready for commitment generation (#{last_batch_number})" + ); + let last_batch_number = + last_batch_number.min(next_batch_number + self.parallelism.get() - 1); + Ok(Some(next_batch_number..=last_batch_number)) + } + + /// Runs this commitment generator indefinitely. It will process L1 batches added to the database + /// processed by the Merkle tree (or a tree fetcher), with a previously configured max parallelism. pub async fn run(self, stop_receiver: watch::Receiver) -> anyhow::Result<()> { + tracing::info!( + "Starting commitment generator with mode {:?} and parallelism {}", + self.commitment_mode, + self.parallelism + ); + if self.connection_pool.max_size() < self.parallelism.get() { + tracing::warn!( + "Connection pool for commitment generation has fewer connections ({pool_size}) than \ + configured max parallelism ({parallelism}); commitment generation may be slowed down as a result", + pool_size = self.connection_pool.max_size(), + parallelism = self.parallelism.get() + ); + } self.health_updater.update(HealthStatus::Ready.into()); + loop { if *stop_receiver.borrow() { tracing::info!("Stop signal received, commitment generator is shutting down"); break; } - let Some(l1_batch_number) = self - .connection_pool - .connection_tagged("commitment_generator") - .await? - .blocks_dal() - .get_next_l1_batch_ready_for_commitment_generation() - .await? - else { + let Some(l1_batch_numbers) = self.next_batch_range().await? else { tokio::time::sleep(SLEEP_INTERVAL).await; continue; }; - tracing::info!("Started commitment generation for L1 batch #{l1_batch_number}"); - self.step(l1_batch_number).await?; - tracing::info!("Finished commitment generation for L1 batch #{l1_batch_number}"); + tracing::info!("Started commitment generation for L1 batches #{l1_batch_numbers:?}"); + let step_latency = METRICS.step_latency.start(); + self.step(l1_batch_numbers.clone()).await?; + let step_latency = step_latency.observe(); + let batch_count = l1_batch_numbers.end().0 - l1_batch_numbers.start().0 + 1; + METRICS.step_batch_count.observe(batch_count.into()); + tracing::info!("Finished commitment generation for L1 batches #{l1_batch_numbers:?} in {step_latency:?} ({:?} per batch)", step_latency / batch_count); } Ok(()) } diff --git a/core/node/commitment_generator/src/metrics.rs b/core/node/commitment_generator/src/metrics.rs index 78cb82fff2bd..767e2874915b 100644 --- a/core/node/commitment_generator/src/metrics.rs +++ b/core/node/commitment_generator/src/metrics.rs @@ -10,19 +10,28 @@ pub(super) enum CommitmentStage { SaveResults, } +const BATCH_COUNT_BUCKETS: Buckets = Buckets::linear(1.0..=16.0, 1.0); + /// Metrics for the commitment generator. #[derive(Debug, Metrics)] #[metrics(prefix = "server_commitment_generator")] pub(super) struct CommitmentGeneratorMetrics { - /// Latency of generating commitment per stage. + /// Latency of generating commitment for a single L1 batch per stage. #[metrics(buckets = Buckets::LATENCIES, unit = Unit::Seconds)] pub generate_commitment_latency_stage: Family>, - /// Latency of generating bootloader content commitment. + /// Latency of generating bootloader content commitment for a single L1 batch. #[metrics(buckets = Buckets::LATENCIES, unit = Unit::Seconds)] pub bootloader_content_commitment_latency: Histogram, - /// Latency of generating events queue commitment. + /// Latency of generating events queue commitment for a single L1 batch. #[metrics(buckets = Buckets::LATENCIES, unit = Unit::Seconds)] pub events_queue_commitment_latency: Histogram, + + /// Latency of processing a continuous chunk of L1 batches during a single step of the generator. + #[metrics(buckets = Buckets::LATENCIES, unit = Unit::Seconds)] + pub step_latency: Histogram, + /// Number of L1 batches processed during a single step. + #[metrics(buckets = BATCH_COUNT_BUCKETS)] + pub step_batch_count: Histogram, } #[vise::register] diff --git a/core/node/commitment_generator/src/tests.rs b/core/node/commitment_generator/src/tests.rs new file mode 100644 index 000000000000..7f3c3eb2e2b1 --- /dev/null +++ b/core/node/commitment_generator/src/tests.rs @@ -0,0 +1,301 @@ +//! Tests for `CommitmentGenerator`. + +use std::thread; + +use rand::{thread_rng, Rng}; +use zksync_dal::Connection; +use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; +use zksync_node_test_utils::{create_l1_batch, create_l2_block}; +use zksync_types::{ + block::L1BatchTreeData, zk_evm_types::LogQuery, AccountTreeId, Address, StorageLog, +}; + +use super::*; + +async fn seal_l1_batch(storage: &mut Connection<'_, Core>, number: L1BatchNumber) { + let l2_block = create_l2_block(number.0); + storage + .blocks_dal() + .insert_l2_block(&l2_block) + .await + .unwrap(); + let storage_key = StorageKey::new( + AccountTreeId::new(Address::repeat_byte(1)), + H256::from_low_u64_be(number.0.into()), + ); + let storage_log = StorageLog::new_write_log(storage_key, H256::repeat_byte(0xff)); + storage + .storage_logs_dal() + .insert_storage_logs(l2_block.number, &[(H256::zero(), vec![storage_log])]) + .await + .unwrap(); + storage + .storage_logs_dedup_dal() + .insert_initial_writes(number, &[storage_key]) + .await + .unwrap(); + + let header = create_l1_batch(number.0); + storage + .blocks_dal() + .insert_mock_l1_batch(&header) + .await + .unwrap(); + storage + .blocks_dal() + .mark_l2_blocks_as_executed_in_l1_batch(number) + .await + .unwrap(); +} + +async fn save_l1_batch_tree_data(storage: &mut Connection<'_, Core>, number: L1BatchNumber) { + let tree_data = L1BatchTreeData { + hash: H256::from_low_u64_be(number.0.into()), + rollup_last_leaf_index: 20 + 10 * u64::from(number.0), + }; + storage + .blocks_dal() + .save_l1_batch_tree_data(number, &tree_data) + .await + .unwrap(); +} + +#[derive(Debug)] +struct MockCommitmentComputer { + delay: Duration, +} + +impl MockCommitmentComputer { + const EVENTS_QUEUE_COMMITMENT: H256 = H256::repeat_byte(1); + const BOOTLOADER_COMMITMENT: H256 = H256::repeat_byte(2); +} + +impl CommitmentComputer for MockCommitmentComputer { + fn events_queue_commitment( + &self, + _events_queue: &[LogQuery], + protocol_version: ProtocolVersionId, + ) -> anyhow::Result { + assert_eq!(protocol_version, ProtocolVersionId::latest()); + thread::sleep(self.delay); + Ok(Self::EVENTS_QUEUE_COMMITMENT) + } + + fn bootloader_initial_content_commitment( + &self, + _initial_bootloader_contents: &[(usize, U256)], + protocol_version: ProtocolVersionId, + ) -> anyhow::Result { + assert_eq!(protocol_version, ProtocolVersionId::latest()); + thread::sleep(self.delay); + Ok(Self::BOOTLOADER_COMMITMENT) + } +} + +fn create_commitment_generator(pool: ConnectionPool) -> CommitmentGenerator { + let mut generator = CommitmentGenerator::new(pool, L1BatchCommitmentMode::Rollup); + generator.computer = Arc::new(MockCommitmentComputer { + delay: Duration::from_millis(20), + }); + generator +} + +fn processed_batch(health: &Health, expected_number: L1BatchNumber) -> bool { + if !matches!(health.status(), HealthStatus::Ready) { + return false; + } + let Some(details) = health.details() else { + return false; + }; + *details == serde_json::json!({ "l1_batch_number": expected_number }) +} + +#[tokio::test] +async fn determining_batch_range() { + let pool = ConnectionPool::::test_pool().await; + let mut storage = pool.connection().await.unwrap(); + insert_genesis_batch(&mut storage, &GenesisParams::mock()) + .await + .unwrap(); + + let mut generator = create_commitment_generator(pool.clone()); + generator.parallelism = NonZeroU32::new(4).unwrap(); // to be deterministic + assert_eq!(generator.next_batch_range().await.unwrap(), None); + + seal_l1_batch(&mut storage, L1BatchNumber(1)).await; + assert_eq!(generator.next_batch_range().await.unwrap(), None); // No tree data for L1 batch #1 + + save_l1_batch_tree_data(&mut storage, L1BatchNumber(1)).await; + assert_eq!( + generator.next_batch_range().await.unwrap(), + Some(L1BatchNumber(1)..=L1BatchNumber(1)) + ); + + seal_l1_batch(&mut storage, L1BatchNumber(2)).await; + assert_eq!( + generator.next_batch_range().await.unwrap(), + Some(L1BatchNumber(1)..=L1BatchNumber(1)) + ); + + save_l1_batch_tree_data(&mut storage, L1BatchNumber(2)).await; + assert_eq!( + generator.next_batch_range().await.unwrap(), + Some(L1BatchNumber(1)..=L1BatchNumber(2)) + ); + + for number in 3..=5 { + seal_l1_batch(&mut storage, L1BatchNumber(number)).await; + } + assert_eq!( + generator.next_batch_range().await.unwrap(), + Some(L1BatchNumber(1)..=L1BatchNumber(2)) + ); + + for number in 3..=5 { + save_l1_batch_tree_data(&mut storage, L1BatchNumber(number)).await; + } + // L1 batch #5 is excluded because of the parallelism limit + assert_eq!( + generator.next_batch_range().await.unwrap(), + Some(L1BatchNumber(1)..=L1BatchNumber(4)) + ); +} + +#[tokio::test] +async fn commitment_generator_normal_operation() { + let pool = ConnectionPool::::test_pool().await; + let mut storage = pool.connection().await.unwrap(); + insert_genesis_batch(&mut storage, &GenesisParams::mock()) + .await + .unwrap(); + + let generator = create_commitment_generator(pool.clone()); + let mut health_check = generator.health_check(); + let (stop_sender, stop_receiver) = watch::channel(false); + let generator_handle = tokio::spawn(generator.run(stop_receiver)); + + for number in 1..=5 { + let number = L1BatchNumber(number); + seal_l1_batch(&mut storage, number).await; + save_l1_batch_tree_data(&mut storage, number).await; + // Wait until the batch is processed by the generator + health_check + .wait_for(|health| processed_batch(health, number)) + .await; + // Check data in Postgres + let metadata = storage + .blocks_dal() + .get_l1_batch_metadata(number) + .await + .unwrap() + .expect("no batch metadata"); + assert_eq!( + metadata.metadata.events_queue_commitment, + Some(MockCommitmentComputer::EVENTS_QUEUE_COMMITMENT) + ); + assert_eq!( + metadata.metadata.bootloader_initial_content_commitment, + Some(MockCommitmentComputer::BOOTLOADER_COMMITMENT) + ); + } + + stop_sender.send_replace(true); + generator_handle.await.unwrap().unwrap(); +} + +#[tokio::test] +async fn commitment_generator_bulk_processing() { + let pool = ConnectionPool::::test_pool().await; + let mut storage = pool.connection().await.unwrap(); + insert_genesis_batch(&mut storage, &GenesisParams::mock()) + .await + .unwrap(); + + for number in 1..=5 { + seal_l1_batch(&mut storage, L1BatchNumber(number)).await; + save_l1_batch_tree_data(&mut storage, L1BatchNumber(number)).await; + } + + let mut generator = create_commitment_generator(pool.clone()); + generator.parallelism = NonZeroU32::new(10).unwrap(); // enough to process all batches at once + let mut health_check = generator.health_check(); + let (stop_sender, stop_receiver) = watch::channel(false); + let generator_handle = tokio::spawn(generator.run(stop_receiver)); + + health_check + .wait_for(|health| processed_batch(health, L1BatchNumber(5))) + .await; + for number in 1..=5 { + let metadata = storage + .blocks_dal() + .get_l1_batch_metadata(L1BatchNumber(number)) + .await + .unwrap() + .expect("no batch metadata"); + assert_eq!( + metadata.metadata.events_queue_commitment, + Some(MockCommitmentComputer::EVENTS_QUEUE_COMMITMENT) + ); + assert_eq!( + metadata.metadata.bootloader_initial_content_commitment, + Some(MockCommitmentComputer::BOOTLOADER_COMMITMENT) + ); + } + + stop_sender.send_replace(true); + generator_handle.await.unwrap().unwrap(); +} + +#[tokio::test] +async fn commitment_generator_with_tree_emulation() { + let pool = ConnectionPool::::test_pool().await; + let mut storage = pool.connection().await.unwrap(); + insert_genesis_batch(&mut storage, &GenesisParams::mock()) + .await + .unwrap(); + drop(storage); + + // Emulates adding new batches to the storage. + let new_batches_pool = pool.clone(); + let new_batches_handle = tokio::spawn(async move { + for number in 1..=10 { + let sleep_delay = Duration::from_millis(thread_rng().gen_range(1..20)); + tokio::time::sleep(sleep_delay).await; + let mut storage = new_batches_pool.connection().await.unwrap(); + seal_l1_batch(&mut storage, L1BatchNumber(number)).await; + } + }); + + let tree_emulator_pool = pool.clone(); + let tree_emulator_handle = tokio::spawn(async move { + for number in 1..=10 { + let mut storage = tree_emulator_pool.connection().await.unwrap(); + while storage + .blocks_dal() + .get_sealed_l1_batch_number() + .await + .unwrap() + < Some(L1BatchNumber(number)) + { + let sleep_delay = Duration::from_millis(thread_rng().gen_range(5..10)); + tokio::time::sleep(sleep_delay).await; + } + save_l1_batch_tree_data(&mut storage, L1BatchNumber(number)).await; + } + }); + + let mut generator = create_commitment_generator(pool.clone()); + generator.parallelism = NonZeroU32::new(10).unwrap(); // enough to process all batches at once + let mut health_check = generator.health_check(); + let (stop_sender, stop_receiver) = watch::channel(false); + let generator_handle = tokio::spawn(generator.run(stop_receiver)); + + health_check + .wait_for(|health| processed_batch(health, L1BatchNumber(10))) + .await; + + new_batches_handle.await.unwrap(); + tree_emulator_handle.await.unwrap(); + stop_sender.send_replace(true); + generator_handle.await.unwrap().unwrap(); +} diff --git a/core/node/commitment_generator/src/utils.rs b/core/node/commitment_generator/src/utils.rs index 433d1345903e..9a12f0c43165 100644 --- a/core/node/commitment_generator/src/utils.rs +++ b/core/node/commitment_generator/src/utils.rs @@ -1,4 +1,7 @@ //! Utils for commitment calculation. + +use std::fmt; + use multivm::utils::get_used_bootloader_memory_bytes; use zk_evm_1_3_3::{ aux_structures::Timestamp as Timestamp_1_3_3, @@ -15,73 +18,96 @@ use zk_evm_1_5_0::{ use zksync_types::{zk_evm_types::LogQuery, ProtocolVersionId, VmVersion, H256, U256}; use zksync_utils::expand_memory_contents; -pub fn events_queue_commitment( - events_queue: &[LogQuery], - protocol_version: ProtocolVersionId, -) -> Option { - match VmVersion::from(protocol_version) { - VmVersion::VmBoojumIntegration => Some(H256( - circuit_sequencer_api_1_4_0::commitments::events_queue_commitment_fixed( - &events_queue - .iter() - .map(|x| to_log_query_1_3_3(*x)) - .collect(), - ), - )), - VmVersion::Vm1_4_1 | VmVersion::Vm1_4_2 => Some(H256( - circuit_sequencer_api_1_4_1::commitments::events_queue_commitment_fixed( - &events_queue - .iter() - .map(|x| to_log_query_1_4_1(*x)) - .collect(), - ), - )), - VmVersion::Vm1_5_0SmallBootloaderMemory | VmVersion::Vm1_5_0IncreasedBootloaderMemory => { - Some(H256( +/// Encapsulates computations of commitment components. +/// +/// - All methods are considered to be blocking. +/// - Returned errors are considered unrecoverable (i.e., they bubble up and lead to commitment generator termination). +pub(crate) trait CommitmentComputer: fmt::Debug + Send + Sync + 'static { + fn events_queue_commitment( + &self, + events_queue: &[LogQuery], + protocol_version: ProtocolVersionId, + ) -> anyhow::Result; + + fn bootloader_initial_content_commitment( + &self, + initial_bootloader_contents: &[(usize, U256)], + protocol_version: ProtocolVersionId, + ) -> anyhow::Result; +} + +#[derive(Debug)] +pub(crate) struct RealCommitmentComputer; + +impl CommitmentComputer for RealCommitmentComputer { + fn events_queue_commitment( + &self, + events_queue: &[LogQuery], + protocol_version: ProtocolVersionId, + ) -> anyhow::Result { + match VmVersion::from(protocol_version) { + VmVersion::VmBoojumIntegration => Ok(H256( + circuit_sequencer_api_1_4_0::commitments::events_queue_commitment_fixed( + &events_queue + .iter() + .map(|x| to_log_query_1_3_3(*x)) + .collect(), + ), + )), + VmVersion::Vm1_4_1 | VmVersion::Vm1_4_2 => Ok(H256( + circuit_sequencer_api_1_4_1::commitments::events_queue_commitment_fixed( + &events_queue + .iter() + .map(|x| to_log_query_1_4_1(*x)) + .collect(), + ), + )), + VmVersion::Vm1_5_0SmallBootloaderMemory + | VmVersion::Vm1_5_0IncreasedBootloaderMemory => Ok(H256( circuit_sequencer_api_1_5_0::commitments::events_queue_commitment_fixed( &events_queue .iter() .map(|x| to_log_query_1_5_0(*x)) .collect(), ), - )) + )), + _ => anyhow::bail!("Unsupported protocol version: {protocol_version:?}"), } - _ => None, } -} -pub fn bootloader_initial_content_commitment( - initial_bootloader_contents: &[(usize, U256)], - protocol_version: ProtocolVersionId, -) -> Option { - let expanded_memory_size = if protocol_version.is_pre_boojum() { - return None; - } else { - get_used_bootloader_memory_bytes(protocol_version.into()) - }; + fn bootloader_initial_content_commitment( + &self, + initial_bootloader_contents: &[(usize, U256)], + protocol_version: ProtocolVersionId, + ) -> anyhow::Result { + let expanded_memory_size = if protocol_version.is_pre_boojum() { + anyhow::bail!("Unsupported protocol version: {protocol_version:?}"); + } else { + get_used_bootloader_memory_bytes(protocol_version.into()) + }; - let full_bootloader_memory = - expand_memory_contents(initial_bootloader_contents, expanded_memory_size); + let full_bootloader_memory = + expand_memory_contents(initial_bootloader_contents, expanded_memory_size); - match VmVersion::from(protocol_version) { - VmVersion::VmBoojumIntegration => Some(H256( - circuit_sequencer_api_1_4_0::commitments::initial_heap_content_commitment_fixed( - &full_bootloader_memory, - ), - )), - VmVersion::Vm1_4_1 | VmVersion::Vm1_4_2 => Some(H256( - circuit_sequencer_api_1_4_1::commitments::initial_heap_content_commitment_fixed( - &full_bootloader_memory, - ), - )), - VmVersion::Vm1_5_0SmallBootloaderMemory | VmVersion::Vm1_5_0IncreasedBootloaderMemory => { - Some(H256( + match VmVersion::from(protocol_version) { + VmVersion::VmBoojumIntegration => Ok(H256( + circuit_sequencer_api_1_4_0::commitments::initial_heap_content_commitment_fixed( + &full_bootloader_memory, + ), + )), + VmVersion::Vm1_4_1 | VmVersion::Vm1_4_2 => Ok(H256( + circuit_sequencer_api_1_4_1::commitments::initial_heap_content_commitment_fixed( + &full_bootloader_memory, + ), + )), + VmVersion::Vm1_5_0SmallBootloaderMemory + | VmVersion::Vm1_5_0IncreasedBootloaderMemory => Ok(H256( circuit_sequencer_api_1_5_0::commitments::initial_heap_content_commitment_fixed( &full_bootloader_memory, ), - )) + )), + _ => unreachable!(), } - _ => unreachable!(), } } diff --git a/core/node/commitment_generator/src/validation_task.rs b/core/node/commitment_generator/src/validation_task.rs index 902e3f7cdf5e..cf93a4899b89 100644 --- a/core/node/commitment_generator/src/validation_task.rs +++ b/core/node/commitment_generator/src/validation_task.rs @@ -124,10 +124,9 @@ impl L1BatchCommitmentModeValidationTask { mod tests { use std::{mem, sync::Mutex}; - use jsonrpsee::types::ErrorObject; use zksync_eth_client::clients::MockEthereum; use zksync_types::{ethabi, U256}; - use zksync_web3_decl::client::MockClient; + use zksync_web3_decl::{client::MockClient, jsonrpsee::types::ErrorObject}; use super::*; diff --git a/core/node/consensus/src/testonly.rs b/core/node/consensus/src/testonly.rs index 6f064d66efce..3b990bf088fe 100644 --- a/core/node/consensus/src/testonly.rs +++ b/core/node/consensus/src/testonly.rs @@ -22,8 +22,8 @@ use zksync_node_test_utils::{create_l1_batch_metadata, create_l2_transaction}; use zksync_state_keeper::{ io::{IoCursor, L1BatchParams, L2BlockParams}, seal_criteria::NoopSealer, - testonly::MockBatchExecutor, - OutputHandler, StateKeeperPersistence, ZkSyncStateKeeper, + testonly::{test_batch_executor::MockReadStorageFactory, MockBatchExecutor}, + OutputHandler, StateKeeperPersistence, TreeWritesPersistence, ZkSyncStateKeeper, }; use zksync_types::{Address, L1BatchNumber, L2BlockNumber, L2ChainId, ProtocolVersionId}; use zksync_web3_decl::client::{Client, DynClient, L2}; @@ -312,6 +312,7 @@ impl StateKeeperRunner { let (stop_send, stop_recv) = sync::watch::channel(false); let (persistence, l2_block_sealer) = StateKeeperPersistence::new(self.pool.0.clone(), Address::repeat_byte(11), 5); + let tree_writes_persistence = TreeWritesPersistence::new(self.pool.0.clone()); let io = ExternalIO::new( self.pool.0.clone(), @@ -342,8 +343,10 @@ impl StateKeeperRunner { Box::new(io), Box::new(MockBatchExecutor), OutputHandler::new(Box::new(persistence.with_tx_insertion())) + .with_handler(Box::new(tree_writes_persistence)) .with_handler(Box::new(self.sync_state.clone())), Arc::new(NoopSealer), + Arc::new(MockReadStorageFactory), ) .run() .await diff --git a/core/node/eth_sender/src/aggregator.rs b/core/node/eth_sender/src/aggregator.rs index aa9b31abd426..5e4696f3bcbe 100644 --- a/core/node/eth_sender/src/aggregator.rs +++ b/core/node/eth_sender/src/aggregator.rs @@ -1,6 +1,6 @@ use std::sync::Arc; -use zksync_config::configs::eth_sender::{ProofLoadingMode, ProofSendingMode, SenderConfig}; +use zksync_config::configs::eth_sender::{ProofSendingMode, SenderConfig}; use zksync_contracts::BaseSystemContractsHashes; use zksync_dal::{Connection, Core, CoreDal}; use zksync_l1_contract_interface::i_executor::methods::{ExecuteBatches, ProveBatches}; @@ -292,7 +292,6 @@ impl Aggregator { async fn load_real_proof_operation( storage: &mut Connection<'_, Core>, l1_verifier_config: L1VerifierConfig, - proof_loading_mode: &ProofLoadingMode, blob_store: &dyn ObjectStore, is_4844_mode: bool, ) -> Option { @@ -336,14 +335,9 @@ impl Aggregator { return None; } } - let proofs = match proof_loading_mode { - ProofLoadingMode::OldProofFromDb => { - unreachable!("OldProofFromDb is not supported anymore") - } - ProofLoadingMode::FriProofFromGcs => { - load_wrapped_fri_proofs_for_range(batch_to_prove, batch_to_prove, blob_store).await - } - }; + let proofs = + load_wrapped_fri_proofs_for_range(batch_to_prove, batch_to_prove, blob_store).await; + if proofs.is_empty() { // The proof for the next L1 batch is not generated yet return None; @@ -423,7 +417,6 @@ impl Aggregator { Self::load_real_proof_operation( storage, l1_verifier_config, - &self.config.proof_loading_mode, &*self.blob_store, self.operate_4844_mode, ) @@ -446,7 +439,6 @@ impl Aggregator { if let Some(op) = Self::load_real_proof_operation( storage, l1_verifier_config, - &self.config.proof_loading_mode, &*self.blob_store, self.operate_4844_mode, ) diff --git a/core/node/house_keeper/src/lib.rs b/core/node/house_keeper/src/lib.rs index e98c7708201d..68d4ad2f8ba4 100644 --- a/core/node/house_keeper/src/lib.rs +++ b/core/node/house_keeper/src/lib.rs @@ -1,12 +1,3 @@ pub mod blocks_state_reporter; -pub mod fri_gpu_prover_archiver; -pub mod fri_proof_compressor_job_retry_manager; -pub mod fri_proof_compressor_queue_monitor; -pub mod fri_prover_job_retry_manager; -pub mod fri_prover_jobs_archiver; -pub mod fri_prover_queue_monitor; -pub mod fri_witness_generator_jobs_retry_manager; -pub mod fri_witness_generator_queue_monitor; -mod metrics; pub mod periodic_job; -pub mod waiting_to_queued_fri_witness_job_mover; +pub mod prover; diff --git a/core/node/house_keeper/src/fri_gpu_prover_archiver.rs b/core/node/house_keeper/src/prover/archiver/fri_gpu_prover_archiver.rs similarity index 83% rename from core/node/house_keeper/src/fri_gpu_prover_archiver.rs rename to core/node/house_keeper/src/prover/archiver/fri_gpu_prover_archiver.rs index 11c727011cd4..2af66a937b33 100644 --- a/core/node/house_keeper/src/fri_gpu_prover_archiver.rs +++ b/core/node/house_keeper/src/prover/archiver/fri_gpu_prover_archiver.rs @@ -1,10 +1,11 @@ use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use crate::{metrics::HOUSE_KEEPER_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::HOUSE_KEEPER_METRICS}; -/// FriGpuProverArchiver is a task that periodically archives old fri GPU prover records. +/// `FriGpuProverArchiver` is a task that periodically archives old fri GPU prover records. /// The task will archive the `dead` prover records that have not been updated for a certain amount of time. +/// Note: These components speed up provers, in their absence, queries would become sub optimal. #[derive(Debug)] pub struct FriGpuProverArchiver { pool: ConnectionPool, diff --git a/core/node/house_keeper/src/fri_prover_jobs_archiver.rs b/core/node/house_keeper/src/prover/archiver/fri_prover_jobs_archiver.rs similarity index 66% rename from core/node/house_keeper/src/fri_prover_jobs_archiver.rs rename to core/node/house_keeper/src/prover/archiver/fri_prover_jobs_archiver.rs index 5ec98f2178da..8e3134c078f2 100644 --- a/core/node/house_keeper/src/fri_prover_jobs_archiver.rs +++ b/core/node/house_keeper/src/prover/archiver/fri_prover_jobs_archiver.rs @@ -1,16 +1,19 @@ use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use crate::{metrics::HOUSE_KEEPER_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::HOUSE_KEEPER_METRICS}; +/// `FriProverJobsArchiver` is a task that periodically archives old finalized prover job. +/// The task will archive the `successful` prover jobs that have been done for a certain amount of time. +/// Note: These components speed up provers, in their absence, queries would become sub optimal. #[derive(Debug)] -pub struct FriProverJobArchiver { +pub struct FriProverJobsArchiver { pool: ConnectionPool, reporting_interval_ms: u64, archiving_interval_secs: u64, } -impl FriProverJobArchiver { +impl FriProverJobsArchiver { pub fn new( pool: ConnectionPool, reporting_interval_ms: u64, @@ -25,8 +28,8 @@ impl FriProverJobArchiver { } #[async_trait::async_trait] -impl PeriodicJob for FriProverJobArchiver { - const SERVICE_NAME: &'static str = "FriProverJobArchiver"; +impl PeriodicJob for FriProverJobsArchiver { + const SERVICE_NAME: &'static str = "FriProverJobsArchiver"; async fn run_routine_task(&mut self) -> anyhow::Result<()> { let archived_jobs = self diff --git a/core/node/house_keeper/src/prover/archiver/mod.rs b/core/node/house_keeper/src/prover/archiver/mod.rs new file mode 100644 index 000000000000..36b82a7735ce --- /dev/null +++ b/core/node/house_keeper/src/prover/archiver/mod.rs @@ -0,0 +1,5 @@ +mod fri_gpu_prover_archiver; +mod fri_prover_jobs_archiver; + +pub use fri_gpu_prover_archiver::FriGpuProverArchiver; +pub use fri_prover_jobs_archiver::FriProverJobsArchiver; diff --git a/core/node/house_keeper/src/metrics.rs b/core/node/house_keeper/src/prover/metrics.rs similarity index 71% rename from core/node/house_keeper/src/metrics.rs rename to core/node/house_keeper/src/prover/metrics.rs index b47031a0f10b..4af13b61b0c5 100644 --- a/core/node/house_keeper/src/metrics.rs +++ b/core/node/house_keeper/src/prover/metrics.rs @@ -1,4 +1,5 @@ use vise::{Counter, EncodeLabelSet, EncodeLabelValue, Family, Gauge, LabeledFamily, Metrics}; +use zksync_types::ProtocolVersionId; #[derive(Debug, Metrics)] #[metrics(prefix = "house_keeper")] @@ -10,8 +11,8 @@ pub(crate) struct HouseKeeperMetrics { #[vise::register] pub(crate) static HOUSE_KEEPER_METRICS: vise::Global = vise::Global::new(); -#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, EncodeLabelValue, EncodeLabelSet)] -#[metrics(label = "type", rename_all = "snake_case")] +#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, EncodeLabelValue)] +#[metrics(rename_all = "snake_case")] #[allow(dead_code)] pub enum JobStatus { Queued, @@ -26,22 +27,27 @@ pub enum JobStatus { #[metrics(prefix = "prover_fri")] pub(crate) struct ProverFriMetrics { pub proof_compressor_requeued_jobs: Counter, - pub proof_compressor_jobs: Family>, + #[metrics(labels = ["type", "protocol_version"])] + pub proof_compressor_jobs: LabeledFamily<(JobStatus, String), Gauge, 2>, pub proof_compressor_oldest_uncompressed_batch: Gauge, } #[vise::register] pub(crate) static PROVER_FRI_METRICS: vise::Global = vise::Global::new(); -const PROVER_JOBS_LABELS: [&str; 4] = - ["type", "circuit_id", "aggregation_round", "prover_group_id"]; -type ProverJobsLabels = (&'static str, String, String, String); +#[derive(Debug, Clone, PartialEq, Eq, Hash, EncodeLabelSet)] +pub(crate) struct ProverJobsLabels { + pub r#type: &'static str, + pub circuit_id: String, + pub aggregation_round: String, + pub prover_group_id: String, + pub protocol_version: String, +} #[derive(Debug, Metrics)] #[metrics(prefix = "fri_prover")] pub(crate) struct FriProverMetrics { - #[metrics(labels = PROVER_JOBS_LABELS)] - pub prover_jobs: LabeledFamily, 4>, + pub prover_jobs: Family>, #[metrics(labels = ["circuit_id", "aggregation_round"])] pub block_number: LabeledFamily<(String, String), Gauge, 2>, pub oldest_unpicked_batch: Gauge, @@ -53,18 +59,20 @@ pub(crate) struct FriProverMetrics { impl FriProverMetrics { pub fn report_prover_jobs( &self, - status: &'static str, + r#type: &'static str, circuit_id: u8, aggregation_round: u8, prover_group_id: u8, + protocol_version: ProtocolVersionId, amount: u64, ) { - self.prover_jobs[&( - status, - circuit_id.to_string(), - aggregation_round.to_string(), - prover_group_id.to_string(), - )] + self.prover_jobs[&ProverJobsLabels { + r#type, + circuit_id: circuit_id.to_string(), + aggregation_round: aggregation_round.to_string(), + prover_group_id: prover_group_id.to_string(), + protocol_version: protocol_version.to_string(), + }] .set(amount); } } @@ -101,10 +109,11 @@ impl From<&str> for WitnessType { pub(crate) struct ServerMetrics { pub prover_fri_requeued_jobs: Counter, pub requeued_jobs: Family>, - #[metrics(labels = ["type", "round"])] - pub witness_generator_jobs_by_round: LabeledFamily<(&'static str, String), Gauge, 2>, - #[metrics(labels = ["type"])] - pub witness_generator_jobs: LabeledFamily<&'static str, Gauge>, + #[metrics(labels = ["type", "round", "protocol_version"])] + pub witness_generator_jobs_by_round: + LabeledFamily<(&'static str, String, String), Gauge, 3>, + #[metrics(labels = ["type", "protocol_version"])] + pub witness_generator_jobs: LabeledFamily<(&'static str, String), Gauge, 2>, pub leaf_fri_witness_generator_waiting_to_queued_jobs_transitions: Counter, pub node_fri_witness_generator_waiting_to_queued_jobs_transitions: Counter, pub recursion_tip_witness_generator_waiting_to_queued_jobs_transitions: Counter, diff --git a/core/node/house_keeper/src/prover/mod.rs b/core/node/house_keeper/src/prover/mod.rs new file mode 100644 index 000000000000..af315c53cb48 --- /dev/null +++ b/core/node/house_keeper/src/prover/mod.rs @@ -0,0 +1,14 @@ +mod archiver; +mod metrics; +mod queue_reporter; +mod retry_manager; +mod waiting_to_queued_fri_witness_job_mover; + +pub use archiver::{FriGpuProverArchiver, FriProverJobsArchiver}; +pub use queue_reporter::{ + FriProofCompressorQueueReporter, FriProverQueueReporter, FriWitnessGeneratorQueueReporter, +}; +pub use retry_manager::{ + FriProofCompressorJobRetryManager, FriProverJobRetryManager, FriWitnessGeneratorJobRetryManager, +}; +pub use waiting_to_queued_fri_witness_job_mover::WaitingToQueuedFriWitnessJobMover; diff --git a/core/node/house_keeper/src/fri_proof_compressor_queue_monitor.rs b/core/node/house_keeper/src/prover/queue_reporter/fri_proof_compressor_queue_reporter.rs similarity index 67% rename from core/node/house_keeper/src/fri_proof_compressor_queue_monitor.rs rename to core/node/house_keeper/src/prover/queue_reporter/fri_proof_compressor_queue_reporter.rs index b9f0bc47704f..06f7a357e893 100644 --- a/core/node/house_keeper/src/fri_proof_compressor_queue_monitor.rs +++ b/core/node/house_keeper/src/prover/queue_reporter/fri_proof_compressor_queue_reporter.rs @@ -1,20 +1,22 @@ use async_trait::async_trait; use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use zksync_types::prover_dal::JobCountStatistics; +use zksync_types::{prover_dal::JobCountStatistics, ProtocolVersionId}; use crate::{ - metrics::{JobStatus, PROVER_FRI_METRICS}, periodic_job::PeriodicJob, + prover::metrics::{JobStatus, PROVER_FRI_METRICS}, }; +/// `FriProofCompressorQueueReporter` is a task that periodically reports compression jobs status. +/// Note: these values will be used for auto-scaling proof compressor #[derive(Debug)] -pub struct FriProofCompressorStatsReporter { +pub struct FriProofCompressorQueueReporter { reporting_interval_ms: u64, pool: ConnectionPool, } -impl FriProofCompressorStatsReporter { +impl FriProofCompressorQueueReporter { pub fn new(reporting_interval_ms: u64, pool: ConnectionPool) -> Self { Self { reporting_interval_ms, @@ -32,11 +34,9 @@ impl FriProofCompressorStatsReporter { } } -/// Invoked periodically to push job statistics to Prometheus -/// Note: these values will be used for auto-scaling proof compressor #[async_trait] -impl PeriodicJob for FriProofCompressorStatsReporter { - const SERVICE_NAME: &'static str = "ProofCompressorStatsReporter"; +impl PeriodicJob for FriProofCompressorQueueReporter { + const SERVICE_NAME: &'static str = "FriProofCompressorQueueReporter"; async fn run_routine_task(&mut self) -> anyhow::Result<()> { let stats = Self::get_job_statistics(&self.pool).await; @@ -49,8 +49,16 @@ impl PeriodicJob for FriProofCompressorStatsReporter { ); } - PROVER_FRI_METRICS.proof_compressor_jobs[&JobStatus::Queued].set(stats.queued as u64); - PROVER_FRI_METRICS.proof_compressor_jobs[&JobStatus::InProgress] + PROVER_FRI_METRICS.proof_compressor_jobs[&( + JobStatus::Queued, + ProtocolVersionId::current_prover_version().to_string(), + )] + .set(stats.queued as u64); + + PROVER_FRI_METRICS.proof_compressor_jobs[&( + JobStatus::InProgress, + ProtocolVersionId::current_prover_version().to_string(), + )] .set(stats.in_progress as u64); let oldest_not_compressed_batch = self diff --git a/core/node/house_keeper/src/fri_prover_queue_monitor.rs b/core/node/house_keeper/src/prover/queue_reporter/fri_prover_queue_reporter.rs similarity index 88% rename from core/node/house_keeper/src/fri_prover_queue_monitor.rs rename to core/node/house_keeper/src/prover/queue_reporter/fri_prover_queue_reporter.rs index 8b76d88d2ba5..1b4ea5de6781 100644 --- a/core/node/house_keeper/src/fri_prover_queue_monitor.rs +++ b/core/node/house_keeper/src/prover/queue_reporter/fri_prover_queue_reporter.rs @@ -2,18 +2,21 @@ use async_trait::async_trait; use prover_dal::{Prover, ProverDal}; use zksync_config::configs::fri_prover_group::FriProverGroupConfig; use zksync_dal::{ConnectionPool, Core, CoreDal}; +use zksync_types::ProtocolVersionId; -use crate::{metrics::FRI_PROVER_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::FRI_PROVER_METRICS}; +/// `FriProverQueueReporter` is a task that periodically reports prover jobs status. +/// Note: these values will be used for auto-scaling provers and Witness Vector Generators. #[derive(Debug)] -pub struct FriProverStatsReporter { +pub struct FriProverQueueReporter { reporting_interval_ms: u64, prover_connection_pool: ConnectionPool, db_connection_pool: ConnectionPool, config: FriProverGroupConfig, } -impl FriProverStatsReporter { +impl FriProverQueueReporter { pub fn new( reporting_interval_ms: u64, prover_connection_pool: ConnectionPool, @@ -29,10 +32,9 @@ impl FriProverStatsReporter { } } -/// Invoked periodically to push prover queued/in-progress job statistics #[async_trait] -impl PeriodicJob for FriProverStatsReporter { - const SERVICE_NAME: &'static str = "FriProverStatsReporter"; +impl PeriodicJob for FriProverQueueReporter { + const SERVICE_NAME: &'static str = "FriProverQueueReporter"; async fn run_routine_task(&mut self) -> anyhow::Result<()> { let mut conn = self.prover_connection_pool.connection().await.unwrap(); @@ -62,13 +64,16 @@ impl PeriodicJob for FriProverStatsReporter { circuit_id, aggregation_round, group_id, + ProtocolVersionId::current_prover_version(), stats.queued as u64, ); + FRI_PROVER_METRICS.report_prover_jobs( "in_progress", circuit_id, aggregation_round, group_id, + ProtocolVersionId::current_prover_version(), stats.in_progress as u64, ); } diff --git a/core/node/house_keeper/src/fri_witness_generator_queue_monitor.rs b/core/node/house_keeper/src/prover/queue_reporter/fri_witness_generator_queue_reporter.rs similarity index 66% rename from core/node/house_keeper/src/fri_witness_generator_queue_monitor.rs rename to core/node/house_keeper/src/prover/queue_reporter/fri_witness_generator_queue_reporter.rs index f8beb88e20e1..5f251a7136eb 100644 --- a/core/node/house_keeper/src/fri_witness_generator_queue_monitor.rs +++ b/core/node/house_keeper/src/prover/queue_reporter/fri_witness_generator_queue_reporter.rs @@ -3,17 +3,21 @@ use std::collections::HashMap; use async_trait::async_trait; use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use zksync_types::{basic_fri_types::AggregationRound, prover_dal::JobCountStatistics}; +use zksync_types::{ + basic_fri_types::AggregationRound, prover_dal::JobCountStatistics, ProtocolVersionId, +}; -use crate::{metrics::SERVER_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::SERVER_METRICS}; +/// `FriWitnessGeneratorQueueReporter` is a task that periodically reports witness generator jobs status. +/// Note: these values will be used for auto-scaling witness generators (Basic, Leaf, Node, Recursion Tip and Scheduler). #[derive(Debug)] -pub struct FriWitnessGeneratorStatsReporter { +pub struct FriWitnessGeneratorQueueReporter { reporting_interval_ms: u64, pool: ConnectionPool, } -impl FriWitnessGeneratorStatsReporter { +impl FriWitnessGeneratorQueueReporter { pub fn new(pool: ConnectionPool, reporting_interval_ms: u64) -> Self { Self { reporting_interval_ms, @@ -68,17 +72,23 @@ fn emit_metrics_for_round(round: AggregationRound, stats: JobCountStatistics) { ); } - SERVER_METRICS.witness_generator_jobs_by_round[&("queued", format!("{:?}", round))] - .set(stats.queued as u64); - SERVER_METRICS.witness_generator_jobs_by_round[&("in_progress", format!("{:?}", round))] + SERVER_METRICS.witness_generator_jobs_by_round[&( + "queued", + format!("{:?}", round), + ProtocolVersionId::current_prover_version().to_string(), + )] .set(stats.queued as u64); + SERVER_METRICS.witness_generator_jobs_by_round[&( + "in_progress", + format!("{:?}", round), + ProtocolVersionId::current_prover_version().to_string(), + )] + .set(stats.in_progress as u64); } -/// Invoked periodically to push job statistics to Prometheus -/// Note: these values will be used for auto-scaling job processors #[async_trait] -impl PeriodicJob for FriWitnessGeneratorStatsReporter { - const SERVICE_NAME: &'static str = "WitnessGeneratorStatsReporter"; +impl PeriodicJob for FriWitnessGeneratorQueueReporter { + const SERVICE_NAME: &'static str = "FriWitnessGeneratorQueueReporter"; async fn run_routine_task(&mut self) -> anyhow::Result<()> { let stats_for_all_rounds = self.get_job_statistics().await; @@ -96,8 +106,17 @@ impl PeriodicJob for FriWitnessGeneratorStatsReporter { ); } - SERVER_METRICS.witness_generator_jobs[&("queued")].set(aggregated.queued as u64); - SERVER_METRICS.witness_generator_jobs[&("in_progress")].set(aggregated.in_progress as u64); + SERVER_METRICS.witness_generator_jobs[&( + "queued", + ProtocolVersionId::current_prover_version().to_string(), + )] + .set(aggregated.queued as u64); + + SERVER_METRICS.witness_generator_jobs[&( + "in_progress", + ProtocolVersionId::current_prover_version().to_string(), + )] + .set(aggregated.in_progress as u64); Ok(()) } diff --git a/core/node/house_keeper/src/prover/queue_reporter/mod.rs b/core/node/house_keeper/src/prover/queue_reporter/mod.rs new file mode 100644 index 000000000000..9eba45320988 --- /dev/null +++ b/core/node/house_keeper/src/prover/queue_reporter/mod.rs @@ -0,0 +1,7 @@ +mod fri_proof_compressor_queue_reporter; +mod fri_prover_queue_reporter; +mod fri_witness_generator_queue_reporter; + +pub use fri_proof_compressor_queue_reporter::FriProofCompressorQueueReporter; +pub use fri_prover_queue_reporter::FriProverQueueReporter; +pub use fri_witness_generator_queue_reporter::FriWitnessGeneratorQueueReporter; diff --git a/core/node/house_keeper/src/fri_proof_compressor_job_retry_manager.rs b/core/node/house_keeper/src/prover/retry_manager/fri_proof_compressor_job_retry_manager.rs similarity index 89% rename from core/node/house_keeper/src/fri_proof_compressor_job_retry_manager.rs rename to core/node/house_keeper/src/prover/retry_manager/fri_proof_compressor_job_retry_manager.rs index 7dfb21090f71..4a27993249f0 100644 --- a/core/node/house_keeper/src/fri_proof_compressor_job_retry_manager.rs +++ b/core/node/house_keeper/src/prover/retry_manager/fri_proof_compressor_job_retry_manager.rs @@ -4,8 +4,9 @@ use async_trait::async_trait; use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use crate::{metrics::PROVER_FRI_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::PROVER_FRI_METRICS}; +/// `FriProofCompressorJobRetryManager` is a task that periodically queues stuck compressor jobs. #[derive(Debug)] pub struct FriProofCompressorJobRetryManager { pool: ConnectionPool, @@ -30,7 +31,6 @@ impl FriProofCompressorJobRetryManager { } } -/// Invoked periodically to re-queue stuck fri prover jobs. #[async_trait] impl PeriodicJob for FriProofCompressorJobRetryManager { const SERVICE_NAME: &'static str = "FriProofCompressorJobRetryManager"; diff --git a/core/node/house_keeper/src/fri_prover_job_retry_manager.rs b/core/node/house_keeper/src/prover/retry_manager/fri_prover_job_retry_manager.rs similarity index 90% rename from core/node/house_keeper/src/fri_prover_job_retry_manager.rs rename to core/node/house_keeper/src/prover/retry_manager/fri_prover_job_retry_manager.rs index 042af2f45e03..f059703a13c5 100644 --- a/core/node/house_keeper/src/fri_prover_job_retry_manager.rs +++ b/core/node/house_keeper/src/prover/retry_manager/fri_prover_job_retry_manager.rs @@ -4,8 +4,9 @@ use async_trait::async_trait; use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use crate::{metrics::SERVER_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::SERVER_METRICS}; +/// `FriProverJobRetryManager` is a task that periodically queues stuck prover jobs. #[derive(Debug)] pub struct FriProverJobRetryManager { pool: ConnectionPool, @@ -30,7 +31,6 @@ impl FriProverJobRetryManager { } } -/// Invoked periodically to re-queue stuck fri prover jobs. #[async_trait] impl PeriodicJob for FriProverJobRetryManager { const SERVICE_NAME: &'static str = "FriProverJobRetryManager"; diff --git a/core/node/house_keeper/src/fri_witness_generator_jobs_retry_manager.rs b/core/node/house_keeper/src/prover/retry_manager/fri_witness_generator_jobs_retry_manager.rs similarity index 96% rename from core/node/house_keeper/src/fri_witness_generator_jobs_retry_manager.rs rename to core/node/house_keeper/src/prover/retry_manager/fri_witness_generator_jobs_retry_manager.rs index 8c24b0980ec0..5b418fe64389 100644 --- a/core/node/house_keeper/src/fri_witness_generator_jobs_retry_manager.rs +++ b/core/node/house_keeper/src/prover/retry_manager/fri_witness_generator_jobs_retry_manager.rs @@ -5,10 +5,11 @@ use zksync_dal::ConnectionPool; use zksync_types::prover_dal::StuckJobs; use crate::{ - metrics::{WitnessType, SERVER_METRICS}, periodic_job::PeriodicJob, + prover::metrics::{WitnessType, SERVER_METRICS}, }; +/// `FriWitnessGeneratorJobRetryManager` is a task that periodically queues stuck prover jobs. #[derive(Debug)] pub struct FriWitnessGeneratorJobRetryManager { pool: ConnectionPool, @@ -110,7 +111,6 @@ impl FriWitnessGeneratorJobRetryManager { } } -/// Invoked periodically to re-queue stuck fri witness generator jobs. #[async_trait] impl PeriodicJob for FriWitnessGeneratorJobRetryManager { const SERVICE_NAME: &'static str = "FriWitnessGeneratorJobRetryManager"; diff --git a/core/node/house_keeper/src/prover/retry_manager/mod.rs b/core/node/house_keeper/src/prover/retry_manager/mod.rs new file mode 100644 index 000000000000..3b4a8b584817 --- /dev/null +++ b/core/node/house_keeper/src/prover/retry_manager/mod.rs @@ -0,0 +1,7 @@ +mod fri_proof_compressor_job_retry_manager; +mod fri_prover_job_retry_manager; +mod fri_witness_generator_jobs_retry_manager; + +pub use fri_proof_compressor_job_retry_manager::FriProofCompressorJobRetryManager; +pub use fri_prover_job_retry_manager::FriProverJobRetryManager; +pub use fri_witness_generator_jobs_retry_manager::FriWitnessGeneratorJobRetryManager; diff --git a/core/node/house_keeper/src/waiting_to_queued_fri_witness_job_mover.rs b/core/node/house_keeper/src/prover/waiting_to_queued_fri_witness_job_mover.rs similarity index 98% rename from core/node/house_keeper/src/waiting_to_queued_fri_witness_job_mover.rs rename to core/node/house_keeper/src/prover/waiting_to_queued_fri_witness_job_mover.rs index 0d4030d94083..bf4e31eee69d 100644 --- a/core/node/house_keeper/src/waiting_to_queued_fri_witness_job_mover.rs +++ b/core/node/house_keeper/src/prover/waiting_to_queued_fri_witness_job_mover.rs @@ -2,7 +2,7 @@ use async_trait::async_trait; use prover_dal::{Prover, ProverDal}; use zksync_dal::ConnectionPool; -use crate::{metrics::SERVER_METRICS, periodic_job::PeriodicJob}; +use crate::{periodic_job::PeriodicJob, prover::metrics::SERVER_METRICS}; #[derive(Debug)] pub struct WaitingToQueuedFriWitnessJobMover { diff --git a/core/node/metadata_calculator/Cargo.toml b/core/node/metadata_calculator/Cargo.toml index 3dcfcd89c211..5f336bb11d4f 100644 --- a/core/node/metadata_calculator/Cargo.toml +++ b/core/node/metadata_calculator/Cargo.toml @@ -29,6 +29,7 @@ thiserror.workspace = true tracing.workspace = true once_cell.workspace = true futures.workspace = true +itertools.workspace = true # dependencies for the tree API server reqwest.workspace = true diff --git a/core/node/metadata_calculator/src/helpers.rs b/core/node/metadata_calculator/src/helpers.rs index 52cb18ea4458..d3f2b43c42bf 100644 --- a/core/node/metadata_calculator/src/helpers.rs +++ b/core/node/metadata_calculator/src/helpers.rs @@ -10,6 +10,7 @@ use std::{ use anyhow::Context as _; use async_trait::async_trait; +use itertools::Itertools; use once_cell::sync::OnceCell; use serde::{Deserialize, Serialize}; #[cfg(test)] @@ -25,7 +26,9 @@ use zksync_merkle_tree::{ TreeEntryWithProof, TreeInstruction, }; use zksync_storage::{RocksDB, RocksDBOptions, StalledWritesRetries, WeakRocksDB}; -use zksync_types::{block::L1BatchHeader, L1BatchNumber, StorageKey, H256}; +use zksync_types::{ + block::L1BatchHeader, writes::TreeWrite, AccountTreeId, L1BatchNumber, StorageKey, H256, +}; use super::{ metrics::{LoadChangesStage, TreeUpdateStage, METRICS}, @@ -297,12 +300,35 @@ impl AsyncTreeReader { } pub async fn info(self) -> MerkleTreeInfo { - tokio::task::spawn_blocking(move || MerkleTreeInfo { - mode: self.mode, - root_hash: self.inner.root_hash(), - next_l1_batch_number: self.inner.next_l1_batch_number(), - min_l1_batch_number: self.inner.min_l1_batch_number(), - leaf_count: self.inner.leaf_count(), + tokio::task::spawn_blocking(move || { + loop { + let next_l1_batch_number = self.inner.next_l1_batch_number(); + let latest_l1_batch_number = next_l1_batch_number.checked_sub(1); + let root_info = if let Some(number) = latest_l1_batch_number { + self.inner.root_info(L1BatchNumber(number)) + } else { + // No L1 batches in the tree yet. + Some((ZkSyncTree::empty_tree_hash(), 0)) + }; + let Some((root_hash, leaf_count)) = root_info else { + // It is possible (although very unlikely) that the latest tree version was removed after requesting it, + // hence the outer loop; RocksDB doesn't provide consistent data views by default. + tracing::info!( + "Tree version at L1 batch {latest_l1_batch_number:?} was removed after requesting the latest tree L1 batch; \ + re-requesting tree information" + ); + continue; + }; + + // `min_l1_batch_number` is not necessarily consistent with other retrieved tree data, but this looks fine. + break MerkleTreeInfo { + mode: self.mode, + root_hash, + next_l1_batch_number, + min_l1_batch_number: self.inner.min_l1_batch_number(), + leaf_count, + }; + } }) .await .unwrap() @@ -392,6 +418,39 @@ impl AsyncTreeRecovery { .recovered_version() } + pub async fn ensure_desired_chunk_size( + &mut self, + desired_chunk_size: u64, + ) -> anyhow::Result<()> { + const CHUNK_SIZE_KEY: &str = "recovery.desired_chunk_size"; + + let mut tree = self.inner.take().expect(Self::INCONSISTENT_MSG); + let tree = tokio::task::spawn_blocking(move || { + // **Important.** Tags should not be mutated on error (i.e., it would be an error to unconditionally call `tags.insert()` + // and then check the previous value). + tree.update_custom_tags(|tags| { + if let Some(chunk_size_in_tree) = tags.get(CHUNK_SIZE_KEY) { + let chunk_size_in_tree: u64 = chunk_size_in_tree + .parse() + .with_context(|| format!("error parsing desired_chunk_size `{chunk_size_in_tree}` in Merkle tree tags"))?; + anyhow::ensure!( + chunk_size_in_tree == desired_chunk_size, + "Mismatch between the configured desired chunk size ({desired_chunk_size}) and one that was used previously ({chunk_size_in_tree}). \ + Either change the desired chunk size in configuration, or reset Merkle tree recovery by clearing its RocksDB directory" + ); + } else { + tags.insert(CHUNK_SIZE_KEY.to_owned(), desired_chunk_size.to_string()); + } + Ok(()) + })?; + anyhow::Ok(tree) + }) + .await??; + + self.inner = Some(tree); + Ok(()) + } + /// Returns an entry for the specified keys. pub async fn entries(&mut self, keys: Vec) -> Vec { let tree = self.inner.take().expect(Self::INCONSISTENT_MSG); @@ -550,8 +609,83 @@ impl L1BatchWithLogs { MerkleTreeMode::Lightweight => HashSet::new(), }; + let load_tree_writes_latency = METRICS.start_load_stage(LoadChangesStage::LoadTreeWrites); + let mut tree_writes = storage + .blocks_dal() + .get_tree_writes(l1_batch_number) + .await?; + if tree_writes.is_none() && l1_batch_number.0 > 0 { + // If `tree_writes` are present for the previous L1 batch, then it is expected them to be eventually present for the current batch as well. + // Waiting for tree writes should be faster then constructing them, so we wait with a reasonable timeout. + let tree_writes_present_for_previous_batch = storage + .blocks_dal() + .check_tree_writes_presence(l1_batch_number - 1) + .await?; + if tree_writes_present_for_previous_batch { + tree_writes = Self::wait_for_tree_writes(storage, l1_batch_number).await?; + } + } + load_tree_writes_latency.observe(); + + let storage_logs = if let Some(tree_writes) = tree_writes { + // If tree writes are present in DB then simply use them. + let writes = tree_writes.into_iter().map(|tree_write| { + let storage_key = + StorageKey::new(AccountTreeId::new(tree_write.address), tree_write.key); + TreeInstruction::write(storage_key, tree_write.leaf_index, tree_write.value) + }); + let reads = protective_reads.into_iter().map(TreeInstruction::Read); + + // `writes` and `reads` are already sorted, we only need to merge them. + writes + .merge_by(reads, |a, b| a.key() <= b.key()) + .collect::>() + } else { + // Otherwise, load writes' data from other tables. + Self::extract_storage_logs_from_db(storage, l1_batch_number, protective_reads).await? + }; + + load_changes_latency.observe(); + + Ok(Some(Self { + header, + storage_logs, + mode, + })) + } + + #[allow(clippy::needless_pass_by_ref_mut)] // false positive + async fn wait_for_tree_writes( + connection: &mut Connection<'_, Core>, + l1_batch_number: L1BatchNumber, + ) -> anyhow::Result>> { + const INTERVAL: Duration = Duration::from_millis(50); + const TIMEOUT: Duration = Duration::from_secs(5); + + tokio::time::timeout(TIMEOUT, async { + loop { + if let Some(tree_writes) = connection + .blocks_dal() + .get_tree_writes(l1_batch_number) + .await? + { + break anyhow::Ok(tree_writes); + } + tokio::time::sleep(INTERVAL).await; + } + }) + .await + .ok() + .transpose() + } + + async fn extract_storage_logs_from_db( + connection: &mut Connection<'_, Core>, + l1_batch_number: L1BatchNumber, + protective_reads: HashSet, + ) -> anyhow::Result>> { let touched_slots_latency = METRICS.start_load_stage(LoadChangesStage::LoadTouchedSlots); - let mut touched_slots = storage + let mut touched_slots = connection .storage_logs_dal() .get_touched_slots_for_l1_batch(l1_batch_number) .await @@ -561,7 +695,7 @@ impl L1BatchWithLogs { let leaf_indices_latency = METRICS.start_load_stage(LoadChangesStage::LoadLeafIndices); let hashed_keys_for_writes: Vec<_> = touched_slots.keys().map(StorageKey::hashed_key).collect(); - let l1_batches_for_initial_writes = storage + let l1_batches_for_initial_writes = connection .storage_logs_dal() .get_l1_batches_and_indices_for_initial_writes(&hashed_keys_for_writes) .await @@ -598,12 +732,7 @@ impl L1BatchWithLogs { } } - load_changes_latency.observe(); - Ok(Some(Self { - header, - storage_logs: storage_logs.into_values().collect(), - mode, - })) + Ok(storage_logs.into_values().collect()) } } @@ -613,7 +742,7 @@ mod tests { use zksync_dal::{ConnectionPool, Core}; use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; use zksync_prover_interface::inputs::PrepareBasicCircuitsJob; - use zksync_types::{StorageKey, StorageLog}; + use zksync_types::{writes::TreeWrite, StorageKey, StorageLog}; use super::*; use crate::tests::{extend_db_state, gen_storage_logs, mock_config, reset_db_state}; @@ -702,6 +831,9 @@ mod tests { reset_db_state(&pool, 5).await; let mut storage = pool.connection().await.unwrap(); + let mut tree_writes = Vec::new(); + + // Check equivalence in case `tree_writes` are not present in DB. for l1_batch_number in 0..=5 { let l1_batch_number = L1BatchNumber(l1_batch_number); let batch_with_logs = @@ -713,6 +845,44 @@ mod tests { .await .unwrap(); assert_eq!(batch_with_logs, slow_batch_with_logs); + + let writes = batch_with_logs + .storage_logs + .into_iter() + .filter_map(|instruction| match instruction { + TreeInstruction::Write(tree_entry) => Some(TreeWrite { + address: *tree_entry.key.address(), + key: *tree_entry.key.key(), + value: tree_entry.value, + leaf_index: tree_entry.leaf_index, + }), + _ => None, + }) + .collect::>(); + tree_writes.push(writes); + } + + // Insert `tree_writes` and check again. + for l1_batch_number in 0..5 { + let l1_batch_number = L1BatchNumber(l1_batch_number); + storage + .blocks_dal() + .set_tree_writes( + l1_batch_number, + tree_writes[l1_batch_number.0 as usize].clone(), + ) + .await + .unwrap(); + + let batch_with_logs = + L1BatchWithLogs::new(&mut storage, l1_batch_number, MerkleTreeMode::Full) + .await + .unwrap() + .expect("no L1 batch"); + let slow_batch_with_logs = L1BatchWithLogs::slow(&mut storage, l1_batch_number) + .await + .unwrap(); + assert_eq!(batch_with_logs, slow_batch_with_logs); } } diff --git a/core/node/metadata_calculator/src/lib.rs b/core/node/metadata_calculator/src/lib.rs index 9f3b0a113a72..50c13ba19644 100644 --- a/core/node/metadata_calculator/src/lib.rs +++ b/core/node/metadata_calculator/src/lib.rs @@ -37,6 +37,24 @@ mod recovery; pub(crate) mod tests; mod updater; +#[derive(Debug, Clone)] +pub struct MetadataCalculatorRecoveryConfig { + /// Approximate chunk size (measured in the number of entries) to recover on a single iteration. + /// Reasonable values are order of 100,000 (meaning an iteration takes several seconds). + /// + /// **Important.** This value cannot be changed in the middle of tree recovery (i.e., if a node is stopped in the middle + /// of recovery and then restarted with a different config). + pub desired_chunk_size: u64, +} + +impl Default for MetadataCalculatorRecoveryConfig { + fn default() -> Self { + Self { + desired_chunk_size: 200_000, + } + } +} + /// Configuration of [`MetadataCalculator`]. #[derive(Debug, Clone)] pub struct MetadataCalculatorConfig { @@ -65,6 +83,8 @@ pub struct MetadataCalculatorConfig { pub memtable_capacity: usize, /// Timeout to wait for the Merkle tree database to run compaction on stalled writes. pub stalled_writes_timeout: Duration, + /// Configuration specific to the Merkle tree recovery. + pub recovery: MetadataCalculatorRecoveryConfig, } impl MetadataCalculatorConfig { @@ -83,6 +103,8 @@ impl MetadataCalculatorConfig { include_indices_and_filters_in_block_cache: false, memtable_capacity: merkle_tree_config.memtable_capacity(), stalled_writes_timeout: merkle_tree_config.stalled_writes_timeout(), + // The main node isn't supposed to be recovered yet, so this value doesn't matter much + recovery: MetadataCalculatorRecoveryConfig::default(), } } } @@ -193,10 +215,11 @@ impl MetadataCalculator { let tree = self.create_tree().await?; let tree = tree .ensure_ready( + &self.config.recovery, &self.pool, self.recovery_pool, - &stop_receiver, &self.health_updater, + &stop_receiver, ) .await?; let Some(mut tree) = tree else { diff --git a/core/node/metadata_calculator/src/metrics.rs b/core/node/metadata_calculator/src/metrics.rs index 074f444dea68..7eb49b95afd4 100644 --- a/core/node/metadata_calculator/src/metrics.rs +++ b/core/node/metadata_calculator/src/metrics.rs @@ -76,6 +76,7 @@ pub(super) enum LoadChangesStage { LoadProtectiveReads, LoadTouchedSlots, LoadLeafIndices, + LoadTreeWrites, } /// Latency metric for a certain stage of the tree update. diff --git a/core/node/metadata_calculator/src/recovery/mod.rs b/core/node/metadata_calculator/src/recovery/mod.rs index 7e621531dc86..94eb397858d5 100644 --- a/core/node/metadata_calculator/src/recovery/mod.rs +++ b/core/node/metadata_calculator/src/recovery/mod.rs @@ -32,7 +32,6 @@ use std::{ }; use anyhow::Context as _; -use async_trait::async_trait; use futures::future; use tokio::sync::{watch, Mutex, Semaphore}; use zksync_dal::{Connection, ConnectionPool, Core, CoreDal}; @@ -47,6 +46,7 @@ use zksync_types::{ use super::{ helpers::{AsyncTree, AsyncTreeRecovery, GenericAsyncTree, MerkleTreeHealth}, metrics::{ChunkRecoveryStage, RecoveryStage, RECOVERY_METRICS}, + MetadataCalculatorRecoveryConfig, }; #[cfg(test)] @@ -54,17 +54,12 @@ mod tests; /// Handler of recovery life cycle events. This functionality is encapsulated in a trait to be able /// to control recovery behavior in tests. -#[async_trait] trait HandleRecoveryEvent: fmt::Debug + Send + Sync { fn recovery_started(&mut self, _chunk_count: u64, _recovered_chunk_count: u64) { // Default implementation does nothing } - async fn chunk_started(&self) { - // Default implementation does nothing - } - - async fn chunk_recovered(&self) { + fn chunk_recovered(&self) { // Default implementation does nothing } } @@ -87,7 +82,6 @@ impl<'a> RecoveryHealthUpdater<'a> { } } -#[async_trait] impl HandleRecoveryEvent for RecoveryHealthUpdater<'_> { fn recovery_started(&mut self, chunk_count: u64, recovered_chunk_count: u64) { self.chunk_count = chunk_count; @@ -97,8 +91,13 @@ impl HandleRecoveryEvent for RecoveryHealthUpdater<'_> { .set(recovered_chunk_count); } - async fn chunk_recovered(&self) { + fn chunk_recovered(&self) { let recovered_chunk_count = self.recovered_chunk_count.fetch_add(1, Ordering::SeqCst) + 1; + let chunks_left = self.chunk_count.saturating_sub(recovered_chunk_count); + tracing::info!( + "Recovered {recovered_chunk_count}/{} Merkle tree chunks, there are {chunks_left} left to process", + self.chunk_count + ); RECOVERY_METRICS .recovered_chunk_count .set(recovered_chunk_count); @@ -115,21 +114,19 @@ struct SnapshotParameters { l2_block: L2BlockNumber, expected_root_hash: H256, log_count: u64, + desired_chunk_size: u64, } impl SnapshotParameters { - /// This is intentionally not configurable because chunks must be the same for the entire recovery - /// (i.e., not changed after a node restart). - const DESIRED_CHUNK_SIZE: u64 = 200_000; - async fn new( pool: &ConnectionPool, recovery: &SnapshotRecoveryStatus, + config: &MetadataCalculatorRecoveryConfig, ) -> anyhow::Result { let l2_block = recovery.l2_block_number; let expected_root_hash = recovery.l1_batch_root_hash; - let mut storage = pool.connection().await?; + let mut storage = pool.connection_tagged("metadata_calculator").await?; let log_count = storage .storage_logs_dal() .get_storage_logs_row_count(l2_block) @@ -139,11 +136,12 @@ impl SnapshotParameters { l2_block, expected_root_hash, log_count, + desired_chunk_size: config.desired_chunk_size, }) } fn chunk_count(&self) -> u64 { - self.log_count.div_ceil(Self::DESIRED_CHUNK_SIZE) + self.log_count.div_ceil(self.desired_chunk_size) } } @@ -163,10 +161,11 @@ impl GenericAsyncTree { /// with other components). pub async fn ensure_ready( self, + config: &MetadataCalculatorRecoveryConfig, main_pool: &ConnectionPool, recovery_pool: ConnectionPool, - stop_receiver: &watch::Receiver, health_updater: &HealthUpdater, + stop_receiver: &watch::Receiver, ) -> anyhow::Result> { let started_at = Instant::now(); let (tree, snapshot_recovery) = match self { @@ -199,8 +198,10 @@ impl GenericAsyncTree { } }; - let snapshot = SnapshotParameters::new(main_pool, &snapshot_recovery).await?; - tracing::debug!("Obtained snapshot parameters: {snapshot:?}"); + let snapshot = SnapshotParameters::new(main_pool, &snapshot_recovery, config).await?; + tracing::debug!( + "Obtained snapshot parameters: {snapshot:?} based on recovery configuration {config:?}" + ); let recovery_options = RecoveryOptions { chunk_count: snapshot.chunk_count(), concurrency_limit: recovery_pool.max_size() as usize, @@ -227,6 +228,9 @@ impl AsyncTreeRecovery { pool: &ConnectionPool, stop_receiver: &watch::Receiver, ) -> anyhow::Result> { + self.ensure_desired_chunk_size(snapshot.desired_chunk_size) + .await?; + let start_time = Instant::now(); let chunk_count = options.chunk_count; let chunks: Vec<_> = (0..chunk_count) @@ -237,7 +241,7 @@ impl AsyncTreeRecovery { options.concurrency_limit ); - let mut storage = pool.connection().await?; + let mut storage = pool.connection_tagged("metadata_calculator").await?; let remaining_chunks = self .filter_chunks(&mut storage, snapshot.l2_block, &chunks) .await?; @@ -257,9 +261,8 @@ impl AsyncTreeRecovery { .acquire() .await .context("semaphore is never closed")?; - options.events.chunk_started().await; Self::recover_key_chunk(&tree, snapshot.l2_block, chunk, pool, stop_receiver).await?; - options.events.chunk_recovered().await; + options.events.chunk_recovered(); anyhow::Ok(()) }); future::try_join_all(chunk_tasks).await?; @@ -339,7 +342,7 @@ impl AsyncTreeRecovery { ) -> anyhow::Result<()> { let acquire_connection_latency = RECOVERY_METRICS.chunk_latency[&ChunkRecoveryStage::AcquireConnection].start(); - let mut storage = pool.connection().await?; + let mut storage = pool.connection_tagged("metadata_calculator").await?; acquire_connection_latency.observe(); if *stop_receiver.borrow() { diff --git a/core/node/metadata_calculator/src/recovery/tests.rs b/core/node/metadata_calculator/src/recovery/tests.rs index 3e2978cd8ccf..2e27eddec6cf 100644 --- a/core/node/metadata_calculator/src/recovery/tests.rs +++ b/core/node/metadata_calculator/src/recovery/tests.rs @@ -33,6 +33,7 @@ fn calculating_chunk_count() { l2_block: L2BlockNumber(1), log_count: 160_000_000, expected_root_hash: H256::zero(), + desired_chunk_size: 200_000, }; assert_eq!(snapshot.chunk_count(), 800); @@ -53,7 +54,8 @@ async fn basic_recovery_workflow() { let pool = ConnectionPool::::test_pool().await; let temp_dir = TempDir::new().expect("failed get temporary directory for RocksDB"); let snapshot_recovery = prepare_recovery_snapshot_with_genesis(pool.clone(), &temp_dir).await; - let snapshot = SnapshotParameters::new(&pool, &snapshot_recovery) + let config = MetadataCalculatorRecoveryConfig::default(); + let snapshot = SnapshotParameters::new(&pool, &snapshot_recovery, &config) .await .unwrap(); @@ -146,13 +148,12 @@ impl TestEventListener { } } -#[async_trait] impl HandleRecoveryEvent for TestEventListener { fn recovery_started(&mut self, _chunk_count: u64, recovered_chunk_count: u64) { assert_eq!(recovered_chunk_count, self.expected_recovered_chunks); } - async fn chunk_recovered(&self) { + fn chunk_recovered(&self) { let processed_chunk_count = self.processed_chunk_count.fetch_add(1, Ordering::SeqCst) + 1; if processed_chunk_count >= self.stop_threshold { self.stop_sender.send_replace(true); @@ -160,6 +161,47 @@ impl HandleRecoveryEvent for TestEventListener { } } +#[tokio::test] +async fn recovery_detects_incorrect_chunk_size_change() { + let pool = ConnectionPool::::test_pool().await; + let temp_dir = TempDir::new().expect("failed get temporary directory for RocksDB"); + let snapshot_recovery = prepare_recovery_snapshot_with_genesis(pool.clone(), &temp_dir).await; + + let tree_path = temp_dir.path().join("recovery"); + let tree = create_tree_recovery(&tree_path, L1BatchNumber(1)).await; + let (stop_sender, stop_receiver) = watch::channel(false); + let recovery_options = RecoveryOptions { + chunk_count: 5, + concurrency_limit: 1, + events: Box::new(TestEventListener::new(1, stop_sender)), + }; + let config = MetadataCalculatorRecoveryConfig::default(); + let mut snapshot = SnapshotParameters::new(&pool, &snapshot_recovery, &config) + .await + .unwrap(); + assert!(tree + .recover(snapshot, recovery_options, &pool, &stop_receiver) + .await + .unwrap() + .is_none()); + + let tree = create_tree_recovery(&tree_path, L1BatchNumber(1)).await; + let health_updater = ReactiveHealthCheck::new("tree").1; + let recovery_options = RecoveryOptions { + chunk_count: 5, + concurrency_limit: 1, + events: Box::new(RecoveryHealthUpdater::new(&health_updater)), + }; + snapshot.desired_chunk_size /= 2; + + let err = tree + .recover(snapshot, recovery_options, &pool, &stop_receiver) + .await + .unwrap_err() + .to_string(); + assert!(err.contains("desired chunk size"), "{err}"); +} + #[test_casing(3, [5, 7, 8])] #[tokio::test] async fn recovery_fault_tolerance(chunk_count: u64) { @@ -175,7 +217,8 @@ async fn recovery_fault_tolerance(chunk_count: u64) { concurrency_limit: 1, events: Box::new(TestEventListener::new(1, stop_sender)), }; - let snapshot = SnapshotParameters::new(&pool, &snapshot_recovery) + let config = MetadataCalculatorRecoveryConfig::default(); + let snapshot = SnapshotParameters::new(&pool, &snapshot_recovery, &config) .await .unwrap(); assert!(tree diff --git a/core/node/metadata_calculator/src/tests.rs b/core/node/metadata_calculator/src/tests.rs index 00522f27896b..1a1b4eb98298 100644 --- a/core/node/metadata_calculator/src/tests.rs +++ b/core/node/metadata_calculator/src/tests.rs @@ -26,6 +26,7 @@ use zksync_utils::u32_to_h256; use super::{ helpers::L1BatchWithLogs, GenericAsyncTree, MetadataCalculator, MetadataCalculatorConfig, + MetadataCalculatorRecoveryConfig, }; const RUN_TIMEOUT: Duration = Duration::from_secs(30); @@ -53,6 +54,7 @@ pub(super) fn mock_config(db_path: &Path) -> MetadataCalculatorConfig { include_indices_and_filters_in_block_cache: false, memtable_capacity: 16 << 20, // 16 MiB stalled_writes_timeout: Duration::ZERO, // writes should never be stalled in tests + recovery: MetadataCalculatorRecoveryConfig::default(), } } diff --git a/core/node/node_framework/Cargo.toml b/core/node/node_framework/Cargo.toml index 8d7afee3c7e2..f95500a3836d 100644 --- a/core/node/node_framework/Cargo.toml +++ b/core/node/node_framework/Cargo.toml @@ -20,7 +20,6 @@ zksync_config.workspace = true zksync_protobuf_config.workspace = true zksync_state.workspace = true zksync_object_store.workspace = true -zksync_core_leftovers.workspace = true zksync_storage.workspace = true zksync_eth_client.workspace = true zksync_contracts.workspace = true @@ -41,6 +40,8 @@ zksync_node_sync.workspace = true zksync_node_api_server.workspace = true zksync_node_consensus.workspace = true zksync_contract_verification_server.workspace = true +zksync_tee_verifier_input_producer.workspace = true +zksync_queued_job_processor.workspace = true tracing.workspace = true thiserror.workspace = true diff --git a/core/node/node_framework/examples/main_node.rs b/core/node/node_framework/examples/main_node.rs index b03ab15189fa..f42cf76d33a2 100644 --- a/core/node/node_framework/examples/main_node.rs +++ b/core/node/node_framework/examples/main_node.rs @@ -9,7 +9,6 @@ use zksync_config::{ CircuitBreakerConfig, MempoolConfig, NetworkConfig, OperationsManagerConfig, StateKeeperConfig, }, - consensus::{ConsensusConfig, ConsensusSecrets}, fri_prover_group::FriProverGroupConfig, house_keeper::HouseKeeperConfig, wallets::Wallets, @@ -19,7 +18,6 @@ use zksync_config::{ ApiConfig, ContractVerifierConfig, ContractsConfig, DBConfig, EthConfig, EthWatchConfig, GasAdjusterConfig, GenesisConfig, ObjectStoreConfig, PostgresConfig, }; -use zksync_core_leftovers::temp_config_store::decode_yaml_repr; use zksync_env_config::FromEnv; use zksync_metadata_calculator::MetadataCalculatorConfig; use zksync_node_api_server::{ @@ -30,9 +28,8 @@ use zksync_node_framework::{ implementations::layers::{ circuit_breaker_checker::CircuitBreakerCheckerLayer, commitment_generator::CommitmentGeneratorLayer, - consensus::{ConsensusLayer, Mode as ConsensusMode}, contract_verification_api::ContractVerificationApiLayer, - eth_sender::EthSenderLayer, + eth_sender::{EthTxAggregatorLayer, EthTxManagerLayer}, eth_watch::EthWatchLayer, healtcheck_server::HealthCheckLayer, house_keeper::HouseKeeperLayer, @@ -58,7 +55,6 @@ use zksync_node_framework::{ }, service::{ZkStackService, ZkStackServiceBuilder, ZkStackServiceError}, }; -use zksync_protobuf_config::proto; struct MainNodeBuilder { node: ZkStackServiceBuilder, @@ -150,15 +146,15 @@ impl MainNodeBuilder { fn add_state_keeper_layer(mut self) -> anyhow::Result { let wallets = Wallets::from_env()?; let mempool_io_layer = MempoolIOLayer::new( - NetworkConfig::from_env()?, + NetworkConfig::from_env()?.zksync_network_id, ContractsConfig::from_env()?, StateKeeperConfig::from_env()?, MempoolConfig::from_env()?, wallets.state_keeper.context("State keeper wallets")?, ); let main_node_batch_executor_builder_layer = - MainBatchExecutorLayer::new(DBConfig::from_env()?, StateKeeperConfig::from_env()?); - let state_keeper_layer = StateKeeperLayer; + MainBatchExecutorLayer::new(StateKeeperConfig::from_env()?); + let state_keeper_layer = StateKeeperLayer::new(DBConfig::from_env()?); self.node .add_layer(mempool_io_layer) .add_layer(main_node_batch_executor_builder_layer) @@ -305,12 +301,14 @@ impl MainNodeBuilder { let network_config = NetworkConfig::from_env()?; let genesis_config = GenesisConfig::from_env()?; - self.node.add_layer(EthSenderLayer::new( - eth_sender_config, + self.node.add_layer(EthTxAggregatorLayer::new( + eth_sender_config.clone(), contracts_config, - network_config, + network_config.zksync_network_id, genesis_config.l1_batch_commit_data_generator_mode, )); + self.node + .add_layer(EthTxManagerLayer::new(eth_sender_config)); Ok(self) } @@ -356,47 +354,6 @@ impl MainNodeBuilder { Ok(self) } - fn add_consensus_layer(mut self) -> anyhow::Result { - // Copy-pasted from the zksync_server codebase. - - fn read_consensus_secrets() -> anyhow::Result> { - // Read public config. - let Ok(path) = std::env::var("CONSENSUS_SECRETS_PATH") else { - return Ok(None); - }; - let secrets = std::fs::read_to_string(&path).context(path)?; - Ok(Some( - decode_yaml_repr::(&secrets) - .context("failed decoding YAML")? - .consensus - .context("No consensus in secrets")?, - )) - } - - fn read_consensus_config() -> anyhow::Result> { - // Read public config. - let Ok(path) = std::env::var("CONSENSUS_CONFIG_PATH") else { - return Ok(None); - }; - let cfg = std::fs::read_to_string(&path).context(path)?; - Ok(Some( - decode_yaml_repr::(&cfg) - .context("failed decoding YAML")?, - )) - } - - let config = read_consensus_config().context("read_consensus_config()")?; - let secrets = read_consensus_secrets().context("read_consensus_secrets()")?; - - self.node.add_layer(ConsensusLayer { - mode: ConsensusMode::Main, - config, - secrets, - }); - - Ok(self) - } - fn build(mut self) -> Result { self.node.build() } @@ -435,7 +392,6 @@ fn main() -> anyhow::Result<()> { .add_house_keeper_layer()? .add_commitment_generator_layer()? .add_contract_verification_api_layer()? - .add_consensus_layer()? .build()? .run()?; diff --git a/core/node/node_framework/src/implementations/layers/commitment_generator.rs b/core/node/node_framework/src/implementations/layers/commitment_generator.rs index 5d2f2d47678b..aeb668dca178 100644 --- a/core/node/node_framework/src/implementations/layers/commitment_generator.rs +++ b/core/node/node_framework/src/implementations/layers/commitment_generator.rs @@ -30,7 +30,8 @@ impl WiringLayer for CommitmentGeneratorLayer { async fn wire(self: Box, mut context: ServiceContext<'_>) -> Result<(), WiringError> { let pool_resource = context.get_resource::>().await?; - let main_pool = pool_resource.get().await?; + let pool_size = CommitmentGenerator::default_parallelism().get(); + let main_pool = pool_resource.get_custom(pool_size).await?; let commitment_generator = CommitmentGenerator::new(main_pool, self.mode); diff --git a/core/node/node_framework/src/implementations/layers/eth_sender.rs b/core/node/node_framework/src/implementations/layers/eth_sender.rs index 54419ec555d0..ed27fe863214 100644 --- a/core/node/node_framework/src/implementations/layers/eth_sender.rs +++ b/core/node/node_framework/src/implementations/layers/eth_sender.rs @@ -1,9 +1,9 @@ use anyhow::Context; use zksync_circuit_breaker::l1_txs::FailedL1TransactionChecker; -use zksync_config::configs::{chain::NetworkConfig, eth_sender::EthConfig, ContractsConfig}; +use zksync_config::configs::{eth_sender::EthConfig, ContractsConfig}; use zksync_eth_client::BoundEthInterface; use zksync_eth_sender::{Aggregator, EthTxAggregator, EthTxManager}; -use zksync_types::commitment::L1BatchCommitmentMode; +use zksync_types::{commitment::L1BatchCommitmentMode, L2ChainId}; use crate::{ implementations::resources::{ @@ -19,33 +19,93 @@ use crate::{ }; #[derive(Debug)] -pub struct EthSenderLayer { +pub struct EthTxManagerLayer { + eth_sender_config: EthConfig, +} + +impl EthTxManagerLayer { + pub fn new(eth_sender_config: EthConfig) -> Self { + Self { eth_sender_config } + } +} + +#[async_trait::async_trait] +impl WiringLayer for EthTxManagerLayer { + fn layer_name(&self) -> &'static str { + "eth_tx_manager_layer" + } + + async fn wire(self: Box, mut context: ServiceContext<'_>) -> Result<(), WiringError> { + // Get resources. + let master_pool_resource = context.get_resource::>().await?; + let master_pool = master_pool_resource.get().await.unwrap(); + let replica_pool_resource = context.get_resource::>().await?; + let replica_pool = replica_pool_resource.get().await.unwrap(); + + let eth_client = context.get_resource::().await?.0; + let eth_client_blobs = match context + .get_resource::() + .await + { + Ok(BoundEthInterfaceForBlobsResource(client)) => Some(client), + Err(WiringError::ResourceLacking { .. }) => None, + Err(err) => return Err(err), + }; + + let config = self.eth_sender_config.sender.context("sender")?; + + let gas_adjuster = context.get_resource::().await?.0; + + let eth_tx_manager_actor = EthTxManager::new( + master_pool, + config, + gas_adjuster, + eth_client, + eth_client_blobs, + ); + + context.add_task(Box::new(EthTxManagerTask { + eth_tx_manager_actor, + })); + + // Insert circuit breaker. + let CircuitBreakersResource { breakers } = context.get_resource_or_default().await; + breakers + .insert(Box::new(FailedL1TransactionChecker { pool: replica_pool })) + .await; + + Ok(()) + } +} + +#[derive(Debug)] +pub struct EthTxAggregatorLayer { eth_sender_config: EthConfig, contracts_config: ContractsConfig, - network_config: NetworkConfig, + zksync_network_id: L2ChainId, l1_batch_commit_data_generator_mode: L1BatchCommitmentMode, } -impl EthSenderLayer { +impl EthTxAggregatorLayer { pub fn new( eth_sender_config: EthConfig, contracts_config: ContractsConfig, - network_config: NetworkConfig, + zksync_network_id: L2ChainId, l1_batch_commit_data_generator_mode: L1BatchCommitmentMode, ) -> Self { Self { eth_sender_config, contracts_config, - network_config, + zksync_network_id, l1_batch_commit_data_generator_mode, } } } #[async_trait::async_trait] -impl WiringLayer for EthSenderLayer { +impl WiringLayer for EthTxAggregatorLayer { fn layer_name(&self) -> &'static str { - "eth_sender_layer" + "eth_tx_aggregator_layer" } async fn wire(self: Box, mut context: ServiceContext<'_>) -> Result<(), WiringError> { @@ -87,7 +147,7 @@ impl WiringLayer for EthSenderLayer { self.contracts_config.validator_timelock_addr, self.contracts_config.l1_multicall3_addr, self.contracts_config.diamond_proxy_addr, - self.network_config.zksync_network_id, + self.zksync_network_id, eth_client_blobs_addr, ) .await; @@ -96,20 +156,6 @@ impl WiringLayer for EthSenderLayer { eth_tx_aggregator_actor, })); - let gas_adjuster = context.get_resource::().await?.0; - - let eth_tx_manager_actor = EthTxManager::new( - master_pool, - config, - gas_adjuster, - eth_client, - eth_client_blobs, - ); - - context.add_task(Box::new(EthTxManagerTask { - eth_tx_manager_actor, - })); - // Insert circuit breaker. let CircuitBreakersResource { breakers } = context.get_resource_or_default().await; breakers diff --git a/core/node/node_framework/src/implementations/layers/house_keeper.rs b/core/node/node_framework/src/implementations/layers/house_keeper.rs index cf0e4954b320..1eb559ea5e1f 100644 --- a/core/node/node_framework/src/implementations/layers/house_keeper.rs +++ b/core/node/node_framework/src/implementations/layers/house_keeper.rs @@ -6,16 +6,14 @@ use zksync_config::configs::{ }; use zksync_dal::{metrics::PostgresMetrics, ConnectionPool, Core}; use zksync_house_keeper::{ - blocks_state_reporter::L1BatchMetricsReporter, fri_gpu_prover_archiver::FriGpuProverArchiver, - fri_proof_compressor_job_retry_manager::FriProofCompressorJobRetryManager, - fri_proof_compressor_queue_monitor::FriProofCompressorStatsReporter, - fri_prover_job_retry_manager::FriProverJobRetryManager, - fri_prover_jobs_archiver::FriProverJobArchiver, - fri_prover_queue_monitor::FriProverStatsReporter, - fri_witness_generator_jobs_retry_manager::FriWitnessGeneratorJobRetryManager, - fri_witness_generator_queue_monitor::FriWitnessGeneratorStatsReporter, + blocks_state_reporter::L1BatchMetricsReporter, periodic_job::PeriodicJob, - waiting_to_queued_fri_witness_job_mover::WaitingToQueuedFriWitnessJobMover, + prover::{ + FriGpuProverArchiver, FriProofCompressorJobRetryManager, FriProofCompressorQueueReporter, + FriProverJobRetryManager, FriProverJobsArchiver, FriProverQueueReporter, + FriWitnessGeneratorJobRetryManager, FriWitnessGeneratorQueueReporter, + WaitingToQueuedFriWitnessJobMover, + }, }; use crate::{ @@ -115,7 +113,7 @@ impl WiringLayer for HouseKeeperLayer { self.house_keeper_config.prover_job_archiver_params() { let fri_prover_job_archiver = - FriProverJobArchiver::new(prover_pool.clone(), archiving_interval, archive_after); + FriProverJobsArchiver::new(prover_pool.clone(), archiving_interval, archive_after); context.add_task(Box::new(FriProverJobArchiverTask { fri_prover_job_archiver, })); @@ -131,7 +129,7 @@ impl WiringLayer for HouseKeeperLayer { })); } - let fri_witness_generator_stats_reporter = FriWitnessGeneratorStatsReporter::new( + let fri_witness_generator_stats_reporter = FriWitnessGeneratorQueueReporter::new( prover_pool.clone(), self.house_keeper_config .witness_generator_stats_reporting_interval_ms, @@ -140,7 +138,7 @@ impl WiringLayer for HouseKeeperLayer { fri_witness_generator_stats_reporter, })); - let fri_prover_stats_reporter = FriProverStatsReporter::new( + let fri_prover_stats_reporter = FriProverQueueReporter::new( self.house_keeper_config.prover_stats_reporting_interval_ms, prover_pool.clone(), replica_pool.clone(), @@ -150,7 +148,7 @@ impl WiringLayer for HouseKeeperLayer { fri_prover_stats_reporter, })); - let fri_proof_compressor_stats_reporter = FriProofCompressorStatsReporter::new( + let fri_proof_compressor_stats_reporter = FriProofCompressorQueueReporter::new( self.house_keeper_config .proof_compressor_stats_reporting_interval_ms, prover_pool.clone(), @@ -268,7 +266,7 @@ impl Task for WaitingToQueuedFriWitnessJobMoverTask { #[derive(Debug)] struct FriWitnessGeneratorStatsReporterTask { - fri_witness_generator_stats_reporter: FriWitnessGeneratorStatsReporter, + fri_witness_generator_stats_reporter: FriWitnessGeneratorQueueReporter, } #[async_trait::async_trait] @@ -286,7 +284,7 @@ impl Task for FriWitnessGeneratorStatsReporterTask { #[derive(Debug)] struct FriProverStatsReporterTask { - fri_prover_stats_reporter: FriProverStatsReporter, + fri_prover_stats_reporter: FriProverQueueReporter, } #[async_trait::async_trait] @@ -302,7 +300,7 @@ impl Task for FriProverStatsReporterTask { #[derive(Debug)] struct FriProofCompressorStatsReporterTask { - fri_proof_compressor_stats_reporter: FriProofCompressorStatsReporter, + fri_proof_compressor_stats_reporter: FriProofCompressorQueueReporter, } #[async_trait::async_trait] @@ -338,7 +336,7 @@ impl Task for FriProofCompressorJobRetryManagerTask { #[derive(Debug)] struct FriProverJobArchiverTask { - fri_prover_job_archiver: FriProverJobArchiver, + fri_prover_job_archiver: FriProverJobsArchiver, } #[async_trait::async_trait] diff --git a/core/node/node_framework/src/implementations/layers/mod.rs b/core/node/node_framework/src/implementations/layers/mod.rs index f5b25ee277a8..cee9a0b6906d 100644 --- a/core/node/node_framework/src/implementations/layers/mod.rs +++ b/core/node/node_framework/src/implementations/layers/mod.rs @@ -17,4 +17,5 @@ pub mod proof_data_handler; pub mod query_eth_client; pub mod sigint; pub mod state_keeper; +pub mod tee_verifier_input_producer; pub mod web3_api; diff --git a/core/node/node_framework/src/implementations/layers/state_keeper/main_batch_executor.rs b/core/node/node_framework/src/implementations/layers/state_keeper/main_batch_executor.rs index 216d29fd81ac..2fb35fb201ab 100644 --- a/core/node/node_framework/src/implementations/layers/state_keeper/main_batch_executor.rs +++ b/core/node/node_framework/src/implementations/layers/state_keeper/main_batch_executor.rs @@ -1,30 +1,21 @@ -use std::sync::Arc; - -use zksync_config::{configs::chain::StateKeeperConfig, DBConfig}; -use zksync_state::{AsyncCatchupTask, RocksdbStorageOptions}; -use zksync_state_keeper::{AsyncRocksdbCache, MainBatchExecutor}; +use zksync_config::configs::chain::StateKeeperConfig; +use zksync_state_keeper::MainBatchExecutor; use crate::{ - implementations::resources::{ - pools::{MasterPool, PoolResource}, - state_keeper::BatchExecutorResource, - }, + implementations::resources::state_keeper::BatchExecutorResource, resource::Unique, - service::{ServiceContext, StopReceiver}, - task::Task, + service::ServiceContext, wiring_layer::{WiringError, WiringLayer}, }; #[derive(Debug)] pub struct MainBatchExecutorLayer { - db_config: DBConfig, state_keeper_config: StateKeeperConfig, } impl MainBatchExecutorLayer { - pub fn new(db_config: DBConfig, state_keeper_config: StateKeeperConfig) -> Self { + pub fn new(state_keeper_config: StateKeeperConfig) -> Self { Self { - db_config, state_keeper_config, } } @@ -37,44 +28,9 @@ impl WiringLayer for MainBatchExecutorLayer { } async fn wire(self: Box, mut context: ServiceContext<'_>) -> Result<(), WiringError> { - let master_pool = context.get_resource::>().await?; - - let cache_options = RocksdbStorageOptions { - block_cache_capacity: self - .db_config - .experimental - .state_keeper_db_block_cache_capacity(), - max_open_files: self.db_config.experimental.state_keeper_db_max_open_files, - }; - let (storage_factory, task) = AsyncRocksdbCache::new( - master_pool.get_singleton().await?, - self.db_config.state_keeper_db_path, - cache_options, - ); - let builder = MainBatchExecutor::new( - Arc::new(storage_factory), - self.state_keeper_config.save_call_traces, - false, - ); + let builder = MainBatchExecutor::new(self.state_keeper_config.save_call_traces, false); context.insert_resource(BatchExecutorResource(Unique::new(Box::new(builder))))?; - context.add_task(Box::new(RocksdbCatchupTask(task))); - Ok(()) - } -} - -#[derive(Debug)] -struct RocksdbCatchupTask(AsyncCatchupTask); - -#[async_trait::async_trait] -impl Task for RocksdbCatchupTask { - fn name(&self) -> &'static str { - "state_keeper/rocksdb_catchup_task" - } - - async fn run(self: Box, mut stop_receiver: StopReceiver) -> anyhow::Result<()> { - self.0.run(stop_receiver.0.clone()).await?; - stop_receiver.0.changed().await?; Ok(()) } } diff --git a/core/node/node_framework/src/implementations/layers/state_keeper/mempool_io.rs b/core/node/node_framework/src/implementations/layers/state_keeper/mempool_io.rs index eaf4b4204343..91be11ea8a8e 100644 --- a/core/node/node_framework/src/implementations/layers/state_keeper/mempool_io.rs +++ b/core/node/node_framework/src/implementations/layers/state_keeper/mempool_io.rs @@ -3,14 +3,16 @@ use std::sync::Arc; use anyhow::Context as _; use zksync_config::{ configs::{ - chain::{MempoolConfig, NetworkConfig, StateKeeperConfig}, + chain::{MempoolConfig, StateKeeperConfig}, wallets, }, ContractsConfig, }; use zksync_state_keeper::{ - MempoolFetcher, MempoolGuard, MempoolIO, OutputHandler, SequencerSealer, StateKeeperPersistence, + io::seal_logic::l2_block_seal_subtasks::L2BlockSealProcess, MempoolFetcher, MempoolGuard, + MempoolIO, OutputHandler, SequencerSealer, StateKeeperPersistence, TreeWritesPersistence, }; +use zksync_types::L2ChainId; use crate::{ implementations::resources::{ @@ -26,7 +28,7 @@ use crate::{ #[derive(Debug)] pub struct MempoolIOLayer { - network_config: NetworkConfig, + zksync_network_id: L2ChainId, contracts_config: ContractsConfig, state_keeper_config: StateKeeperConfig, mempool_config: MempoolConfig, @@ -35,14 +37,14 @@ pub struct MempoolIOLayer { impl MempoolIOLayer { pub fn new( - network_config: NetworkConfig, + zksync_network_id: L2ChainId, contracts_config: ContractsConfig, state_keeper_config: StateKeeperConfig, mempool_config: MempoolConfig, wallets: wallets::StateKeeper, ) -> Self { Self { - network_config, + zksync_network_id, contracts_config, state_keeper_config, mempool_config, @@ -79,16 +81,20 @@ impl WiringLayer for MempoolIOLayer { let batch_fee_input_provider = context.get_resource::().await?.0; let master_pool = context.get_resource::>().await?; - // Create miniblock sealer task. + // Create L2 block sealer task and output handler. + // L2 Block sealing process is parallelized, so we have to provide enough pooled connections. + let persistence_pool = master_pool + .get_custom(L2BlockSealProcess::subtasks_len()) + .await + .context("Get master pool")?; let (persistence, l2_block_sealer) = StateKeeperPersistence::new( - master_pool - .get_singleton() - .await - .context("Get master pool")?, + persistence_pool.clone(), self.contracts_config.l2_shared_bridge_addr.unwrap(), self.state_keeper_config.l2_block_seal_queue_capacity, ); - let output_handler = OutputHandler::new(Box::new(persistence)); + let tree_writes_persistence = TreeWritesPersistence::new(persistence_pool); + let output_handler = OutputHandler::new(Box::new(persistence)) + .with_handler(Box::new(tree_writes_persistence)); context.insert_resource(OutputHandlerResource(Unique::new(output_handler)))?; context.add_task(Box::new(L2BlockSealerTask(l2_block_sealer))); @@ -118,7 +124,7 @@ impl WiringLayer for MempoolIOLayer { &self.state_keeper_config, self.wallets.fee_account.address(), self.mempool_config.delay_interval(), - self.network_config.zksync_network_id, + self.zksync_network_id, ) .await?; context.insert_resource(StateKeeperIOResource(Unique::new(Box::new(io))))?; diff --git a/core/node/node_framework/src/implementations/layers/state_keeper/mod.rs b/core/node/node_framework/src/implementations/layers/state_keeper/mod.rs index 3b6becfe73c8..8d56bdd671a4 100644 --- a/core/node/node_framework/src/implementations/layers/state_keeper/mod.rs +++ b/core/node/node_framework/src/implementations/layers/state_keeper/mod.rs @@ -1,9 +1,11 @@ use std::sync::Arc; use anyhow::Context; +use zksync_config::DBConfig; +use zksync_state::{AsyncCatchupTask, ReadStorageFactory, RocksdbStorageOptions}; use zksync_state_keeper::{ - seal_criteria::ConditionalSealer, BatchExecutor, OutputHandler, StateKeeperIO, - ZkSyncStateKeeper, + seal_criteria::ConditionalSealer, AsyncRocksdbCache, BatchExecutor, OutputHandler, + StateKeeperIO, ZkSyncStateKeeper, }; use zksync_storage::RocksDB; @@ -11,9 +13,12 @@ pub mod main_batch_executor; pub mod mempool_io; use crate::{ - implementations::resources::state_keeper::{ - BatchExecutorResource, ConditionalSealerResource, OutputHandlerResource, - StateKeeperIOResource, + implementations::resources::{ + pools::{MasterPool, PoolResource}, + state_keeper::{ + BatchExecutorResource, ConditionalSealerResource, OutputHandlerResource, + StateKeeperIOResource, + }, }, service::{ServiceContext, StopReceiver}, task::Task, @@ -26,7 +31,15 @@ use crate::{ /// - `ConditionalSealerResource` /// #[derive(Debug)] -pub struct StateKeeperLayer; +pub struct StateKeeperLayer { + db_config: DBConfig, +} + +impl StateKeeperLayer { + pub fn new(db_config: DBConfig) -> Self { + Self { db_config } + } +} #[async_trait::async_trait] impl WiringLayer for StateKeeperLayer { @@ -54,12 +67,28 @@ impl WiringLayer for StateKeeperLayer { .take() .context("HandleStateKeeperOutput was provided but taken by another task")?; let sealer = context.get_resource::().await?.0; + let master_pool = context.get_resource::>().await?; + + let cache_options = RocksdbStorageOptions { + block_cache_capacity: self + .db_config + .experimental + .state_keeper_db_block_cache_capacity(), + max_open_files: self.db_config.experimental.state_keeper_db_max_open_files, + }; + let (storage_factory, task) = AsyncRocksdbCache::new( + master_pool.get_custom(2).await?, + self.db_config.state_keeper_db_path, + cache_options, + ); + context.add_task(Box::new(RocksdbCatchupTask(task))); context.add_task(Box::new(StateKeeperTask { io, batch_executor_base, output_handler, sealer, + storage_factory: Arc::new(storage_factory), })); Ok(()) } @@ -71,6 +100,7 @@ struct StateKeeperTask { batch_executor_base: Box, output_handler: OutputHandler, sealer: Arc, + storage_factory: Arc, } #[async_trait::async_trait] @@ -86,6 +116,7 @@ impl Task for StateKeeperTask { self.batch_executor_base, self.output_handler, self.sealer, + self.storage_factory, ); let result = state_keeper.run().await; @@ -97,3 +128,19 @@ impl Task for StateKeeperTask { result } } + +#[derive(Debug)] +struct RocksdbCatchupTask(AsyncCatchupTask); + +#[async_trait::async_trait] +impl Task for RocksdbCatchupTask { + fn name(&self) -> &'static str { + "state_keeper/rocksdb_catchup_task" + } + + async fn run(self: Box, mut stop_receiver: StopReceiver) -> anyhow::Result<()> { + self.0.run(stop_receiver.0.clone()).await?; + stop_receiver.0.changed().await?; + Ok(()) + } +} diff --git a/core/node/node_framework/src/implementations/layers/tee_verifier_input_producer.rs b/core/node/node_framework/src/implementations/layers/tee_verifier_input_producer.rs new file mode 100644 index 000000000000..a595e2eeb20b --- /dev/null +++ b/core/node/node_framework/src/implementations/layers/tee_verifier_input_producer.rs @@ -0,0 +1,62 @@ +use zksync_queued_job_processor::JobProcessor; +use zksync_tee_verifier_input_producer::TeeVerifierInputProducer; +use zksync_types::L2ChainId; + +use crate::{ + implementations::resources::{ + object_store::ObjectStoreResource, + pools::{MasterPool, PoolResource}, + }, + service::{ServiceContext, StopReceiver}, + task::Task, + wiring_layer::{WiringError, WiringLayer}, +}; + +#[derive(Debug)] +pub struct TeeVerifierInputProducerLayer { + l2_chain_id: L2ChainId, +} + +impl TeeVerifierInputProducerLayer { + pub fn new(l2_chain_id: L2ChainId) -> Self { + Self { l2_chain_id } + } +} + +#[async_trait::async_trait] +impl WiringLayer for TeeVerifierInputProducerLayer { + fn layer_name(&self) -> &'static str { + "tee_verifier_input_producer_layer" + } + + async fn wire(self: Box, mut context: ServiceContext<'_>) -> Result<(), WiringError> { + // Get resources. + let pool_resource = context + .get_resource::>() + .await? + .get() + .await?; + let object_store = context.get_resource::().await?; + let tee = + TeeVerifierInputProducer::new(pool_resource, object_store.0, self.l2_chain_id).await?; + + context.add_task(Box::new(TeeVerifierInputProducerTask { tee })); + + Ok(()) + } +} + +pub struct TeeVerifierInputProducerTask { + tee: TeeVerifierInputProducer, +} + +#[async_trait::async_trait] +impl Task for TeeVerifierInputProducerTask { + fn name(&self) -> &'static str { + "tee_verifier_input_producer" + } + + async fn run(self: Box, stop_receiver: StopReceiver) -> anyhow::Result<()> { + self.tee.run(stop_receiver.0, None).await + } +} diff --git a/core/node/node_framework/src/implementations/layers/web3_api/tree_api_client.rs b/core/node/node_framework/src/implementations/layers/web3_api/tree_api_client.rs index 065eabf6170b..42166e16b1dd 100644 --- a/core/node/node_framework/src/implementations/layers/web3_api/tree_api_client.rs +++ b/core/node/node_framework/src/implementations/layers/web3_api/tree_api_client.rs @@ -10,6 +10,10 @@ use crate::{ wiring_layer::{WiringError, WiringLayer}, }; +/// Layer that inserts the `TreeApiHttpClient` into the `ServiceContext` resources, if there is no +/// other client already inserted. +/// +/// In case a client is already provided in the contest, the layer does nothing. #[derive(Debug)] pub struct TreeApiClientLayer { url: Option, @@ -30,11 +34,25 @@ impl WiringLayer for TreeApiClientLayer { async fn wire(self: Box, mut context: ServiceContext<'_>) -> Result<(), WiringError> { if let Some(url) = &self.url { let client = Arc::new(TreeApiHttpClient::new(url)); + match context.insert_resource(TreeApiClientResource(client.clone())) { + Ok(()) => { + // There was no client added before, we added one. + } + Err(WiringError::ResourceAlreadyProvided { .. }) => { + // Some other client was already added. We don't want to replace it. + return Ok(()); + } + err @ Err(_) => { + // Propagate any other error. + return err; + } + } + + // Only provide the health check if necessary. let AppHealthCheckResource(app_health) = context.get_resource_or_default().await; app_health - .insert_custom_component(client.clone()) + .insert_custom_component(client) .map_err(WiringError::internal)?; - context.insert_resource(TreeApiClientResource(client))?; } Ok(()) } diff --git a/core/node/node_framework/src/service/mod.rs b/core/node/node_framework/src/service/mod.rs index 38902c254613..4a504f393c3a 100644 --- a/core/node/node_framework/src/service/mod.rs +++ b/core/node/node_framework/src/service/mod.rs @@ -146,6 +146,7 @@ impl ZkStackService { for resource in self.resources.values_mut() { resource.stored_resource_wired(); } + drop(self.resources); // Decrement reference counters for resources. tracing::info!("Wiring complete"); // Create a system task that is cancellation-aware and will only exit on either oneshot task failure or @@ -196,6 +197,7 @@ impl ZkStackService { tracing::info!("Remaining tasks finished without reaching timeouts"); } + tracing::info!("Exiting the service"); result?; Ok(()) } diff --git a/core/node/node_sync/src/tests.rs b/core/node/node_sync/src/tests.rs index 47c98d5cb693..1d6b3cd73505 100644 --- a/core/node/node_sync/src/tests.rs +++ b/core/node/node_sync/src/tests.rs @@ -13,8 +13,8 @@ use zksync_node_test_utils::{ use zksync_state_keeper::{ io::{L1BatchParams, L2BlockParams}, seal_criteria::NoopSealer, - testonly::test_batch_executor::TestBatchExecutorBuilder, - OutputHandler, StateKeeperPersistence, ZkSyncStateKeeper, + testonly::test_batch_executor::{MockReadStorageFactory, TestBatchExecutorBuilder}, + OutputHandler, StateKeeperPersistence, TreeWritesPersistence, ZkSyncStateKeeper, }; use zksync_types::{ api, @@ -105,7 +105,9 @@ impl StateKeeperHandles { let sync_state = SyncState::default(); let (persistence, l2_block_sealer) = StateKeeperPersistence::new(pool.clone(), Address::repeat_byte(1), 5); + let tree_writes_persistence = TreeWritesPersistence::new(pool.clone()); let output_handler = OutputHandler::new(Box::new(persistence.with_tx_insertion())) + .with_handler(Box::new(tree_writes_persistence)) .with_handler(Box::new(sync_state.clone())); tokio::spawn(l2_block_sealer.run()); @@ -130,6 +132,7 @@ impl StateKeeperHandles { Box::new(batch_executor_base), output_handler, Arc::new(NoopSealer), + Arc::new(MockReadStorageFactory), ); Self { diff --git a/core/node/node_sync/src/tree_data_fetcher/mod.rs b/core/node/node_sync/src/tree_data_fetcher/mod.rs index 9d9b663bd0ea..f143cc79198a 100644 --- a/core/node/node_sync/src/tree_data_fetcher/mod.rs +++ b/core/node/node_sync/src/tree_data_fetcher/mod.rs @@ -7,7 +7,7 @@ use serde::Serialize; #[cfg(test)] use tokio::sync::mpsc; use tokio::sync::watch; -use zksync_dal::{Connection, ConnectionPool, Core, CoreDal, DalError}; +use zksync_dal::{ConnectionPool, Core, CoreDal, DalError}; use zksync_health_check::{Health, HealthStatus, HealthUpdater, ReactiveHealthCheck}; use zksync_types::{block::L1BatchTreeData, Address, L1BatchNumber}; use zksync_web3_decl::{ @@ -169,27 +169,6 @@ impl TreeDataFetcher { }) } - async fn get_rollup_last_leaf_index( - storage: &mut Connection<'_, Core>, - mut l1_batch_number: L1BatchNumber, - ) -> anyhow::Result { - // With overwhelming probability, there's at least one initial write in an L1 batch, - // so this loop will execute for 1 iteration. - loop { - let maybe_index = storage - .storage_logs_dedup_dal() - .max_enumeration_index_for_l1_batch(l1_batch_number) - .await?; - if let Some(index) = maybe_index { - return Ok(index + 1); - } - tracing::warn!( - "No initial writes in L1 batch #{l1_batch_number}; trying the previous batch" - ); - l1_batch_number -= 1; - } - } - async fn step(&mut self) -> Result { let Some(l1_batch_to_fetch) = self.get_batch_to_fetch().await? else { return Ok(StepOutcome::NoProgress); @@ -218,8 +197,12 @@ impl TreeDataFetcher { let stage_latency = self.metrics.stage_latency[&ProcessingStage::Persistence].start(); let mut storage = self.pool.connection_tagged("tree_data_fetcher").await?; - let rollup_last_leaf_index = - Self::get_rollup_last_leaf_index(&mut storage, l1_batch_to_fetch).await?; + let rollup_last_leaf_index = storage + .storage_logs_dedup_dal() + .max_enumeration_index_by_l1_batch(l1_batch_to_fetch) + .await? + .unwrap_or(0) + + 1; let tree_data = L1BatchTreeData { hash: root_hash, rollup_last_leaf_index, diff --git a/core/node/node_sync/src/tree_data_fetcher/tests.rs b/core/node/node_sync/src/tree_data_fetcher/tests.rs index cb4cf030731e..cb25842f0517 100644 --- a/core/node/node_sync/src/tree_data_fetcher/tests.rs +++ b/core/node/node_sync/src/tree_data_fetcher/tests.rs @@ -10,6 +10,7 @@ use std::{ use assert_matches::assert_matches; use async_trait::async_trait; use test_casing::test_casing; +use zksync_dal::Connection; use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; use zksync_node_test_utils::{create_l1_batch, create_l2_block, prepare_recovery_snapshot}; use zksync_types::{AccountTreeId, Address, L2BlockNumber, StorageKey, StorageLog, H256}; diff --git a/core/node/state_keeper/src/batch_executor/main_executor.rs b/core/node/state_keeper/src/batch_executor/main_executor.rs index fa3bd5197f6d..ddbe166a04c1 100644 --- a/core/node/state_keeper/src/batch_executor/main_executor.rs +++ b/core/node/state_keeper/src/batch_executor/main_executor.rs @@ -30,19 +30,13 @@ use crate::{ /// Creates a "real" batch executor which maintains the VM (as opposed to the test builder which doesn't use the VM). #[derive(Debug, Clone)] pub struct MainBatchExecutor { - storage_factory: Arc, save_call_traces: bool, optional_bytecode_compression: bool, } impl MainBatchExecutor { - pub fn new( - storage_factory: Arc, - save_call_traces: bool, - optional_bytecode_compression: bool, - ) -> Self { + pub fn new(save_call_traces: bool, optional_bytecode_compression: bool) -> Self { Self { - storage_factory, save_call_traces, optional_bytecode_compression, } @@ -53,6 +47,7 @@ impl MainBatchExecutor { impl BatchExecutor for MainBatchExecutor { async fn init_batch( &mut self, + storage_factory: Arc, l1_batch_params: L1BatchEnv, system_env: SystemEnv, stop_receiver: &watch::Receiver, @@ -66,7 +61,6 @@ impl BatchExecutor for MainBatchExecutor { commands: commands_receiver, }; - let storage_factory = self.storage_factory.clone(); let stop_receiver = stop_receiver.clone(); let handle = tokio::task::spawn_blocking(move || { if let Some(storage) = Handle::current() diff --git a/core/node/state_keeper/src/batch_executor/mod.rs b/core/node/state_keeper/src/batch_executor/mod.rs index 671695503ecb..cc216c07bd44 100644 --- a/core/node/state_keeper/src/batch_executor/mod.rs +++ b/core/node/state_keeper/src/batch_executor/mod.rs @@ -1,4 +1,4 @@ -use std::fmt; +use std::{fmt, sync::Arc}; use async_trait::async_trait; use multivm::interface::{ @@ -8,6 +8,7 @@ use tokio::{ sync::{mpsc, oneshot, watch}, task::JoinHandle, }; +use zksync_state::ReadStorageFactory; use zksync_types::{vm_trace::Call, Transaction}; use zksync_utils::bytecode::CompressedBytecodeInfo; @@ -23,7 +24,7 @@ pub mod main_executor; /// Representation of a transaction executed in the virtual machine. #[derive(Debug, Clone)] -pub(crate) enum TxExecutionResult { +pub enum TxExecutionResult { /// Successful execution of the tx and the block tip dry run. Success { tx_result: Box, @@ -58,6 +59,7 @@ impl TxExecutionResult { pub trait BatchExecutor: 'static + Send + Sync + fmt::Debug { async fn init_batch( &mut self, + storage_factory: Arc, l1_batch_params: L1BatchEnv, system_env: SystemEnv, stop_receiver: &watch::Receiver, @@ -81,7 +83,7 @@ impl BatchExecutorHandle { Self { handle, commands } } - pub(super) async fn execute_tx(&self, tx: Transaction) -> TxExecutionResult { + pub async fn execute_tx(&self, tx: Transaction) -> TxExecutionResult { let tx_gas_limit = tx.gas_limit().as_u64(); let (response_sender, response_receiver) = oneshot::channel(); @@ -113,7 +115,7 @@ impl BatchExecutorHandle { res } - pub(super) async fn start_next_l2_block(&self, env: L2BlockEnv) { + pub async fn start_next_l2_block(&self, env: L2BlockEnv) { // While we don't get anything from the channel, it's useful to have it as a confirmation that the operation // indeed has been processed. let (response_sender, response_receiver) = oneshot::channel(); @@ -128,7 +130,7 @@ impl BatchExecutorHandle { latency.observe(); } - pub(super) async fn rollback_last_tx(&self) { + pub async fn rollback_last_tx(&self) { // While we don't get anything from the channel, it's useful to have it as a confirmation that the operation // indeed has been processed. let (response_sender, response_receiver) = oneshot::channel(); @@ -143,7 +145,7 @@ impl BatchExecutorHandle { latency.observe(); } - pub(super) async fn finish_batch(self) -> FinishedL1Batch { + pub async fn finish_batch(self) -> FinishedL1Batch { let (response_sender, response_receiver) = oneshot::channel(); self.commands .send(Command::FinishBatch(response_sender)) diff --git a/core/node/state_keeper/src/batch_executor/tests/tester.rs b/core/node/state_keeper/src/batch_executor/tests/tester.rs index b77d044f136f..380e34bf29be 100644 --- a/core/node/state_keeper/src/batch_executor/tests/tester.rs +++ b/core/node/state_keeper/src/batch_executor/tests/tester.rs @@ -145,11 +145,10 @@ impl Tester { l1_batch_env: L1BatchEnv, system_env: SystemEnv, ) -> BatchExecutorHandle { - let mut batch_executor = - MainBatchExecutor::new(storage_factory, self.config.save_call_traces, false); + let mut batch_executor = MainBatchExecutor::new(self.config.save_call_traces, false); let (_stop_sender, stop_receiver) = watch::channel(false); batch_executor - .init_batch(l1_batch_env, system_env, &stop_receiver) + .init_batch(storage_factory, l1_batch_env, system_env, &stop_receiver) .await .expect("Batch executor was interrupted") } diff --git a/core/node/state_keeper/src/io/mod.rs b/core/node/state_keeper/src/io/mod.rs index 6cd6f818f401..8cdfbd59121f 100644 --- a/core/node/state_keeper/src/io/mod.rs +++ b/core/node/state_keeper/src/io/mod.rs @@ -12,7 +12,7 @@ use zksync_types::{ pub use self::{ common::IoCursor, output_handler::{OutputHandler, StateKeeperOutputHandler}, - persistence::{L2BlockSealerTask, StateKeeperPersistence}, + persistence::{L2BlockSealerTask, StateKeeperPersistence, TreeWritesPersistence}, }; use super::seal_criteria::IoSealCriteria; diff --git a/core/node/state_keeper/src/io/persistence.rs b/core/node/state_keeper/src/io/persistence.rs index aaaf0712efa3..25b1ae9e6ea4 100644 --- a/core/node/state_keeper/src/io/persistence.rs +++ b/core/node/state_keeper/src/io/persistence.rs @@ -4,10 +4,12 @@ use std::{sync::Arc, time::Instant}; use anyhow::Context as _; use async_trait::async_trait; +use multivm::zk_evm_latest::ethereum_types::H256; use tokio::sync::{mpsc, oneshot}; -use zksync_dal::{ConnectionPool, Core}; +use zksync_dal::{ConnectionPool, Core, CoreDal}; use zksync_shared_metrics::{BlockStage, APP_METRICS}; -use zksync_types::Address; +use zksync_types::{writes::TreeWrite, AccountTreeId, Address, StorageKey}; +use zksync_utils::u256_to_h256; use crate::{ io::{ @@ -247,6 +249,109 @@ impl L2BlockSealerTask { } } +/// Stores tree writes for L1 batches to Postgres. +/// It is expected to be run after `StateKeeperPersistence` as it appends data to `l1_batches` table. +#[derive(Debug)] +pub struct TreeWritesPersistence { + pool: ConnectionPool, +} + +impl TreeWritesPersistence { + pub fn new(pool: ConnectionPool) -> Self { + Self { pool } + } +} + +#[async_trait] +impl StateKeeperOutputHandler for TreeWritesPersistence { + async fn handle_l2_block(&mut self, _updates_manager: &UpdatesManager) -> anyhow::Result<()> { + Ok(()) + } + + async fn handle_l1_batch( + &mut self, + updates_manager: Arc, + ) -> anyhow::Result<()> { + let mut connection = self.pool.connection_tagged("state_keeper").await?; + let finished_batch = updates_manager + .l1_batch + .finished + .as_ref() + .context("L1 batch is not actually finished")?; + + let mut next_index = connection + .storage_logs_dedup_dal() + .max_enumeration_index_by_l1_batch(updates_manager.l1_batch.number - 1) + .await? + .unwrap_or(0) + + 1; + let tree_input: Vec<_> = if let Some(state_diffs) = &finished_batch.state_diffs { + state_diffs + .iter() + .map(|diff| { + let leaf_index = if diff.is_write_initial() { + next_index += 1; + next_index - 1 + } else { + diff.enumeration_index + }; + TreeWrite { + address: diff.address, + key: u256_to_h256(diff.key), + value: u256_to_h256(diff.final_value), + leaf_index, + } + }) + .collect() + } else { + let deduplicated_writes = finished_batch + .final_execution_state + .deduplicated_storage_log_queries + .iter() + .filter(|log_query| log_query.rw_flag); + let deduplicated_writes_hashed_keys: Vec<_> = deduplicated_writes + .clone() + .map(|log| { + H256(StorageKey::raw_hashed_key( + &log.address, + &u256_to_h256(log.key), + )) + }) + .collect(); + let non_initial_writes = connection + .storage_logs_dal() + .get_l1_batches_and_indices_for_initial_writes(&deduplicated_writes_hashed_keys) + .await?; + deduplicated_writes + .map(|log| { + let key = + StorageKey::new(AccountTreeId::new(log.address), u256_to_h256(log.key)); + let leaf_index = + if let Some((_, leaf_index)) = non_initial_writes.get(&key.hashed_key()) { + *leaf_index + } else { + next_index += 1; + next_index - 1 + }; + TreeWrite { + address: log.address, + key: u256_to_h256(log.key), + value: u256_to_h256(log.written_value), + leaf_index, + } + }) + .collect() + }; + + connection + .blocks_dal() + .set_tree_writes(updates_manager.l1_batch.number, tree_input) + .await?; + + Ok(()) + } +} + #[cfg(test)] mod tests { use std::collections::HashSet; @@ -257,8 +362,9 @@ mod tests { use zksync_dal::CoreDal; use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; use zksync_types::{ - api::TransactionStatus, block::BlockGasCount, tx::ExecutionMetrics, AccountTreeId, - L1BatchNumber, L2BlockNumber, StorageKey, StorageLogQueryType, + api::TransactionStatus, block::BlockGasCount, tx::ExecutionMetrics, + writes::StateDiffRecord, AccountTreeId, L1BatchNumber, L2BlockNumber, StorageKey, + StorageLogQueryType, }; use zksync_utils::u256_to_h256; @@ -270,6 +376,7 @@ mod tests { create_execution_result, create_transaction, create_updates_manager, default_l1_batch_env, default_system_env, Query, }, + OutputHandler, }; async fn test_l2_block_and_l1_batch_processing( @@ -280,6 +387,12 @@ mod tests { insert_genesis_batch(&mut storage, &GenesisParams::mock()) .await .unwrap(); + let initial_writes_in_genesis_batch = storage + .storage_logs_dedup_dal() + .max_enumeration_index_by_l1_batch(L1BatchNumber(0)) + .await + .unwrap() + .unwrap(); // Save metadata for the genesis L1 batch so that we don't hang in `seal_l1_batch`. storage .blocks_dal() @@ -288,10 +401,12 @@ mod tests { .unwrap(); drop(storage); - let (mut persistence, l2_block_sealer) = + let (persistence, l2_block_sealer) = StateKeeperPersistence::new(pool.clone(), Address::default(), l2_block_sealer_capacity); + let mut output_handler = OutputHandler::new(Box::new(persistence)) + .with_handler(Box::new(TreeWritesPersistence::new(pool.clone()))); tokio::spawn(l2_block_sealer.run()); - execute_mock_batch(&mut persistence).await; + execute_mock_batch(&mut output_handler).await; // Check that L2 block #1 and L1 batch #1 are persisted. let mut storage = pool.connection().await.unwrap(); @@ -327,9 +442,20 @@ mod tests { .await .unwrap(); assert_eq!(protective_reads.len(), 1, "{protective_reads:?}"); + let tree_writes = storage + .blocks_dal() + .get_tree_writes(L1BatchNumber(1)) + .await + .unwrap() + .unwrap(); + assert_eq!(tree_writes.len(), 1, "{tree_writes:?}"); + // This write is initial and should have the next index. + let actual_index = tree_writes[0].leaf_index; + let expected_index = initial_writes_in_genesis_batch + 1; + assert_eq!(actual_index, expected_index); } - async fn execute_mock_batch(persistence: &mut StateKeeperPersistence) -> H256 { + async fn execute_mock_batch(output_handler: &mut OutputHandler) -> H256 { let l1_batch_env = default_l1_batch_env(1, 1, Address::random()); let mut updates = UpdatesManager::new(&l1_batch_env, &default_system_env()); @@ -349,7 +475,7 @@ mod tests { ExecutionMetrics::default(), vec![], ); - persistence.handle_l2_block(&updates).await.unwrap(); + output_handler.handle_l2_block(&updates).await.unwrap(); updates.push_l2_block(L2BlockParams { timestamp: 1, virtual_blocks: 1, @@ -360,7 +486,7 @@ mod tests { .final_execution_state .deduplicated_storage_log_queries = storage_logs.iter().map(|query| query.log_query).collect(); - batch_result.initially_written_slots = Some( + batch_result.state_diffs = Some( storage_logs .into_iter() .filter(|&log| log.log_type == StorageLogQueryType::InitialWrite) @@ -369,13 +495,20 @@ mod tests { AccountTreeId::new(log.log_query.address), u256_to_h256(log.log_query.key), ); - key.hashed_key() + StateDiffRecord { + address: log.log_query.address, + key: log.log_query.key, + derived_key: key.hashed_key().0, + enumeration_index: 0, + initial_value: log.log_query.read_value, + final_value: log.log_query.written_value, + } }) .collect(), ); updates.finish_batch(batch_result); - persistence + output_handler .handle_l1_batch(Arc::new(updates)) .await .unwrap(); @@ -413,9 +546,10 @@ mod tests { let (mut persistence, l2_block_sealer) = StateKeeperPersistence::new(pool.clone(), Address::default(), 1); persistence = persistence.with_tx_insertion().without_protective_reads(); + let mut output_handler = OutputHandler::new(Box::new(persistence)); tokio::spawn(l2_block_sealer.run()); - let tx_hash = execute_mock_batch(&mut persistence).await; + let tx_hash = execute_mock_batch(&mut output_handler).await; // Check that the transaction is persisted. let mut storage = pool.connection().await.unwrap(); diff --git a/core/node/state_keeper/src/io/seal_logic/mod.rs b/core/node/state_keeper/src/io/seal_logic/mod.rs index 1880503ff638..3e8277485d2f 100644 --- a/core/node/state_keeper/src/io/seal_logic/mod.rs +++ b/core/node/state_keeper/src/io/seal_logic/mod.rs @@ -169,13 +169,15 @@ impl UpdatesManager { .await?; progress.observe(None); - let (deduplicated_writes, protective_reads): (Vec<_>, Vec<_>) = finished_batch - .final_execution_state - .deduplicated_storage_log_queries - .iter() - .partition(|log_query| log_query.rw_flag); if insert_protective_reads { let progress = L1_BATCH_METRICS.start(L1BatchSealStage::InsertProtectiveReads); + let protective_reads: Vec<_> = finished_batch + .final_execution_state + .deduplicated_storage_log_queries + .iter() + .filter(|log_query| !log_query.rw_flag) + .copied() + .collect(); transaction .storage_logs_dedup_dal() .insert_protective_reads(self.l1_batch.number, &protective_reads) @@ -184,50 +186,62 @@ impl UpdatesManager { } let progress = L1_BATCH_METRICS.start(L1BatchSealStage::FilterWrittenSlots); - let written_storage_keys: Vec<_> = - if let Some(initially_written_slots) = &finished_batch.initially_written_slots { - deduplicated_writes - .iter() - .filter_map(|log| { - let key = - StorageKey::new(AccountTreeId::new(log.address), u256_to_h256(log.key)); - initially_written_slots - .contains(&key.hashed_key()) - .then_some(key) - }) - .collect() - } else { - let deduplicated_writes_hashed_keys: Vec<_> = deduplicated_writes + let (initial_writes, all_writes_len): (Vec<_>, usize) = if let Some(state_diffs) = + &finished_batch.state_diffs + { + let all_writes_len = state_diffs.len(); + + ( + state_diffs .iter() - .map(|log| { - H256(StorageKey::raw_hashed_key( - &log.address, - &u256_to_h256(log.key), - )) + .filter(|diff| diff.is_write_initial()) + .map(|diff| { + StorageKey::new(AccountTreeId::new(diff.address), u256_to_h256(diff.key)) }) - .collect(); - let non_initial_writes = transaction - .storage_logs_dedup_dal() - .filter_written_slots(&deduplicated_writes_hashed_keys) - .await?; + .collect(), + all_writes_len, + ) + } else { + let deduplicated_writes = finished_batch + .final_execution_state + .deduplicated_storage_log_queries + .iter() + .filter(|log_query| log_query.rw_flag); + + let deduplicated_writes_hashed_keys: Vec<_> = deduplicated_writes + .clone() + .map(|log| { + H256(StorageKey::raw_hashed_key( + &log.address, + &u256_to_h256(log.key), + )) + }) + .collect(); + let all_writes_len = deduplicated_writes_hashed_keys.len(); + let non_initial_writes = transaction + .storage_logs_dedup_dal() + .filter_written_slots(&deduplicated_writes_hashed_keys) + .await?; + ( deduplicated_writes - .iter() .filter_map(|log| { let key = StorageKey::new(AccountTreeId::new(log.address), u256_to_h256(log.key)); (!non_initial_writes.contains(&key.hashed_key())).then_some(key) }) - .collect() - }; - progress.observe(deduplicated_writes.len()); + .collect(), + all_writes_len, + ) + }; + progress.observe(all_writes_len); let progress = L1_BATCH_METRICS.start(L1BatchSealStage::InsertInitialWrites); transaction .storage_logs_dedup_dal() - .insert_initial_writes(self.l1_batch.number, &written_storage_keys) + .insert_initial_writes(self.l1_batch.number, &initial_writes) .await?; - progress.observe(written_storage_keys.len()); + progress.observe(initial_writes.len()); let progress = L1_BATCH_METRICS.start(L1BatchSealStage::CommitL1Batch); transaction.commit().await?; @@ -236,7 +250,7 @@ impl UpdatesManager { let writes_metrics = self.storage_writes_deduplicator.metrics(); // Sanity check metrics. anyhow::ensure!( - deduplicated_writes.len() + all_writes_len == writes_metrics.initial_storage_writes + writes_metrics.repeated_storage_writes, "Results of in-flight and common deduplications are mismatched" ); diff --git a/core/node/state_keeper/src/keeper.rs b/core/node/state_keeper/src/keeper.rs index 6aee5bb0c1ea..d04e4c2e5920 100644 --- a/core/node/state_keeper/src/keeper.rs +++ b/core/node/state_keeper/src/keeper.rs @@ -7,6 +7,7 @@ use std::{ use anyhow::Context as _; use multivm::interface::{Halt, L1BatchEnv, SystemEnv}; use tokio::sync::watch; +use zksync_state::ReadStorageFactory; use zksync_types::{ block::L2BlockExecutionData, l2::TransactionType, protocol_upgrade::ProtocolUpgradeTx, protocol_version::ProtocolVersionId, storage_writes_deduplicator::StorageWritesDeduplicator, @@ -61,6 +62,7 @@ pub struct ZkSyncStateKeeper { output_handler: OutputHandler, batch_executor_base: Box, sealer: Arc, + storage_factory: Arc, } impl ZkSyncStateKeeper { @@ -70,6 +72,7 @@ impl ZkSyncStateKeeper { batch_executor_base: Box, output_handler: OutputHandler, sealer: Arc, + storage_factory: Arc, ) -> Self { Self { stop_receiver, @@ -77,6 +80,7 @@ impl ZkSyncStateKeeper { batch_executor_base, output_handler, sealer, + storage_factory, } } @@ -142,6 +146,7 @@ impl ZkSyncStateKeeper { let mut batch_executor = self .batch_executor_base .init_batch( + self.storage_factory.clone(), l1_batch_env.clone(), system_env.clone(), &self.stop_receiver, @@ -194,6 +199,7 @@ impl ZkSyncStateKeeper { batch_executor = self .batch_executor_base .init_batch( + self.storage_factory.clone(), l1_batch_env.clone(), system_env.clone(), &self.stop_receiver, diff --git a/core/node/state_keeper/src/lib.rs b/core/node/state_keeper/src/lib.rs index 2e48160b4530..4920e2514b0a 100644 --- a/core/node/state_keeper/src/lib.rs +++ b/core/node/state_keeper/src/lib.rs @@ -10,16 +10,18 @@ use zksync_node_fee_model::BatchFeeModelInputProvider; use zksync_types::L2ChainId; pub use self::{ - batch_executor::{main_executor::MainBatchExecutor, BatchExecutor}, + batch_executor::{ + main_executor::MainBatchExecutor, BatchExecutor, BatchExecutorHandle, TxExecutionResult, + }, io::{ - mempool::MempoolIO, L2BlockSealerTask, OutputHandler, StateKeeperIO, - StateKeeperOutputHandler, StateKeeperPersistence, + mempool::MempoolIO, L2BlockParams, L2BlockSealerTask, OutputHandler, StateKeeperIO, + StateKeeperOutputHandler, StateKeeperPersistence, TreeWritesPersistence, }, keeper::ZkSyncStateKeeper, mempool_actor::MempoolFetcher, seal_criteria::SequencerSealer, state_keeper_storage::AsyncRocksdbCache, - types::MempoolGuard, + types::{ExecutionMetricsForCriteria, MempoolGuard}, updates::UpdatesManager, }; @@ -50,11 +52,7 @@ pub async fn create_state_keeper( output_handler: OutputHandler, stop_receiver: watch::Receiver, ) -> ZkSyncStateKeeper { - let batch_executor_base = MainBatchExecutor::new( - Arc::new(async_cache), - state_keeper_config.save_call_traces, - false, - ); + let batch_executor_base = MainBatchExecutor::new(state_keeper_config.save_call_traces, false); let io = MempoolIO::new( mempool, @@ -76,5 +74,6 @@ pub async fn create_state_keeper( Box::new(batch_executor_base), output_handler, Arc::new(sealer), + Arc::new(async_cache), ) } diff --git a/core/node/state_keeper/src/testonly/mod.rs b/core/node/state_keeper/src/testonly/mod.rs index 56c8a773c473..77e913fb8b75 100644 --- a/core/node/state_keeper/src/testonly/mod.rs +++ b/core/node/state_keeper/src/testonly/mod.rs @@ -1,6 +1,8 @@ //! Test utilities that can be used for testing sequencer that may //! be useful outside of this crate. +use std::sync::Arc; + use async_trait::async_trait; use multivm::{ interface::{ @@ -12,6 +14,7 @@ use multivm::{ use once_cell::sync::Lazy; use tokio::sync::{mpsc, watch}; use zksync_contracts::BaseSystemContracts; +use zksync_state::ReadStorageFactory; use crate::{ batch_executor::{BatchExecutor, BatchExecutorHandle, Command, TxExecutionResult}, @@ -45,7 +48,7 @@ pub(super) fn default_vm_batch_result() -> FinishedL1Batch { }, final_bootloader_memory: Some(vec![]), pubdata_input: Some(vec![]), - initially_written_slots: Some(vec![]), + state_diffs: Some(vec![]), } } @@ -76,6 +79,7 @@ pub struct MockBatchExecutor; impl BatchExecutor for MockBatchExecutor { async fn init_batch( &mut self, + _storage_factory: Arc, _l1batch_params: L1BatchEnv, _system_env: SystemEnv, _stop_receiver: &watch::Receiver, diff --git a/core/node/state_keeper/src/testonly/test_batch_executor.rs b/core/node/state_keeper/src/testonly/test_batch_executor.rs index 39bc20a5d9fc..c748a25ed79f 100644 --- a/core/node/state_keeper/src/testonly/test_batch_executor.rs +++ b/core/node/state_keeper/src/testonly/test_batch_executor.rs @@ -17,9 +17,10 @@ use multivm::{ interface::{ExecutionResult, L1BatchEnv, SystemEnv, VmExecutionResultAndLogs}, vm_latest::constants::BATCH_COMPUTATIONAL_GAS_LIMIT, }; -use tokio::sync::{mpsc, watch}; +use tokio::sync::{mpsc, watch, watch::Receiver}; use zksync_contracts::BaseSystemContracts; use zksync_node_test_utils::create_l2_transaction; +use zksync_state::{PgOrRocksdbStorage, ReadStorageFactory}; use zksync_types::{ fee_model::BatchFeeInput, protocol_upgrade::ProtocolUpgradeTx, Address, L1BatchNumber, L2BlockNumber, L2ChainId, ProtocolVersionId, Transaction, H256, @@ -204,6 +205,7 @@ impl TestScenario { Box::new(batch_executor_base), output_handler, Arc::new(sealer), + Arc::new(MockReadStorageFactory), ); let sk_thread = tokio::spawn(state_keeper.run()); @@ -410,6 +412,7 @@ impl TestBatchExecutorBuilder { impl BatchExecutor for TestBatchExecutorBuilder { async fn init_batch( &mut self, + _storage_factory: Arc, _l1batch_params: L1BatchEnv, _system_env: SystemEnv, _stop_receiver: &watch::Receiver, @@ -810,6 +813,7 @@ pub(crate) struct MockBatchExecutor; impl BatchExecutor for MockBatchExecutor { async fn init_batch( &mut self, + _storage_factory: Arc, _l1batch_params: L1BatchEnv, _system_env: SystemEnv, _stop_receiver: &watch::Receiver, @@ -833,3 +837,18 @@ impl BatchExecutor for MockBatchExecutor { Some(BatchExecutorHandle::from_raw(handle, send)) } } + +#[derive(Debug)] +pub struct MockReadStorageFactory; + +#[async_trait] +impl ReadStorageFactory for MockReadStorageFactory { + async fn access_storage( + &self, + _stop_receiver: &Receiver, + _l1_batch_number: L1BatchNumber, + ) -> anyhow::Result>> { + // Presume that the storage is never accessed in mocked environment + unimplemented!() + } +} diff --git a/core/node/state_keeper/src/tests/mod.rs b/core/node/state_keeper/src/tests/mod.rs index 2b347c0629ea..18d25faf4a4c 100644 --- a/core/node/state_keeper/src/tests/mod.rs +++ b/core/node/state_keeper/src/tests/mod.rs @@ -38,7 +38,7 @@ use crate::{ successful_exec, test_batch_executor::{ random_tx, random_upgrade_tx, rejected_exec, successful_exec_with_metrics, - TestBatchExecutorBuilder, TestIO, TestScenario, FEE_ACCOUNT, + MockReadStorageFactory, TestBatchExecutorBuilder, TestIO, TestScenario, FEE_ACCOUNT, }, BASE_SYSTEM_CONTRACTS, }, @@ -444,6 +444,7 @@ async fn load_upgrade_tx() { Box::new(batch_executor_base), output_handler, Arc::new(sealer), + Arc::new(MockReadStorageFactory), ); // Since the version hasn't changed, and we are not using shared bridge, we should not load any diff --git a/core/node/state_keeper/src/updates/mod.rs b/core/node/state_keeper/src/updates/mod.rs index 05de56b69ecd..bb33a6f58678 100644 --- a/core/node/state_keeper/src/updates/mod.rs +++ b/core/node/state_keeper/src/updates/mod.rs @@ -102,7 +102,7 @@ impl UpdatesManager { self.protocol_version } - pub(crate) fn extend_from_executed_transaction( + pub fn extend_from_executed_transaction( &mut self, tx: Transaction, tx_execution_result: VmExecutionResultAndLogs, @@ -148,7 +148,7 @@ impl UpdatesManager { /// Pushes a new L2 block with the specified timestamp into this manager. The previously /// held L2 block is considered sealed and is used to extend the L1 batch data. - pub(crate) fn push_l2_block(&mut self, l2_block_params: L2BlockParams) { + pub fn push_l2_block(&mut self, l2_block_params: L2BlockParams) { let new_l2_block_updates = L2BlockUpdates::new( l2_block_params.timestamp, self.l2_block.number + 1, diff --git a/core/node/tee_verifier_input_producer/src/lib.rs b/core/node/tee_verifier_input_producer/src/lib.rs index 4b335a218c70..47ae9cd87c3f 100644 --- a/core/node/tee_verifier_input_producer/src/lib.rs +++ b/core/node/tee_verifier_input_producer/src/lib.rs @@ -15,7 +15,7 @@ use multivm::zk_evm_latest::ethereum_types::H256; use tokio::{runtime::Handle, task::JoinHandle}; use vm_utils::storage::L1BatchParamsProvider; use zksync_dal::{tee_verifier_input_producer_dal::JOB_MAX_ATTEMPT, ConnectionPool, Core, CoreDal}; -use zksync_object_store::{ObjectStore, ObjectStoreFactory}; +use zksync_object_store::ObjectStore; use zksync_prover_interface::inputs::PrepareBasicCircuitsJob; use zksync_queued_job_processor::JobProcessor; use zksync_state::{PostgresStorage, ReadStorage}; @@ -38,12 +38,12 @@ pub struct TeeVerifierInputProducer { impl TeeVerifierInputProducer { pub async fn new( connection_pool: ConnectionPool, - store_factory: &ObjectStoreFactory, + object_store: Arc, l2_chain_id: L2ChainId, ) -> anyhow::Result { Ok(TeeVerifierInputProducer { connection_pool, - object_store: store_factory.create_store().await, + object_store, l2_chain_id, }) } diff --git a/core/node/vm_runner/Cargo.toml b/core/node/vm_runner/Cargo.toml index 94d5fa01443d..67de95f60cb0 100644 --- a/core/node/vm_runner/Cargo.toml +++ b/core/node/vm_runner/Cargo.toml @@ -29,6 +29,8 @@ dashmap.workspace = true [dev-dependencies] zksync_node_test_utils.workspace = true zksync_node_genesis.workspace = true +zksync_test_account.workspace = true +zksync_utils.workspace = true backon.workspace = true futures = { workspace = true, features = ["compat"] } rand.workspace = true diff --git a/core/node/vm_runner/src/io.rs b/core/node/vm_runner/src/io.rs index 2b2e85abd439..e67da0e8235c 100644 --- a/core/node/vm_runner/src/io.rs +++ b/core/node/vm_runner/src/io.rs @@ -9,7 +9,7 @@ use zksync_types::L1BatchNumber; #[async_trait] pub trait VmRunnerIo: Debug + Send + Sync + 'static { /// Unique name of the VM runner instance. - fn name() -> &'static str; + fn name(&self) -> &'static str; /// Returns the last L1 batch number that has been processed by this VM runner instance. /// diff --git a/core/node/vm_runner/src/lib.rs b/core/node/vm_runner/src/lib.rs index 44db8564450d..4664d4eb8e11 100644 --- a/core/node/vm_runner/src/lib.rs +++ b/core/node/vm_runner/src/lib.rs @@ -5,6 +5,7 @@ mod io; mod output_handler; +mod process; mod storage; #[cfg(test)] @@ -14,4 +15,5 @@ pub use io::VmRunnerIo; pub use output_handler::{ ConcurrentOutputHandlerFactory, ConcurrentOutputHandlerFactoryTask, OutputHandlerFactory, }; +pub use process::VmRunner; pub use storage::{BatchExecuteData, VmRunnerStorage}; diff --git a/core/node/vm_runner/src/output_handler.rs b/core/node/vm_runner/src/output_handler.rs index 39cb1d33615a..30fe9e0c9010 100644 --- a/core/node/vm_runner/src/output_handler.rs +++ b/core/node/vm_runner/src/output_handler.rs @@ -103,7 +103,7 @@ impl OutputHandlerFactory &mut self, l1_batch_number: L1BatchNumber, ) -> anyhow::Result> { - let mut conn = self.pool.connection_tagged(Io::name()).await?; + let mut conn = self.pool.connection_tagged(self.io.name()).await?; let latest_processed_batch = self.io.latest_processed_batch(&mut conn).await?; let last_processable_batch = self.io.last_ready_to_be_loaded_batch(&mut conn).await?; drop(conn); @@ -211,7 +211,7 @@ impl ConcurrentOutputHandlerFactoryTask { pub async fn run(self, stop_receiver: watch::Receiver) -> anyhow::Result<()> { const SLEEP_INTERVAL: Duration = Duration::from_millis(50); - let mut conn = self.pool.connection_tagged(Io::name()).await?; + let mut conn = self.pool.connection_tagged(self.io.name()).await?; let mut latest_processed_batch = self.io.latest_processed_batch(&mut conn).await?; drop(conn); loop { @@ -239,7 +239,7 @@ impl ConcurrentOutputHandlerFactoryTask { .await .context("failed to await for batch to be processed")??; latest_processed_batch += 1; - let mut conn = self.pool.connection_tagged(Io::name()).await?; + let mut conn = self.pool.connection_tagged(self.io.name()).await?; self.io .mark_l1_batch_as_completed(&mut conn, latest_processed_batch) .await?; @@ -248,294 +248,3 @@ impl ConcurrentOutputHandlerFactoryTask { } } } - -#[cfg(test)] -mod tests { - use std::{collections::HashMap, sync::Arc, time::Duration}; - - use async_trait::async_trait; - use backon::{ConstantBuilder, Retryable}; - use multivm::interface::{L1BatchEnv, L2BlockEnv, SystemEnv, TxExecutionMode}; - use tokio::{ - sync::{watch, RwLock}, - task::JoinHandle, - }; - use zksync_contracts::{BaseSystemContracts, SystemContractCode}; - use zksync_dal::{Connection, ConnectionPool, Core}; - use zksync_state_keeper::{StateKeeperOutputHandler, UpdatesManager}; - use zksync_types::L1BatchNumber; - - use crate::{ConcurrentOutputHandlerFactory, OutputHandlerFactory, VmRunnerIo}; - - #[derive(Debug, Default)] - struct IoMock { - current: L1BatchNumber, - max: u32, - } - - #[async_trait] - impl VmRunnerIo for Arc> { - fn name() -> &'static str { - "io_mock" - } - - async fn latest_processed_batch( - &self, - _conn: &mut Connection<'_, Core>, - ) -> anyhow::Result { - Ok(self.read().await.current) - } - - async fn last_ready_to_be_loaded_batch( - &self, - _conn: &mut Connection<'_, Core>, - ) -> anyhow::Result { - let io = self.read().await; - Ok(io.current + io.max) - } - - async fn mark_l1_batch_as_completed( - &self, - _conn: &mut Connection<'_, Core>, - l1_batch_number: L1BatchNumber, - ) -> anyhow::Result<()> { - self.write().await.current = l1_batch_number; - Ok(()) - } - } - - #[derive(Debug)] - struct TestOutputFactory { - delays: HashMap, - } - - #[async_trait] - impl OutputHandlerFactory for TestOutputFactory { - async fn create_handler( - &mut self, - l1_batch_number: L1BatchNumber, - ) -> anyhow::Result> { - let delay = self.delays.get(&l1_batch_number).copied(); - #[derive(Debug)] - struct TestOutputHandler { - delay: Option, - } - #[async_trait] - impl StateKeeperOutputHandler for TestOutputHandler { - async fn handle_l2_block( - &mut self, - _updates_manager: &UpdatesManager, - ) -> anyhow::Result<()> { - Ok(()) - } - - async fn handle_l1_batch( - &mut self, - _updates_manager: Arc, - ) -> anyhow::Result<()> { - if let Some(delay) = self.delay { - tokio::time::sleep(delay).await - } - Ok(()) - } - } - Ok(Box::new(TestOutputHandler { delay })) - } - } - - struct OutputHandlerTester { - io: Arc>, - output_factory: ConcurrentOutputHandlerFactory>, TestOutputFactory>, - tasks: Vec>, - stop_sender: watch::Sender, - } - - impl OutputHandlerTester { - fn new( - io: Arc>, - pool: ConnectionPool, - delays: HashMap, - ) -> Self { - let test_factory = TestOutputFactory { delays }; - let (output_factory, task) = - ConcurrentOutputHandlerFactory::new(pool, io.clone(), test_factory); - let (stop_sender, stop_receiver) = watch::channel(false); - let join_handle = - tokio::task::spawn(async move { task.run(stop_receiver).await.unwrap() }); - let tasks = vec![join_handle]; - Self { - io, - output_factory, - tasks, - stop_sender, - } - } - - async fn spawn_test_task(&mut self, l1_batch_number: L1BatchNumber) -> anyhow::Result<()> { - let mut output_handler = self.output_factory.create_handler(l1_batch_number).await?; - let join_handle = tokio::task::spawn(async move { - let l1_batch_env = L1BatchEnv { - previous_batch_hash: None, - number: Default::default(), - timestamp: 0, - fee_input: Default::default(), - fee_account: Default::default(), - enforced_base_fee: None, - first_l2_block: L2BlockEnv { - number: 0, - timestamp: 0, - prev_block_hash: Default::default(), - max_virtual_blocks_to_create: 0, - }, - }; - let system_env = SystemEnv { - zk_porter_available: false, - version: Default::default(), - base_system_smart_contracts: BaseSystemContracts { - bootloader: SystemContractCode { - code: vec![], - hash: Default::default(), - }, - default_aa: SystemContractCode { - code: vec![], - hash: Default::default(), - }, - }, - bootloader_gas_limit: 0, - execution_mode: TxExecutionMode::VerifyExecute, - default_validation_computational_gas_limit: 0, - chain_id: Default::default(), - }; - let updates_manager = UpdatesManager::new(&l1_batch_env, &system_env); - output_handler - .handle_l2_block(&updates_manager) - .await - .unwrap(); - output_handler - .handle_l1_batch(Arc::new(updates_manager)) - .await - .unwrap(); - }); - self.tasks.push(join_handle); - Ok(()) - } - - async fn wait_for_batch( - &self, - l1_batch_number: L1BatchNumber, - timeout: Duration, - ) -> anyhow::Result<()> { - const RETRY_INTERVAL: Duration = Duration::from_millis(500); - - let max_tries = (timeout.as_secs_f64() / RETRY_INTERVAL.as_secs_f64()).ceil() as u64; - (|| async { - let current = self.io.read().await.current; - anyhow::ensure!( - current == l1_batch_number, - "Batch #{} has not been processed yet (current is #{})", - l1_batch_number, - current - ); - Ok(()) - }) - .retry( - &ConstantBuilder::default() - .with_delay(RETRY_INTERVAL) - .with_max_times(max_tries as usize), - ) - .await - } - - async fn wait_for_batch_progressively( - &self, - l1_batch_number: L1BatchNumber, - timeout: Duration, - ) -> anyhow::Result<()> { - const SLEEP_INTERVAL: Duration = Duration::from_millis(500); - - let mut current = self.io.read().await.current; - let max_tries = (timeout.as_secs_f64() / SLEEP_INTERVAL.as_secs_f64()).ceil() as u64; - let mut try_num = 0; - loop { - tokio::time::sleep(SLEEP_INTERVAL).await; - try_num += 1; - if try_num >= max_tries { - anyhow::bail!("Timeout"); - } - let new_current = self.io.read().await.current; - // Ensure we did not go back in latest processed batch - if new_current < current { - anyhow::bail!( - "Latest processed batch regressed to #{} back from #{}", - new_current, - current - ); - } - current = new_current; - if current >= l1_batch_number { - return Ok(()); - } - } - } - - async fn stop_and_wait_for_all_tasks(self) -> anyhow::Result<()> { - self.stop_sender.send(true)?; - futures::future::join_all(self.tasks).await; - Ok(()) - } - } - - #[tokio::test(flavor = "multi_thread", worker_threads = 10)] - async fn monotonically_progress_processed_batches() -> anyhow::Result<()> { - let pool = ConnectionPool::::test_pool().await; - let io = Arc::new(RwLock::new(IoMock { - current: 0.into(), - max: 10, - })); - // Distribute progressively higher delays for higher batches so that we can observe - // each batch being marked as processed. In other words, batch 1 would be marked as processed, - // then there will be a minimum 1 sec of delay (more in <10 thread environments), then batch - // 2 would be marked as processed etc. - let delays = (1..10) - .map(|i| (L1BatchNumber(i), Duration::from_secs(i as u64))) - .collect(); - let mut tester = OutputHandlerTester::new(io.clone(), pool, delays); - for i in 1..10 { - tester.spawn_test_task(i.into()).await?; - } - assert_eq!(io.read().await.current, L1BatchNumber(0)); - for i in 1..10 { - tester - .wait_for_batch(i.into(), Duration::from_secs(10)) - .await?; - } - tester.stop_and_wait_for_all_tasks().await?; - assert_eq!(io.read().await.current, L1BatchNumber(9)); - Ok(()) - } - - #[tokio::test(flavor = "multi_thread", worker_threads = 10)] - async fn do_not_progress_with_gaps() -> anyhow::Result<()> { - let pool = ConnectionPool::::test_pool().await; - let io = Arc::new(RwLock::new(IoMock { - current: 0.into(), - max: 10, - })); - // Distribute progressively lower delays for higher batches so that we can observe last - // processed batch not move until the first batch (with longest delay) is processed. - let delays = (1..10) - .map(|i| (L1BatchNumber(i), Duration::from_secs(10 - i as u64))) - .collect(); - let mut tester = OutputHandlerTester::new(io.clone(), pool, delays); - for i in 1..10 { - tester.spawn_test_task(i.into()).await?; - } - assert_eq!(io.read().await.current, L1BatchNumber(0)); - tester - .wait_for_batch_progressively(L1BatchNumber(9), Duration::from_secs(60)) - .await?; - tester.stop_and_wait_for_all_tasks().await?; - assert_eq!(io.read().await.current, L1BatchNumber(9)); - Ok(()) - } -} diff --git a/core/node/vm_runner/src/process.rs b/core/node/vm_runner/src/process.rs new file mode 100644 index 000000000000..8fafc715c59f --- /dev/null +++ b/core/node/vm_runner/src/process.rs @@ -0,0 +1,188 @@ +use std::{sync::Arc, time::Duration}; + +use anyhow::Context; +use multivm::interface::L2BlockEnv; +use tokio::{sync::watch, task::JoinHandle}; +use zksync_dal::{ConnectionPool, Core}; +use zksync_state_keeper::{ + BatchExecutor, BatchExecutorHandle, ExecutionMetricsForCriteria, L2BlockParams, + StateKeeperOutputHandler, TxExecutionResult, UpdatesManager, +}; +use zksync_types::{block::L2BlockExecutionData, L1BatchNumber}; + +use crate::{storage::StorageLoader, OutputHandlerFactory, VmRunnerIo}; + +/// VM runner represents a logic layer of L1 batch / L2 block processing flow akin to that of state +/// keeper. The difference is that VM runner is designed to be run on batches/blocks that have +/// already been processed by state keeper but still require some extra handling as regulated by +/// [`OutputHandlerFactory`]. +/// +/// It's responsible for taking unprocessed data from the [`VmRunnerIo`], feeding it into +/// [`BatchExecutor`] and calling [`OutputHandlerFactory`] on the result of the execution (batch +/// execution state in the [`UpdatesManager`]). +/// +/// You can think of VM runner as a concurrent processor of a continuous stream of newly committed +/// batches/blocks. +#[derive(Debug)] +pub struct VmRunner { + pool: ConnectionPool, + io: Box, + loader: Arc, + output_handler_factory: Box, + batch_processor: Box, +} + +impl VmRunner { + /// Initializes VM runner with its constituents. In order to make VM runner concurrent each + /// parameter here needs to support concurrent execution mode. See + /// [`ConcurrentOutputHandlerFactory`], [`VmRunnerStorage`]. + /// + /// Caller is expected to provide a component-specific implementation of [`VmRunnerIo`] and + /// an underlying implementation of [`OutputHandlerFactory`]. + pub fn new( + pool: ConnectionPool, + io: Box, + loader: Arc, + output_handler_factory: Box, + batch_processor: Box, + ) -> Self { + Self { + pool, + io, + loader, + output_handler_factory, + batch_processor, + } + } + + async fn process_batch( + batch_executor: BatchExecutorHandle, + l2_blocks: Vec, + mut updates_manager: UpdatesManager, + mut output_handler: Box, + ) -> anyhow::Result<()> { + for (i, l2_block) in l2_blocks.into_iter().enumerate() { + if i > 0 { + // First L2 block in every batch is already preloaded + updates_manager.push_l2_block(L2BlockParams { + timestamp: l2_block.timestamp, + virtual_blocks: l2_block.virtual_blocks, + }); + batch_executor + .start_next_l2_block(L2BlockEnv::from_l2_block_data(&l2_block)) + .await; + } + for tx in l2_block.txs { + let exec_result = batch_executor.execute_tx(tx.clone()).await; + let TxExecutionResult::Success { + tx_result, + tx_metrics, + call_tracer_result, + compressed_bytecodes, + .. + } = exec_result + else { + anyhow::bail!("Unexpected non-successful transaction"); + }; + let ExecutionMetricsForCriteria { + l1_gas: tx_l1_gas_this_tx, + execution_metrics: tx_execution_metrics, + } = *tx_metrics; + updates_manager.extend_from_executed_transaction( + tx, + *tx_result, + compressed_bytecodes, + tx_l1_gas_this_tx, + tx_execution_metrics, + call_tracer_result, + ); + } + output_handler + .handle_l2_block(&updates_manager) + .await + .context("VM runner failed to handle L2 block")?; + } + batch_executor.finish_batch().await; + output_handler + .handle_l1_batch(Arc::new(updates_manager)) + .await + .context("VM runner failed to handle L1 batch")?; + Ok(()) + } + + /// Consumes VM runner to execute a loop that continuously pulls data from [`VmRunnerIo`] and + /// processes it. + pub async fn run(mut self, stop_receiver: &watch::Receiver) -> anyhow::Result<()> { + const SLEEP_INTERVAL: Duration = Duration::from_millis(50); + + // Join handles for asynchronous tasks that are being run in the background + let mut task_handles: Vec<(L1BatchNumber, JoinHandle>)> = Vec::new(); + let mut next_batch = self + .io + .latest_processed_batch(&mut self.pool.connection().await?) + .await? + + 1; + loop { + // Traverse all handles and filter out tasks that have been finished. Also propagates + // any panic/error that might have happened during the task's execution. + let mut retained_handles = Vec::new(); + for (l1_batch_number, handle) in task_handles { + if handle.is_finished() { + handle + .await + .with_context(|| format!("Processing batch #{} panicked", l1_batch_number))? + .with_context(|| format!("Failed to process batch #{}", l1_batch_number))?; + } else { + retained_handles.push((l1_batch_number, handle)); + } + } + task_handles = retained_handles; + + let last_ready_batch = self + .io + .last_ready_to_be_loaded_batch(&mut self.pool.connection().await?) + .await?; + if next_batch > last_ready_batch { + // Next batch is not ready to be processed yet + tokio::time::sleep(SLEEP_INTERVAL).await; + continue; + } + let Some(batch_data) = self.loader.load_batch(next_batch).await? else { + // Next batch has not been loaded yet + tokio::time::sleep(SLEEP_INTERVAL).await; + continue; + }; + let updates_manager = + UpdatesManager::new(&batch_data.l1_batch_env, &batch_data.system_env); + let Some(batch_executor) = self + .batch_processor + .init_batch( + self.loader.clone().upcast(), + batch_data.l1_batch_env, + batch_data.system_env, + stop_receiver, + ) + .await + else { + tracing::info!("VM runner was interrupted"); + break; + }; + let output_handler = self + .output_handler_factory + .create_handler(next_batch) + .await?; + + let handle = tokio::task::spawn(Self::process_batch( + batch_executor, + batch_data.l2_blocks, + updates_manager, + output_handler, + )); + task_handles.push((next_batch, handle)); + + next_batch += 1; + } + + Ok(()) + } +} diff --git a/core/node/vm_runner/src/storage.rs b/core/node/vm_runner/src/storage.rs index 03f1b6baa4fe..5ffd1d11e70d 100644 --- a/core/node/vm_runner/src/storage.rs +++ b/core/node/vm_runner/src/storage.rs @@ -1,7 +1,6 @@ use std::{ collections::{BTreeMap, HashMap}, fmt::Debug, - marker::PhantomData, sync::Arc, time::Duration, }; @@ -22,6 +21,30 @@ use zksync_types::{block::L2BlockExecutionData, L1BatchNumber, L2ChainId}; use crate::VmRunnerIo; +#[async_trait] +pub trait StorageLoader: ReadStorageFactory { + /// Loads next unprocessed L1 batch along with all transactions that VM runner needs to + /// re-execute. These are the transactions that are included in a sealed L2 block belonging + /// to a sealed L1 batch (with state keeper being the source of truth). The order of the + /// transactions is the same as it was when state keeper executed them. + /// + /// Can return `None` if the requested batch is not available yet. + /// + /// # Errors + /// + /// Propagates DB errors. + async fn load_batch( + &self, + l1_batch_number: L1BatchNumber, + ) -> anyhow::Result>; + + /// A workaround for Rust's limitations on upcasting coercion. See + /// https://github.com/rust-lang/rust/issues/65991. + /// + /// Should always be implementable as [`StorageLoader`] requires [`ReadStorageFactory`]. + fn upcast(self: Arc) -> Arc; +} + /// Data needed to execute an L1 batch. #[derive(Debug, Clone)] pub struct BatchExecuteData { @@ -54,7 +77,7 @@ pub struct VmRunnerStorage { l1_batch_params_provider: L1BatchParamsProvider, chain_id: L2ChainId, state: Arc>, - _marker: PhantomData, + io: Io, } #[derive(Debug)] @@ -71,7 +94,7 @@ impl State { } } -impl VmRunnerStorage { +impl VmRunnerStorage { /// Creates a new VM runner storage using provided Postgres pool and RocksDB path. pub async fn new( pool: ConnectionPool, @@ -79,7 +102,7 @@ impl VmRunnerStorage { io: Io, chain_id: L2ChainId, ) -> anyhow::Result<(Self, StorageSyncTask)> { - let mut conn = pool.connection_tagged(Io::name()).await?; + let mut conn = pool.connection_tagged(io.name()).await?; let l1_batch_params_provider = L1BatchParamsProvider::new(&mut conn) .await .context("Failed initializing L1 batch params provider")?; @@ -89,20 +112,28 @@ impl VmRunnerStorage { l1_batch_number: L1BatchNumber(0), storage: BTreeMap::new(), })); - let task = - StorageSyncTask::new(pool.clone(), chain_id, rocksdb_path, io, state.clone()).await?; + let task = StorageSyncTask::new( + pool.clone(), + chain_id, + rocksdb_path, + io.clone(), + state.clone(), + ) + .await?; Ok(( Self { pool, l1_batch_params_provider, chain_id, state, - _marker: PhantomData, + io, }, task, )) } +} +impl VmRunnerStorage { async fn access_storage_inner( &self, _stop_receiver: &watch::Receiver, @@ -143,24 +174,17 @@ impl VmRunnerStorage { }, ))) } +} - /// Loads next unprocessed L1 batch along with all transactions that VM runner needs to - /// re-execute. These are the transactions that are included in a sealed L2 block belonging - /// to a sealed L1 batch (with state keeper being the source of truth). The order of the - /// transactions is the same as it was when state keeper executed them. - /// - /// Can return `None` if there are no batches to be processed. - /// - /// # Errors - /// - /// Propagates DB errors. - pub async fn load_batch( +#[async_trait] +impl StorageLoader for VmRunnerStorage { + async fn load_batch( &self, l1_batch_number: L1BatchNumber, ) -> anyhow::Result> { let state = self.state.read().await; if state.rocksdb.is_none() { - let mut conn = self.pool.connection_tagged(Io::name()).await?; + let mut conn = self.pool.connection_tagged(self.io.name()).await?; return StorageSyncTask::::load_batch_execute_data( &mut conn, l1_batch_number, @@ -182,6 +206,10 @@ impl VmRunnerStorage { Some(batch_data) => Ok(Some(batch_data.execute_data.clone())), } } + + fn upcast(self: Arc) -> Arc { + self + } } #[async_trait] @@ -219,7 +247,7 @@ impl StorageSyncTask { io: Io, state: Arc>, ) -> anyhow::Result { - let mut conn = pool.connection_tagged(Io::name()).await?; + let mut conn = pool.connection_tagged(io.name()).await?; let l1_batch_params_provider = L1BatchParamsProvider::new(&mut conn) .await .context("Failed initializing L1 batch params provider")?; @@ -255,7 +283,7 @@ impl StorageSyncTask { tracing::info!("`StorageSyncTask` was interrupted"); return Ok(()); } - let mut conn = self.pool.connection_tagged(Io::name()).await?; + let mut conn = self.pool.connection_tagged(self.io.name()).await?; let latest_processed_batch = self.io.latest_processed_batch(&mut conn).await?; let rocksdb_builder = RocksdbStorageBuilder::from_rocksdb(rocksdb.clone()); if rocksdb_builder.l1_batch_number().await == Some(latest_processed_batch + 1) { diff --git a/core/node/vm_runner/src/tests/mod.rs b/core/node/vm_runner/src/tests/mod.rs index dbbc4089dffe..d0374e0d5fa0 100644 --- a/core/node/vm_runner/src/tests/mod.rs +++ b/core/node/vm_runner/src/tests/mod.rs @@ -1,40 +1,42 @@ use std::{collections::HashMap, ops, sync::Arc, time::Duration}; use async_trait::async_trait; -use backon::{ConstantBuilder, ExponentialBuilder, Retryable}; -use rand::Rng; -use tempfile::TempDir; -use tokio::{ - runtime::Handle, - sync::{watch, RwLock}, - task::JoinHandle, -}; +use rand::{prelude::SliceRandom, Rng}; +use tokio::sync::RwLock; use zksync_contracts::BaseSystemContractsHashes; use zksync_dal::{Connection, ConnectionPool, Core, CoreDal}; -use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; use zksync_node_test_utils::{ - create_l1_batch_metadata, create_l2_block, create_l2_transaction, execute_l2_transaction, + create_l1_batch_metadata, create_l2_block, execute_l2_transaction, l1_batch_metadata_to_commitment_artifacts, }; -use zksync_state::{PgOrRocksdbStorage, PostgresStorage, ReadStorage, ReadStorageFactory}; +use zksync_state_keeper::{StateKeeperOutputHandler, UpdatesManager}; +use zksync_test_account::Account; use zksync_types::{ - block::{BlockGasCount, L1BatchHeader}, - fee::TransactionExecutionMetrics, - AccountTreeId, L1BatchNumber, L2ChainId, ProtocolVersionId, StorageKey, StorageLog, - StorageLogKind, StorageValue, H160, H256, + block::{BlockGasCount, L1BatchHeader, L2BlockHasher}, + fee::{Fee, TransactionExecutionMetrics}, + get_intrinsic_constants, + l2::L2Tx, + utils::storage_key_for_standard_token_balance, + AccountTreeId, Address, Execute, L1BatchNumber, L2BlockNumber, ProtocolVersionId, StorageKey, + StorageLog, StorageLogKind, StorageValue, H160, H256, L2_BASE_TOKEN_ADDRESS, U256, }; +use zksync_utils::u256_to_h256; + +use super::{OutputHandlerFactory, VmRunnerIo}; -use super::{BatchExecuteData, VmRunnerIo, VmRunnerStorage}; +mod output_handler; +mod process; +mod storage; #[derive(Debug, Default)] struct IoMock { current: L1BatchNumber, - max: L1BatchNumber, + max: u32, } #[async_trait] impl VmRunnerIo for Arc> { - fn name() -> &'static str { + fn name(&self) -> &'static str { "io_mock" } @@ -49,116 +51,156 @@ impl VmRunnerIo for Arc> { &self, _conn: &mut Connection<'_, Core>, ) -> anyhow::Result { - Ok(self.read().await.max) + let io = self.read().await; + Ok(io.current + io.max) } async fn mark_l1_batch_as_completed( &self, _conn: &mut Connection<'_, Core>, - _l1_batch_number: L1BatchNumber, + l1_batch_number: L1BatchNumber, ) -> anyhow::Result<()> { + self.write().await.current = l1_batch_number; Ok(()) } } -#[derive(Debug)] -struct VmRunnerTester { - db_dir: TempDir, - pool: ConnectionPool, - tasks: Vec>, -} +mod wait { + use std::{sync::Arc, time::Duration}; -impl VmRunnerTester { - fn new(pool: ConnectionPool) -> Self { - Self { - db_dir: TempDir::new().unwrap(), - pool, - tasks: Vec::new(), - } - } + use backon::{ConstantBuilder, Retryable}; + use tokio::sync::RwLock; + use zksync_types::L1BatchNumber; - async fn create_storage( - &mut self, - io_mock: Arc>, - ) -> anyhow::Result>>> { - let (vm_runner_storage, task) = VmRunnerStorage::new( - self.pool.clone(), - self.db_dir.path().to_str().unwrap().to_owned(), - io_mock, - L2ChainId::from(270), - ) - .await?; - let handle = tokio::task::spawn(async move { - let (_stop_sender, stop_receiver) = watch::channel(false); - task.run(stop_receiver).await.unwrap() - }); - self.tasks.push(handle); - Ok(vm_runner_storage) - } -} + use crate::tests::IoMock; -impl VmRunnerStorage { - async fn load_batch_eventually( - &self, - number: L1BatchNumber, - ) -> anyhow::Result { - (|| async { - self.load_batch(number) - .await? - .ok_or_else(|| anyhow::anyhow!("Batch #{} is not available yet", number)) - }) - .retry(&ExponentialBuilder::default()) - .await - } + pub(super) async fn for_batch( + io: Arc>, + l1_batch_number: L1BatchNumber, + timeout: Duration, + ) -> anyhow::Result<()> { + const RETRY_INTERVAL: Duration = Duration::from_millis(500); - async fn access_storage_eventually( - &self, - stop_receiver: &watch::Receiver, - number: L1BatchNumber, - ) -> anyhow::Result> { + let max_tries = (timeout.as_secs_f64() / RETRY_INTERVAL.as_secs_f64()).ceil() as u64; (|| async { - self.access_storage(stop_receiver, number) - .await? - .ok_or_else(|| { - anyhow::anyhow!("Storage for batch #{} is not available yet", number) - }) + let current = io.read().await.current; + anyhow::ensure!( + current == l1_batch_number, + "Batch #{} has not been processed yet (current is #{})", + l1_batch_number, + current + ); + Ok(()) }) - .retry(&ExponentialBuilder::default()) + .retry( + &ConstantBuilder::default() + .with_delay(RETRY_INTERVAL) + .with_max_times(max_tries as usize), + ) .await } - async fn ensure_batch_unloads_eventually(&self, number: L1BatchNumber) -> anyhow::Result<()> { - (|| async { - Ok(anyhow::ensure!( - self.load_batch(number).await?.is_none(), - "Batch #{} is still available", - number - )) - }) - .retry(&ExponentialBuilder::default()) - .await + pub(super) async fn for_batch_progressively( + io: Arc>, + l1_batch_number: L1BatchNumber, + timeout: Duration, + ) -> anyhow::Result<()> { + const SLEEP_INTERVAL: Duration = Duration::from_millis(500); + + let mut current = io.read().await.current; + let max_tries = (timeout.as_secs_f64() / SLEEP_INTERVAL.as_secs_f64()).ceil() as u64; + let mut try_num = 0; + loop { + tokio::time::sleep(SLEEP_INTERVAL).await; + try_num += 1; + if try_num >= max_tries { + anyhow::bail!("Timeout"); + } + let new_current = io.read().await.current; + // Ensure we did not go back in latest processed batch + if new_current < current { + anyhow::bail!( + "Latest processed batch regressed to #{} back from #{}", + new_current, + current + ); + } + current = new_current; + if current >= l1_batch_number { + return Ok(()); + } + } } +} - async fn batch_stays_unloaded(&self, number: L1BatchNumber) -> bool { - (|| async { - self.load_batch(number) - .await? - .ok_or_else(|| anyhow::anyhow!("Batch #{} is not available yet", number)) - }) - .retry( - &ConstantBuilder::default() - .with_delay(Duration::from_millis(100)) - .with_max_times(3), - ) - .await - .is_err() +#[derive(Debug)] +struct TestOutputFactory { + delays: HashMap, +} + +#[async_trait] +impl OutputHandlerFactory for TestOutputFactory { + async fn create_handler( + &mut self, + l1_batch_number: L1BatchNumber, + ) -> anyhow::Result> { + let delay = self.delays.get(&l1_batch_number).copied(); + #[derive(Debug)] + struct TestOutputHandler { + delay: Option, + } + #[async_trait] + impl StateKeeperOutputHandler for TestOutputHandler { + async fn handle_l2_block( + &mut self, + _updates_manager: &UpdatesManager, + ) -> anyhow::Result<()> { + Ok(()) + } + + async fn handle_l1_batch( + &mut self, + _updates_manager: Arc, + ) -> anyhow::Result<()> { + if let Some(delay) = self.delay { + tokio::time::sleep(delay).await + } + Ok(()) + } + } + Ok(Box::new(TestOutputHandler { delay })) } } -async fn store_l2_blocks( +/// Creates an L2 transaction with randomized parameters. +pub fn create_l2_transaction( + account: &mut Account, + fee_per_gas: u64, + gas_per_pubdata: u64, +) -> L2Tx { + let fee = Fee { + gas_limit: (get_intrinsic_constants().l2_tx_intrinsic_gas * 10).into(), + max_fee_per_gas: fee_per_gas.into(), + max_priority_fee_per_gas: 0_u64.into(), + gas_per_pubdata_limit: gas_per_pubdata.into(), + }; + let tx = account.get_l2_tx_for_execute( + Execute { + contract_address: Address::random(), + calldata: vec![], + value: Default::default(), + factory_deps: None, + }, + Some(fee), + ); + L2Tx::try_from(tx).unwrap() +} + +async fn store_l1_batches( conn: &mut Connection<'_, Core>, numbers: ops::RangeInclusive, contract_hashes: BaseSystemContractsHashes, + accounts: &mut [Account], ) -> anyhow::Result> { let mut rng = rand::thread_rng(); let mut batches = Vec::new(); @@ -169,9 +211,20 @@ async fn store_l2_blocks( .map(|m| m.number) .unwrap_or_default() + 1; + let mut last_l2_block_hash = if l2_block_number == 1.into() { + // First L2 block ever has a special `prev_l2_block_hash` + L2BlockHasher::legacy_hash(L2BlockNumber(0)) + } else { + conn.blocks_dal() + .get_l2_block_header(l2_block_number - 1) + .await? + .unwrap() + .hash + }; for l1_batch_number in numbers { let l1_batch_number = L1BatchNumber(l1_batch_number); - let tx = create_l2_transaction(10, 100); + let account = accounts.choose_mut(&mut rng).unwrap(); + let tx = create_l2_transaction(account, 1000000, 100); conn.transactions_dal() .insert_transaction_l2(&tx, TransactionExecutionMetrics::default()) .await?; @@ -201,15 +254,25 @@ async fn store_l2_blocks( .insert_factory_deps(l2_block_number, &factory_deps) .await?; let mut new_l2_block = create_l2_block(l2_block_number.0); + + let mut digest = L2BlockHasher::new( + new_l2_block.number, + new_l2_block.timestamp, + last_l2_block_hash, + ); + digest.push_tx_hash(tx.hash()); + new_l2_block.hash = digest.finalize(ProtocolVersionId::latest()); + l2_block_number += 1; new_l2_block.base_system_contracts_hashes = contract_hashes; new_l2_block.l2_tx_count = 1; conn.blocks_dal().insert_l2_block(&new_l2_block).await?; - let tx_result = execute_l2_transaction(tx); + last_l2_block_hash = new_l2_block.hash; + let tx_result = execute_l2_transaction(tx.clone()); conn.transactions_dal() .mark_txs_as_executed_in_l2_block( new_l2_block.number, - &[tx_result], + &[tx_result.clone()], 1.into(), ProtocolVersionId::latest(), false, @@ -217,9 +280,17 @@ async fn store_l2_blocks( .await?; // Insert a fictive L2 block at the end of the batch - let fictive_l2_block = create_l2_block(l2_block_number.0); + let mut fictive_l2_block = create_l2_block(l2_block_number.0); + let mut digest = L2BlockHasher::new( + fictive_l2_block.number, + fictive_l2_block.timestamp, + last_l2_block_hash, + ); + digest.push_tx_hash(tx.hash()); + fictive_l2_block.hash = digest.finalize(ProtocolVersionId::latest()); l2_block_number += 1; conn.blocks_dal().insert_l2_block(&fictive_l2_block).await?; + last_l2_block_hash = fictive_l2_block.hash; let header = L1BatchHeader::new( l1_batch_number, @@ -238,6 +309,9 @@ async fn store_l2_blocks( conn.blocks_dal() .mark_l2_blocks_as_executed_in_l1_batch(l1_batch_number) .await?; + conn.transactions_dal() + .mark_txs_as_executed_in_l1_batch(l1_batch_number, &[tx_result]) + .await?; let metadata = create_l1_batch_metadata(l1_batch_number.0); conn.blocks_dal() @@ -255,241 +329,34 @@ async fn store_l2_blocks( Ok(batches) } -#[tokio::test] -async fn rerun_storage_on_existing_data() -> anyhow::Result<()> { - let connection_pool = ConnectionPool::::test_pool().await; - let mut conn = connection_pool.connection().await.unwrap(); - let genesis_params = GenesisParams::mock(); - insert_genesis_batch(&mut conn, &genesis_params) - .await - .unwrap(); - drop(conn); +async fn fund(pool: &ConnectionPool, accounts: &[Account]) { + let mut conn = pool.connection().await.unwrap(); - // Generate 10 batches worth of data and persist it in Postgres - let batches = store_l2_blocks( - &mut connection_pool.connection().await?, - 1u32..=10u32, - genesis_params.base_system_contracts().hashes(), - ) - .await?; + let eth_amount = U256::from(10).pow(U256::from(32)); //10^32 wei - let mut tester = VmRunnerTester::new(connection_pool.clone()); - let io_mock = Arc::new(RwLock::new(IoMock { - current: 0.into(), - max: 10.into(), - })); - let storage = tester.create_storage(io_mock.clone()).await?; - // Check that existing batches are returned in the exact same order with the exact same data - for batch in &batches { - let batch_data = storage.load_batch_eventually(batch.number).await?; - let mut conn = connection_pool.connection().await.unwrap(); - let (previous_batch_hash, _) = conn - .blocks_dal() - .get_l1_batch_state_root_and_timestamp(batch_data.l1_batch_env.number - 1) - .await? - .unwrap(); - assert_eq!( - batch_data.l1_batch_env.previous_batch_hash, - Some(previous_batch_hash) - ); - assert_eq!(batch_data.l1_batch_env.number, batch.number); - assert_eq!(batch_data.l1_batch_env.timestamp, batch.timestamp); - let (first_l2_block_number, _) = conn - .blocks_dal() - .get_l2_block_range_of_l1_batch(batch.number) - .await? - .unwrap(); - let previous_l2_block_header = conn - .blocks_dal() - .get_l2_block_header(first_l2_block_number - 1) - .await? - .unwrap(); - let l2_block_header = conn - .blocks_dal() - .get_l2_block_header(first_l2_block_number) - .await? - .unwrap(); - assert_eq!( - batch_data.l1_batch_env.first_l2_block.number, - l2_block_header.number.0 - ); - assert_eq!( - batch_data.l1_batch_env.first_l2_block.timestamp, - l2_block_header.timestamp + for account in accounts { + let key = storage_key_for_standard_token_balance( + AccountTreeId::new(L2_BASE_TOKEN_ADDRESS), + &account.address, ); - assert_eq!( - batch_data.l1_batch_env.first_l2_block.prev_block_hash, - previous_l2_block_header.hash - ); - let l2_blocks = conn - .transactions_dal() - .get_l2_blocks_to_execute_for_l1_batch(batch_data.l1_batch_env.number) - .await?; - assert_eq!(batch_data.l2_blocks, l2_blocks); - } - - // "Mark" these batches as processed - io_mock.write().await.current += batches.len() as u32; - - // All old batches should no longer be loadable - for batch in batches { - storage - .ensure_batch_unloads_eventually(batch.number) - .await?; - } - - Ok(()) -} - -#[tokio::test] -async fn continuously_load_new_batches() -> anyhow::Result<()> { - let connection_pool = ConnectionPool::::test_pool().await; - let mut conn = connection_pool.connection().await.unwrap(); - let genesis_params = GenesisParams::mock(); - insert_genesis_batch(&mut conn, &genesis_params) - .await - .unwrap(); - drop(conn); - - let mut tester = VmRunnerTester::new(connection_pool.clone()); - let io_mock = Arc::new(RwLock::new(IoMock::default())); - let storage = tester.create_storage(io_mock.clone()).await?; - // No batches available yet - assert!(storage.load_batch(L1BatchNumber(1)).await?.is_none()); - - // Generate one batch and persist it in Postgres - store_l2_blocks( - &mut connection_pool.connection().await?, - 1u32..=1u32, - genesis_params.base_system_contracts().hashes(), - ) - .await?; - io_mock.write().await.max += 1; + let value = u256_to_h256(eth_amount); + let storage_log = StorageLog::new_write_log(key, value); - // Load batch and mark it as processed - assert_eq!( - storage - .load_batch_eventually(L1BatchNumber(1)) - .await? - .l1_batch_env - .number, - L1BatchNumber(1) - ); - io_mock.write().await.current += 1; - - // No more batches after that - assert!(storage.batch_stays_unloaded(L1BatchNumber(2)).await); - - // Generate one more batch and persist it in Postgres - store_l2_blocks( - &mut connection_pool.connection().await?, - 2u32..=2u32, - genesis_params.base_system_contracts().hashes(), - ) - .await?; - io_mock.write().await.max += 1; - - // Load batch and mark it as processed - - assert_eq!( - storage - .load_batch_eventually(L1BatchNumber(2)) - .await? - .l1_batch_env - .number, - L1BatchNumber(2) - ); - io_mock.write().await.current += 1; - - // No more batches after that - assert!(storage.batch_stays_unloaded(L1BatchNumber(3)).await); - - Ok(()) -} - -#[tokio::test] -async fn access_vm_runner_storage() -> anyhow::Result<()> { - let connection_pool = ConnectionPool::::test_pool().await; - let mut conn = connection_pool.connection().await.unwrap(); - let genesis_params = GenesisParams::mock(); - insert_genesis_batch(&mut conn, &genesis_params) - .await - .unwrap(); - drop(conn); - - // Generate 10 batches worth of data and persist it in Postgres - let batch_range = 1u32..=10u32; - store_l2_blocks( - &mut connection_pool.connection().await?, - batch_range, - genesis_params.base_system_contracts().hashes(), - ) - .await?; - - let mut conn = connection_pool.connection().await?; - let storage_logs = conn - .storage_logs_dal() - .dump_all_storage_logs_for_tests() - .await; - let factory_deps = conn - .factory_deps_dal() - .dump_all_factory_deps_for_tests() - .await; - drop(conn); - - let (_sender, receiver) = watch::channel(false); - let mut tester = VmRunnerTester::new(connection_pool.clone()); - let io_mock = Arc::new(RwLock::new(IoMock { - current: 0.into(), - max: 10.into(), - })); - let rt_handle = Handle::current(); - let handle = tokio::task::spawn_blocking(move || { - let vm_runner_storage = - rt_handle.block_on(async { tester.create_storage(io_mock.clone()).await.unwrap() }); - for i in 1..=10 { - let mut conn = rt_handle.block_on(connection_pool.connection()).unwrap(); - let (_, last_l2_block_number) = rt_handle - .block_on( - conn.blocks_dal() - .get_l2_block_range_of_l1_batch(L1BatchNumber(i)), - )? + conn.storage_logs_dal() + .append_storage_logs(L2BlockNumber(0), &[(H256::zero(), vec![storage_log])]) + .await + .unwrap(); + if conn + .storage_logs_dedup_dal() + .filter_written_slots(&[storage_log.key.hashed_key()]) + .await + .unwrap() + .is_empty() + { + conn.storage_logs_dedup_dal() + .insert_initial_writes(L1BatchNumber(0), &[storage_log.key]) + .await .unwrap(); - let mut pg_storage = - PostgresStorage::new(rt_handle.clone(), conn, last_l2_block_number, true); - let mut vm_storage = rt_handle.block_on(async { - vm_runner_storage - .access_storage_eventually(&receiver, L1BatchNumber(i)) - .await - })?; - // Check that both storages have identical key-value pairs written in them - for storage_log in &storage_logs { - let storage_key = - StorageKey::new(AccountTreeId::new(storage_log.address), storage_log.key); - assert_eq!( - pg_storage.read_value(&storage_key), - vm_storage.read_value(&storage_key) - ); - assert_eq!( - pg_storage.get_enumeration_index(&storage_key), - vm_storage.get_enumeration_index(&storage_key) - ); - assert_eq!( - pg_storage.is_write_initial(&storage_key), - vm_storage.is_write_initial(&storage_key) - ); - } - for hash in factory_deps.keys() { - assert_eq!( - pg_storage.load_factory_dep(*hash), - vm_storage.load_factory_dep(*hash) - ); - } } - - anyhow::Ok(()) - }); - handle.await??; - - Ok(()) + } } diff --git a/core/node/vm_runner/src/tests/output_handler.rs b/core/node/vm_runner/src/tests/output_handler.rs new file mode 100644 index 000000000000..97ea59db63b0 --- /dev/null +++ b/core/node/vm_runner/src/tests/output_handler.rs @@ -0,0 +1,146 @@ +use std::{collections::HashMap, sync::Arc, time::Duration}; + +use multivm::interface::{L1BatchEnv, L2BlockEnv, SystemEnv, TxExecutionMode}; +use tokio::{ + sync::{watch, RwLock}, + task::JoinHandle, +}; +use zksync_contracts::{BaseSystemContracts, SystemContractCode}; +use zksync_dal::{ConnectionPool, Core}; +use zksync_state_keeper::UpdatesManager; +use zksync_types::L1BatchNumber; + +use crate::{ + tests::{wait, IoMock, TestOutputFactory}, + ConcurrentOutputHandlerFactory, OutputHandlerFactory, +}; + +struct OutputHandlerTester { + output_factory: ConcurrentOutputHandlerFactory>, TestOutputFactory>, + tasks: Vec>, + stop_sender: watch::Sender, +} + +impl OutputHandlerTester { + fn new( + io: Arc>, + pool: ConnectionPool, + delays: HashMap, + ) -> Self { + let test_factory = TestOutputFactory { delays }; + let (output_factory, task) = ConcurrentOutputHandlerFactory::new(pool, io, test_factory); + let (stop_sender, stop_receiver) = watch::channel(false); + let join_handle = tokio::task::spawn(async move { task.run(stop_receiver).await.unwrap() }); + let tasks = vec![join_handle]; + Self { + output_factory, + tasks, + stop_sender, + } + } + + async fn spawn_test_task(&mut self, l1_batch_number: L1BatchNumber) -> anyhow::Result<()> { + let mut output_handler = self.output_factory.create_handler(l1_batch_number).await?; + let join_handle = tokio::task::spawn(async move { + let l1_batch_env = L1BatchEnv { + previous_batch_hash: None, + number: Default::default(), + timestamp: 0, + fee_input: Default::default(), + fee_account: Default::default(), + enforced_base_fee: None, + first_l2_block: L2BlockEnv { + number: 0, + timestamp: 0, + prev_block_hash: Default::default(), + max_virtual_blocks_to_create: 0, + }, + }; + let system_env = SystemEnv { + zk_porter_available: false, + version: Default::default(), + base_system_smart_contracts: BaseSystemContracts { + bootloader: SystemContractCode { + code: vec![], + hash: Default::default(), + }, + default_aa: SystemContractCode { + code: vec![], + hash: Default::default(), + }, + }, + bootloader_gas_limit: 0, + execution_mode: TxExecutionMode::VerifyExecute, + default_validation_computational_gas_limit: 0, + chain_id: Default::default(), + }; + let updates_manager = UpdatesManager::new(&l1_batch_env, &system_env); + output_handler + .handle_l2_block(&updates_manager) + .await + .unwrap(); + output_handler + .handle_l1_batch(Arc::new(updates_manager)) + .await + .unwrap(); + }); + self.tasks.push(join_handle); + Ok(()) + } + + async fn stop_and_wait_for_all_tasks(self) -> anyhow::Result<()> { + self.stop_sender.send(true)?; + futures::future::join_all(self.tasks).await; + Ok(()) + } +} + +#[tokio::test(flavor = "multi_thread", worker_threads = 10)] +async fn monotonically_progress_processed_batches() -> anyhow::Result<()> { + let pool = ConnectionPool::::test_pool().await; + let io = Arc::new(RwLock::new(IoMock { + current: 0.into(), + max: 10, + })); + // Distribute progressively higher delays for higher batches so that we can observe + // each batch being marked as processed. In other words, batch 1 would be marked as processed, + // then there will be a minimum 1 sec of delay (more in <10 thread environments), then batch + // 2 would be marked as processed etc. + let delays = (1..10) + .map(|i| (L1BatchNumber(i), Duration::from_secs(i as u64))) + .collect(); + let mut tester = OutputHandlerTester::new(io.clone(), pool, delays); + for i in 1..10 { + tester.spawn_test_task(i.into()).await?; + } + assert_eq!(io.read().await.current, L1BatchNumber(0)); + for i in 1..10 { + wait::for_batch(io.clone(), i.into(), Duration::from_secs(10)).await?; + } + tester.stop_and_wait_for_all_tasks().await?; + assert_eq!(io.read().await.current, L1BatchNumber(9)); + Ok(()) +} + +#[tokio::test(flavor = "multi_thread", worker_threads = 10)] +async fn do_not_progress_with_gaps() -> anyhow::Result<()> { + let pool = ConnectionPool::::test_pool().await; + let io = Arc::new(RwLock::new(IoMock { + current: 0.into(), + max: 10, + })); + // Distribute progressively lower delays for higher batches so that we can observe last + // processed batch not move until the first batch (with longest delay) is processed. + let delays = (1..10) + .map(|i| (L1BatchNumber(i), Duration::from_secs(10 - i as u64))) + .collect(); + let mut tester = OutputHandlerTester::new(io.clone(), pool, delays); + for i in 1..10 { + tester.spawn_test_task(i.into()).await?; + } + assert_eq!(io.read().await.current, L1BatchNumber(0)); + wait::for_batch_progressively(io.clone(), L1BatchNumber(9), Duration::from_secs(60)).await?; + tester.stop_and_wait_for_all_tasks().await?; + assert_eq!(io.read().await.current, L1BatchNumber(9)); + Ok(()) +} diff --git a/core/node/vm_runner/src/tests/process.rs b/core/node/vm_runner/src/tests/process.rs new file mode 100644 index 000000000000..664bdeebf855 --- /dev/null +++ b/core/node/vm_runner/src/tests/process.rs @@ -0,0 +1,83 @@ +use std::{collections::HashMap, sync::Arc, time::Duration}; + +use tempfile::TempDir; +use tokio::sync::{watch, RwLock}; +use zksync_dal::{ConnectionPool, Core}; +use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; +use zksync_state_keeper::MainBatchExecutor; +use zksync_test_account::Account; +use zksync_types::L2ChainId; + +use crate::{ + tests::{fund, store_l1_batches, wait, IoMock, TestOutputFactory}, + ConcurrentOutputHandlerFactory, VmRunner, VmRunnerStorage, +}; + +// Testing more than a one-batch scenario is pretty difficult as that requires storage to have +// completely valid state after each L2 block execution (current block number, hash, rolling txs +// hash etc written to the correct places). To achieve this we could run state keeper e2e but that +// is pretty difficult to set up. +// +// Instead, we rely on integration tests to verify the correctness of VM runner main process. +#[tokio::test] +async fn process_one_batch() -> anyhow::Result<()> { + let rocksdb_dir = TempDir::new()?; + let connection_pool = ConnectionPool::::test_pool().await; + let mut conn = connection_pool.connection().await.unwrap(); + let genesis_params = GenesisParams::mock(); + insert_genesis_batch(&mut conn, &genesis_params) + .await + .unwrap(); + let alice = Account::random(); + let bob = Account::random(); + let mut accounts = vec![alice, bob]; + fund(&connection_pool, &accounts).await; + + let batches = store_l1_batches( + &mut conn, + 1..=1, + genesis_params.base_system_contracts().hashes(), + &mut accounts, + ) + .await?; + drop(conn); + + let io = Arc::new(RwLock::new(IoMock { + current: 0.into(), + max: 1, + })); + let (storage, task) = VmRunnerStorage::new( + connection_pool.clone(), + rocksdb_dir.path().to_str().unwrap().to_owned(), + io.clone(), + L2ChainId::default(), + ) + .await?; + let (_, stop_receiver) = watch::channel(false); + let storage_stop_receiver = stop_receiver.clone(); + tokio::task::spawn(async move { task.run(storage_stop_receiver).await.unwrap() }); + let test_factory = TestOutputFactory { + delays: HashMap::new(), + }; + let (output_factory, task) = + ConcurrentOutputHandlerFactory::new(connection_pool.clone(), io.clone(), test_factory); + let output_stop_receiver = stop_receiver.clone(); + tokio::task::spawn(async move { task.run(output_stop_receiver).await.unwrap() }); + + let storage = Arc::new(storage); + let batch_executor = MainBatchExecutor::new(false, false); + let vm_runner = VmRunner::new( + connection_pool, + Box::new(io.clone()), + storage, + Box::new(output_factory), + Box::new(batch_executor), + ); + tokio::task::spawn(async move { vm_runner.run(&stop_receiver).await.unwrap() }); + + for batch in batches { + wait::for_batch(io.clone(), batch.number, Duration::from_secs(1)).await?; + } + + Ok(()) +} diff --git a/core/node/vm_runner/src/tests/storage.rs b/core/node/vm_runner/src/tests/storage.rs new file mode 100644 index 000000000000..afeaac8a8364 --- /dev/null +++ b/core/node/vm_runner/src/tests/storage.rs @@ -0,0 +1,369 @@ +use std::{sync::Arc, time::Duration}; + +use backon::{ConstantBuilder, ExponentialBuilder, Retryable}; +use tempfile::TempDir; +use tokio::{ + runtime::Handle, + sync::{watch, RwLock}, + task::JoinHandle, +}; +use zksync_dal::{ConnectionPool, Core, CoreDal}; +use zksync_node_genesis::{insert_genesis_batch, GenesisParams}; +use zksync_state::{PgOrRocksdbStorage, PostgresStorage, ReadStorage, ReadStorageFactory}; +use zksync_test_account::Account; +use zksync_types::{AccountTreeId, L1BatchNumber, L2ChainId, StorageKey}; + +use crate::{ + storage::StorageLoader, + tests::{fund, store_l1_batches, IoMock}, + BatchExecuteData, VmRunnerIo, VmRunnerStorage, +}; + +#[derive(Debug)] +struct StorageTester { + db_dir: TempDir, + pool: ConnectionPool, + tasks: Vec>, +} + +impl StorageTester { + fn new(pool: ConnectionPool) -> Self { + Self { + db_dir: TempDir::new().unwrap(), + pool, + tasks: Vec::new(), + } + } + + async fn create_storage( + &mut self, + io_mock: Arc>, + ) -> anyhow::Result>>> { + let (vm_runner_storage, task) = VmRunnerStorage::new( + self.pool.clone(), + self.db_dir.path().to_str().unwrap().to_owned(), + io_mock, + L2ChainId::default(), + ) + .await?; + let handle = tokio::task::spawn(async move { + let (_stop_sender, stop_receiver) = watch::channel(false); + task.run(stop_receiver).await.unwrap() + }); + self.tasks.push(handle); + Ok(vm_runner_storage) + } +} + +impl VmRunnerStorage { + async fn load_batch_eventually( + &self, + number: L1BatchNumber, + ) -> anyhow::Result { + (|| async { + self.load_batch(number) + .await? + .ok_or_else(|| anyhow::anyhow!("Batch #{} is not available yet", number)) + }) + .retry(&ExponentialBuilder::default()) + .await + } + + async fn access_storage_eventually( + &self, + stop_receiver: &watch::Receiver, + number: L1BatchNumber, + ) -> anyhow::Result> { + (|| async { + self.access_storage(stop_receiver, number) + .await? + .ok_or_else(|| { + anyhow::anyhow!("Storage for batch #{} is not available yet", number) + }) + }) + .retry(&ExponentialBuilder::default()) + .await + } + + async fn ensure_batch_unloads_eventually(&self, number: L1BatchNumber) -> anyhow::Result<()> { + (|| async { + Ok(anyhow::ensure!( + self.load_batch(number).await?.is_none(), + "Batch #{} is still available", + number + )) + }) + .retry(&ExponentialBuilder::default()) + .await + } + + async fn batch_stays_unloaded(&self, number: L1BatchNumber) -> bool { + (|| async { + self.load_batch(number) + .await? + .ok_or_else(|| anyhow::anyhow!("Batch #{} is not available yet", number)) + }) + .retry( + &ConstantBuilder::default() + .with_delay(Duration::from_millis(100)) + .with_max_times(3), + ) + .await + .is_err() + } +} + +#[tokio::test] +async fn rerun_storage_on_existing_data() -> anyhow::Result<()> { + let connection_pool = ConnectionPool::::test_pool().await; + let mut conn = connection_pool.connection().await.unwrap(); + let genesis_params = GenesisParams::mock(); + insert_genesis_batch(&mut conn, &genesis_params) + .await + .unwrap(); + drop(conn); + let alice = Account::random(); + let bob = Account::random(); + let mut accounts = vec![alice, bob]; + fund(&connection_pool, &accounts).await; + + // Generate 10 batches worth of data and persist it in Postgres + let batches = store_l1_batches( + &mut connection_pool.connection().await?, + 1..=10, + genesis_params.base_system_contracts().hashes(), + &mut accounts, + ) + .await?; + + let mut tester = StorageTester::new(connection_pool.clone()); + let io_mock = Arc::new(RwLock::new(IoMock { + current: 0.into(), + max: 10, + })); + let storage = tester.create_storage(io_mock.clone()).await?; + // Check that existing batches are returned in the exact same order with the exact same data + for batch in &batches { + let batch_data = storage.load_batch_eventually(batch.number).await?; + let mut conn = connection_pool.connection().await.unwrap(); + let (previous_batch_hash, _) = conn + .blocks_dal() + .get_l1_batch_state_root_and_timestamp(batch_data.l1_batch_env.number - 1) + .await? + .unwrap(); + assert_eq!( + batch_data.l1_batch_env.previous_batch_hash, + Some(previous_batch_hash) + ); + assert_eq!(batch_data.l1_batch_env.number, batch.number); + assert_eq!(batch_data.l1_batch_env.timestamp, batch.timestamp); + let (first_l2_block_number, _) = conn + .blocks_dal() + .get_l2_block_range_of_l1_batch(batch.number) + .await? + .unwrap(); + let previous_l2_block_header = conn + .blocks_dal() + .get_l2_block_header(first_l2_block_number - 1) + .await? + .unwrap(); + let l2_block_header = conn + .blocks_dal() + .get_l2_block_header(first_l2_block_number) + .await? + .unwrap(); + assert_eq!( + batch_data.l1_batch_env.first_l2_block.number, + l2_block_header.number.0 + ); + assert_eq!( + batch_data.l1_batch_env.first_l2_block.timestamp, + l2_block_header.timestamp + ); + assert_eq!( + batch_data.l1_batch_env.first_l2_block.prev_block_hash, + previous_l2_block_header.hash + ); + let l2_blocks = conn + .transactions_dal() + .get_l2_blocks_to_execute_for_l1_batch(batch_data.l1_batch_env.number) + .await?; + assert_eq!(batch_data.l2_blocks, l2_blocks); + } + + // "Mark" these batches as processed + io_mock.write().await.current += batches.len() as u32; + + // All old batches should no longer be loadable + for batch in batches { + storage + .ensure_batch_unloads_eventually(batch.number) + .await?; + } + + Ok(()) +} + +#[tokio::test] +async fn continuously_load_new_batches() -> anyhow::Result<()> { + let connection_pool = ConnectionPool::::test_pool().await; + let mut conn = connection_pool.connection().await.unwrap(); + let genesis_params = GenesisParams::mock(); + insert_genesis_batch(&mut conn, &genesis_params) + .await + .unwrap(); + drop(conn); + let alice = Account::random(); + let bob = Account::random(); + let mut accounts = vec![alice, bob]; + fund(&connection_pool, &accounts).await; + + let mut tester = StorageTester::new(connection_pool.clone()); + let io_mock = Arc::new(RwLock::new(IoMock::default())); + let storage = tester.create_storage(io_mock.clone()).await?; + // No batches available yet + assert!(storage.load_batch(L1BatchNumber(1)).await?.is_none()); + + // Generate one batch and persist it in Postgres + store_l1_batches( + &mut connection_pool.connection().await?, + 1..=1, + genesis_params.base_system_contracts().hashes(), + &mut accounts, + ) + .await?; + io_mock.write().await.max += 1; + + // Load batch and mark it as processed + assert_eq!( + storage + .load_batch_eventually(L1BatchNumber(1)) + .await? + .l1_batch_env + .number, + L1BatchNumber(1) + ); + io_mock.write().await.current += 1; + + // No more batches after that + assert!(storage.batch_stays_unloaded(L1BatchNumber(2)).await); + + // Generate one more batch and persist it in Postgres + store_l1_batches( + &mut connection_pool.connection().await?, + 2..=2, + genesis_params.base_system_contracts().hashes(), + &mut accounts, + ) + .await?; + io_mock.write().await.max += 1; + + // Load batch and mark it as processed + + assert_eq!( + storage + .load_batch_eventually(L1BatchNumber(2)) + .await? + .l1_batch_env + .number, + L1BatchNumber(2) + ); + io_mock.write().await.current += 1; + + // No more batches after that + assert!(storage.batch_stays_unloaded(L1BatchNumber(3)).await); + + Ok(()) +} + +#[tokio::test] +async fn access_vm_runner_storage() -> anyhow::Result<()> { + let connection_pool = ConnectionPool::::test_pool().await; + let mut conn = connection_pool.connection().await.unwrap(); + let genesis_params = GenesisParams::mock(); + insert_genesis_batch(&mut conn, &genesis_params) + .await + .unwrap(); + drop(conn); + let alice = Account::random(); + let bob = Account::random(); + let mut accounts = vec![alice, bob]; + fund(&connection_pool, &accounts).await; + + // Generate 10 batches worth of data and persist it in Postgres + let batch_range = 1..=10; + store_l1_batches( + &mut connection_pool.connection().await?, + batch_range, + genesis_params.base_system_contracts().hashes(), + &mut accounts, + ) + .await?; + + let mut conn = connection_pool.connection().await?; + let storage_logs = conn + .storage_logs_dal() + .dump_all_storage_logs_for_tests() + .await; + let factory_deps = conn + .factory_deps_dal() + .dump_all_factory_deps_for_tests() + .await; + drop(conn); + + let (_sender, receiver) = watch::channel(false); + let mut tester = StorageTester::new(connection_pool.clone()); + let io_mock = Arc::new(RwLock::new(IoMock { + current: 0.into(), + max: 10, + })); + let rt_handle = Handle::current(); + let handle = tokio::task::spawn_blocking(move || { + let vm_runner_storage = + rt_handle.block_on(async { tester.create_storage(io_mock.clone()).await.unwrap() }); + for i in 1..=10 { + let mut conn = rt_handle.block_on(connection_pool.connection()).unwrap(); + let (_, last_l2_block_number) = rt_handle + .block_on( + conn.blocks_dal() + .get_l2_block_range_of_l1_batch(L1BatchNumber(i)), + )? + .unwrap(); + let mut pg_storage = + PostgresStorage::new(rt_handle.clone(), conn, last_l2_block_number, true); + let mut vm_storage = rt_handle.block_on(async { + vm_runner_storage + .access_storage_eventually(&receiver, L1BatchNumber(i)) + .await + })?; + // Check that both storages have identical key-value pairs written in them + for storage_log in &storage_logs { + let storage_key = + StorageKey::new(AccountTreeId::new(storage_log.address), storage_log.key); + assert_eq!( + pg_storage.read_value(&storage_key), + vm_storage.read_value(&storage_key) + ); + assert_eq!( + pg_storage.get_enumeration_index(&storage_key), + vm_storage.get_enumeration_index(&storage_key) + ); + assert_eq!( + pg_storage.is_write_initial(&storage_key), + vm_storage.is_write_initial(&storage_key) + ); + } + for hash in factory_deps.keys() { + assert_eq!( + pg_storage.load_factory_dep(*hash), + vm_storage.load_factory_dep(*hash) + ); + } + } + + anyhow::Ok(()) + }); + handle.await??; + + Ok(()) +} diff --git a/core/tests/loadnext/src/config.rs b/core/tests/loadnext/src/config.rs index c8487e4d595e..7f3e1e258305 100644 --- a/core/tests/loadnext/src/config.rs +++ b/core/tests/loadnext/src/config.rs @@ -4,6 +4,7 @@ use serde::Deserialize; use tokio::sync::Semaphore; use zksync_contracts::test_contracts::LoadnextContractExecutionParams; use zksync_types::{network::Network, Address, L2ChainId, H160}; +use zksync_utils::workspace_dir_or_current_dir; use crate::fs_utils::read_tokens; @@ -189,14 +190,8 @@ fn default_main_token() -> H160 { } fn default_test_contracts_path() -> PathBuf { - let test_contracts_path = { - let home = std::env::var("ZKSYNC_HOME").unwrap(); - let path = PathBuf::from(&home); - path.join("etc/contracts-test-data") - }; - + let test_contracts_path = workspace_dir_or_current_dir().join("etc/contracts-test-data"); tracing::info!("Test contracts path: {}", test_contracts_path.display()); - test_contracts_path } @@ -346,3 +341,16 @@ impl RequestLimiters { } } } + +#[cfg(test)] +mod tests { + + use super::*; + use crate::fs_utils::loadnext_contract; + + #[test] + fn check_read_test_contract() { + let test_contracts_path = default_test_contracts_path(); + loadnext_contract(&test_contracts_path).unwrap(); + } +} diff --git a/core/tests/loadnext/src/executor.rs b/core/tests/loadnext/src/executor.rs index 080dd45dbb93..a7b1fa47c994 100644 --- a/core/tests/loadnext/src/executor.rs +++ b/core/tests/loadnext/src/executor.rs @@ -117,7 +117,7 @@ impl Executor { ); LOADTEST_METRICS .master_account_balance - .set(eth_balance.as_u128() as u64); + .set(eth_balance.as_u128() as f64); Ok(()) } diff --git a/core/tests/loadnext/src/fs_utils.rs b/core/tests/loadnext/src/fs_utils.rs index 9fee9916f916..8af9df8afee7 100644 --- a/core/tests/loadnext/src/fs_utils.rs +++ b/core/tests/loadnext/src/fs_utils.rs @@ -5,6 +5,7 @@ use std::{fs::File, io::BufReader, path::Path}; use serde::Deserialize; use zksync_types::{ethabi::Contract, network::Network, Address}; +use zksync_utils::workspace_dir_or_current_dir; /// A token stored in `etc/tokens/{network}.json` files. #[derive(Debug, Deserialize)] @@ -26,10 +27,8 @@ pub struct TestContract { } pub fn read_tokens(network: Network) -> anyhow::Result> { - let home = std::env::var("ZKSYNC_HOME")?; - let path = Path::new(&home); - let path = path.join(format!("etc/tokens/{network}.json")); - + let home = workspace_dir_or_current_dir(); + let path = home.join(format!("etc/tokens/{network}.json")); let file = File::open(path)?; let reader = BufReader::new(file); @@ -86,21 +85,3 @@ pub fn loadnext_contract(path: &Path) -> anyhow::Result { let path = path.join("artifacts-zk/contracts/loadnext/loadnext_contract.sol"); read_contract_dir(&path) } - -#[cfg(test)] -mod tests { - use std::path::PathBuf; - - use super::*; - - #[test] - fn check_read_test_contract() { - let test_contracts_path = { - let home = std::env::var("ZKSYNC_HOME").unwrap(); - let path = PathBuf::from(&home); - path.join("etc/contracts-test-data") - }; - - loadnext_contract(&test_contracts_path).unwrap(); - } -} diff --git a/core/tests/loadnext/src/metrics.rs b/core/tests/loadnext/src/metrics.rs index bebc1f0f4a31..2ea273225787 100644 --- a/core/tests/loadnext/src/metrics.rs +++ b/core/tests/loadnext/src/metrics.rs @@ -5,7 +5,7 @@ use vise::{Gauge, LabeledFamily, Metrics}; pub(crate) struct LoadtestMetrics { #[metrics(labels = ["label"])] pub tps: LabeledFamily>, - pub master_account_balance: Gauge, + pub master_account_balance: Gauge, } #[vise::register] diff --git a/core/tests/recovery-test/README.md b/core/tests/recovery-test/README.md new file mode 100644 index 000000000000..78833cc08057 --- /dev/null +++ b/core/tests/recovery-test/README.md @@ -0,0 +1,18 @@ +# Recovery Integration Tests + +These integration tests verify that a full node can initialize from an application snapshot or from genesis and then +sync with the main node. + +## Running locally + +The tests require that the main node is running; you can start it with a command like + +```shell +zk server &>server.log & +``` + +- [**Snapshot recovery test**](tests/snapshot-recovery.test.ts) can be run using + `yarn recovery-test snapshot-recovery-test`. It outputs logs to the files in the test directory: + `snapshot-creator.log` (snapshot creator logs) and `snapshot-recovery.log` (full node logs). +- [**Genesis recovery test**](tests/genesis-recovery.test.ts) can be run using + `yarn recovery-test genesis-recovery-test`. It outputs full node logs to `genesis-recovery.log` in the test directory. diff --git a/core/tests/snapshot-recovery-test/package.json b/core/tests/recovery-test/package.json similarity index 86% rename from core/tests/snapshot-recovery-test/package.json rename to core/tests/recovery-test/package.json index bdf6549d5194..adbbd1212696 100644 --- a/core/tests/snapshot-recovery-test/package.json +++ b/core/tests/recovery-test/package.json @@ -1,5 +1,5 @@ { - "name": "snapshot-recovery-test", + "name": "recovery-test", "version": "1.0.0", "license": "MIT", "mocha": { @@ -13,7 +13,8 @@ ] }, "scripts": { - "snapshot-recovery-test": "mocha tests/snapshot-recovery.test.ts" + "snapshot-recovery-test": "mocha tests/snapshot-recovery.test.ts", + "genesis-recovery-test": "mocha tests/genesis-recovery.test.ts" }, "devDependencies": { "@types/chai": "^4.2.21", diff --git a/core/tests/recovery-test/src/index.ts b/core/tests/recovery-test/src/index.ts new file mode 100644 index 000000000000..ca11a0d3b4cd --- /dev/null +++ b/core/tests/recovery-test/src/index.ts @@ -0,0 +1,259 @@ +/** + * Shared utils for recovery tests. + */ + +import fs, { FileHandle } from 'node:fs/promises'; +import fetch, { FetchError } from 'node-fetch'; +import { promisify } from 'node:util'; +import { ChildProcess, exec, spawn } from 'node:child_process'; +import * as zksync from 'zksync-ethers'; +import { ethers } from 'ethers'; +import path from 'node:path'; +import { expect } from 'chai'; + +export interface Health { + readonly status: string; + readonly details?: T; +} + +export interface SnapshotRecoveryDetails { + readonly snapshot_l1_batch: number; + readonly snapshot_l2_block: number; + readonly factory_deps_recovered: boolean; + readonly tokens_recovered: boolean; + readonly storage_logs_chunks_left_to_process: number; +} + +export interface ConsistencyCheckerDetails { + readonly first_checked_batch?: number; + readonly last_checked_batch?: number; +} + +export interface ReorgDetectorDetails { + readonly last_correct_l1_batch?: number; + readonly last_correct_l2_block?: number; +} + +export interface TreeDetails { + readonly min_l1_batch_number?: number | null; + readonly next_l1_batch_number?: number; +} + +export interface DbPrunerDetails { + readonly last_soft_pruned_l1_batch?: number; + readonly last_hard_pruned_l1_batch?: number; +} + +export interface TreeDataFetcherDetails { + readonly last_updated_l1_batch?: number; +} + +export interface HealthCheckResponse { + readonly status: string; + readonly components: { + snapshot_recovery?: Health; + consistency_checker?: Health; + reorg_detector?: Health; + tree?: Health; + db_pruner?: Health; + tree_pruner?: Health<{}>; + tree_data_fetcher?: Health; + }; +} + +export async function sleep(millis: number) { + await new Promise((resolve) => setTimeout(resolve, millis)); +} + +export async function getExternalNodeHealth() { + const EXTERNAL_NODE_HEALTH_URL = 'http://127.0.0.1:3081/health'; + + try { + const response: HealthCheckResponse = await fetch(EXTERNAL_NODE_HEALTH_URL).then((response) => response.json()); + return response; + } catch (e) { + let displayedError = e; + if (e instanceof FetchError && e.code === 'ECONNREFUSED') { + displayedError = '(connection refused)'; // Don't spam logs with "connection refused" messages + } + console.log( + `Request to EN health check server failed: ${displayedError}. In CI, you can see more details ` + + 'in "Show * logs" steps' + ); + return null; + } +} + +export async function dropNodeDatabase(env: { [key: string]: string }) { + await executeNodeCommand(env, 'zk db reset'); +} + +export async function dropNodeStorage(env: { [key: string]: string }) { + await executeNodeCommand(env, 'zk clean --database'); +} + +async function executeNodeCommand(env: { [key: string]: string }, command: string) { + const childProcess = spawn(command, { + cwd: process.env.ZKSYNC_HOME!!, + stdio: 'inherit', + shell: true, + env + }); + try { + await waitForProcess(childProcess, true); + } finally { + childProcess.kill(); + } +} + +export async function executeCommandWithLogs(command: string, logsPath: string) { + const logs = await fs.open(logsPath, 'w'); + const childProcess = spawn(command, { + cwd: process.env.ZKSYNC_HOME!!, + stdio: [null, logs.fd, logs.fd], + shell: true + }); + try { + await waitForProcess(childProcess, true); + } finally { + childProcess.kill(); + await logs.close(); + } +} + +export enum NodeComponents { + STANDARD = 'all', + WITH_TREE_FETCHER = 'all,tree_fetcher', + WITH_TREE_FETCHER_AND_NO_TREE = 'core,api,tree_fetcher' +} + +function externalNodeArgs(components: NodeComponents = NodeComponents.STANDARD) { + const enableConsensus = process.env.ENABLE_CONSENSUS === 'true'; + const args = ['external-node', '--', `--components=${components}`]; + if (enableConsensus) { + args.push('--enable-consensus'); + } + return args; +} + +export class NodeProcess { + static async stopAll(signal: 'INT' | 'KILL' = 'INT') { + interface ChildProcessError extends Error { + readonly code: number | null; + } + + try { + await promisify(exec)(`killall -q -${signal} zksync_external_node`); + } catch (err) { + const typedErr = err as ChildProcessError; + if (typedErr.code === 1) { + // No matching processes were found; this is fine. + } else { + throw err; + } + } + } + + static async spawn( + env: { [key: string]: string }, + logsFile: FileHandle | string, + components: NodeComponents = NodeComponents.STANDARD + ) { + const logs = typeof logsFile === 'string' ? await fs.open(logsFile, 'w') : logsFile; + const childProcess = spawn('zk', externalNodeArgs(components), { + cwd: process.env.ZKSYNC_HOME!!, + stdio: [null, logs.fd, logs.fd], + shell: true, + env + }); + return new NodeProcess(childProcess, logs); + } + + private constructor(private childProcess: ChildProcess, readonly logs: FileHandle) {} + + exitCode() { + return this.childProcess.exitCode; + } + + async stopAndWait(signal: 'INT' | 'KILL' = 'INT') { + await NodeProcess.stopAll(signal); + await waitForProcess(this.childProcess, signal === 'INT'); + } +} + +async function waitForProcess(childProcess: ChildProcess, checkExitCode: boolean) { + await new Promise((resolve, reject) => { + childProcess.on('error', (error) => { + reject(error); + }); + childProcess.on('exit', (code) => { + if (!checkExitCode || code === 0) { + resolve(undefined); + } else { + reject(new Error(`Process exited with non-zero code: ${code}`)); + } + }); + }); +} + +/** + * Funded wallet wrapper that can be used to generate L1 batches. + */ +export class FundedWallet { + static async create(mainNode: zksync.Provider, eth: ethers.providers.Provider): Promise { + const testConfigPath = path.join(process.env.ZKSYNC_HOME!, `etc/test_config/constant/eth.json`); + const ethTestConfig = JSON.parse(await fs.readFile(testConfigPath, { encoding: 'utf-8' })); + const mnemonic = ethTestConfig.test_mnemonic as string; + const wallet = zksync.Wallet.fromMnemonic(mnemonic, "m/44'/60'/0'/0/0").connect(mainNode).connectToL1(eth); + return new FundedWallet(wallet); + } + + private constructor(private readonly wallet: zksync.Wallet) {} + + /** Ensure that this wallet is funded on L2, depositing funds from L1 if necessary. */ + async ensureIsFunded() { + const balance = await this.wallet.getBalance(); + const minExpectedBalance = ethers.utils.parseEther('0.001'); + if (balance.gte(minExpectedBalance)) { + console.log('Wallet has acceptable balance on L2', balance); + return; + } + + const l1Balance = await this.wallet.getBalanceL1(); + expect(l1Balance.gte(minExpectedBalance), 'L1 balance of funded wallet is too small').to.be.true; + + const baseTokenAddress = await this.wallet.getBaseToken(); + const isETHBasedChain = baseTokenAddress == zksync.utils.ETH_ADDRESS_IN_CONTRACTS; + const depositParams = { + token: isETHBasedChain ? zksync.utils.LEGACY_ETH_ADDRESS : baseTokenAddress, + amount: minExpectedBalance, + to: this.wallet.address, + approveBaseERC20: true, + approveERC20: true + }; + console.log('Depositing funds on L2', depositParams); + const depositTx = await this.wallet.deposit(depositParams); + await depositTx.waitFinalize(); + } + + /** Generates at least one L1 batch by transfering funds to itself. */ + async generateL1Batch(): Promise { + const transactionResponse = await this.wallet.transfer({ + to: this.wallet.address, + amount: 1, + token: zksync.utils.ETH_ADDRESS + }); + console.log('Generated a transaction from funded wallet', transactionResponse); + const receipt = await transactionResponse.wait(); + console.log('Got finalized transaction receipt', receipt); + + // Wait until an L1 batch with the transaction is sealed. + const pastL1BatchNumber = await this.wallet.provider.getL1BatchNumber(); + let newL1BatchNumber: number; + while ((newL1BatchNumber = await this.wallet.provider.getL1BatchNumber()) <= pastL1BatchNumber) { + await sleep(1000); + } + console.log(`Sealed L1 batch #${newL1BatchNumber}`); + return newL1BatchNumber; + } +} diff --git a/core/tests/recovery-test/tests/genesis-recovery.test.ts b/core/tests/recovery-test/tests/genesis-recovery.test.ts new file mode 100644 index 000000000000..ebcf2b5a7e89 --- /dev/null +++ b/core/tests/recovery-test/tests/genesis-recovery.test.ts @@ -0,0 +1,240 @@ +import { expect } from 'chai'; +import * as zksync from 'zksync-ethers'; +import { ethers } from 'ethers'; + +import { + NodeProcess, + dropNodeDatabase, + dropNodeStorage, + getExternalNodeHealth, + NodeComponents, + sleep, + FundedWallet +} from '../src'; + +/** + * Tests recovery of an external node from scratch. + * + * Assumptions: + * + * - Main node is run for the duration of the test. + * - "Rich wallet" 0x36615Cf349d7F6344891B1e7CA7C72883F5dc049 is funded on L1. This is always true if the environment + * was initialized via `zk init`. + * - `ZKSYNC_ENV` variable is not set (checked at the start of the test). For this reason, + * the test doesn't have a `zk` wrapper; it should be launched using `yarn`. + */ +describe('genesis recovery', () => { + /** Number of L1 batches for the node to process during each phase of the test. */ + const CATCH_UP_BATCH_COUNT = 3; + + const externalNodeEnvProfile = + 'ext-node' + + (process.env.DEPLOYMENT_MODE === 'Validium' ? '-validium' : '') + + (process.env.IN_DOCKER ? '-docker' : ''); + console.log('Using external node env profile', externalNodeEnvProfile); + let externalNodeEnv: { [key: string]: string } = { + ...process.env, + ZKSYNC_ENV: externalNodeEnvProfile, + EN_SNAPSHOTS_RECOVERY_ENABLED: 'false' + }; + + let mainNode: zksync.Provider; + let externalNode: zksync.Provider; + let externalNodeProcess: NodeProcess; + let externalNodeBatchNumber: number; + + before('prepare environment', async () => { + expect(process.env.ZKSYNC_ENV, '`ZKSYNC_ENV` should not be set to allow running both server and EN components') + .to.be.undefined; + mainNode = new zksync.Provider('http://127.0.0.1:3050'); + externalNode = new zksync.Provider('http://127.0.0.1:3060'); + await NodeProcess.stopAll('KILL'); + }); + + let fundedWallet: FundedWallet; + + before('create test wallet', async () => { + const ethRpcUrl = process.env.ETH_CLIENT_WEB3_URL ?? 'http://127.0.0.1:8545'; + console.log(`Using L1 RPC at ${ethRpcUrl}`); + const eth = new ethers.providers.JsonRpcProvider(ethRpcUrl); + fundedWallet = await FundedWallet.create(mainNode, eth); + }); + + after(async () => { + if (externalNodeProcess) { + await externalNodeProcess.stopAndWait('KILL'); + await externalNodeProcess.logs.close(); + } + }); + + step('ensure that wallet has L2 funds', async () => { + await fundedWallet.ensureIsFunded(); + }); + + step('generate new batches if necessary', async () => { + let pastL1BatchNumber = await mainNode.getL1BatchNumber(); + while (pastL1BatchNumber < CATCH_UP_BATCH_COUNT) { + pastL1BatchNumber = await fundedWallet.generateL1Batch(); + } + }); + + step('drop external node database', async () => { + await dropNodeDatabase(externalNodeEnv); + }); + + step('drop external node storage', async () => { + await dropNodeStorage(externalNodeEnv); + }); + + step('initialize external node w/o a tree', async () => { + externalNodeProcess = await NodeProcess.spawn( + externalNodeEnv, + 'genesis-recovery.log', + NodeComponents.WITH_TREE_FETCHER_AND_NO_TREE + ); + + const mainNodeBatchNumber = await mainNode.getL1BatchNumber(); + expect(mainNodeBatchNumber).to.be.greaterThanOrEqual(CATCH_UP_BATCH_COUNT); + console.log(`Catching up to L1 batch #${CATCH_UP_BATCH_COUNT}`); + + let reorgDetectorSucceeded = false; + let treeFetcherSucceeded = false; + let consistencyCheckerSucceeded = false; + + while (!treeFetcherSucceeded || !reorgDetectorSucceeded || !consistencyCheckerSucceeded) { + await sleep(1000); + const health = await getExternalNodeHealth(); + if (health === null) { + continue; + } + + if (!treeFetcherSucceeded) { + const status = health.components.tree_data_fetcher?.status; + const details = health.components.tree_data_fetcher?.details; + if (status === 'ready' && details !== undefined && details.last_updated_l1_batch !== undefined) { + console.log('Received tree health details', details); + treeFetcherSucceeded = details.last_updated_l1_batch >= CATCH_UP_BATCH_COUNT; + } + } + + if (!reorgDetectorSucceeded) { + const status = health.components.reorg_detector?.status; + expect(status).to.be.oneOf([undefined, 'not_ready', 'ready']); + const details = health.components.reorg_detector?.details; + if (status === 'ready' && details !== undefined) { + console.log('Received reorg detector health details', details); + if (details.last_correct_l1_batch !== undefined) { + reorgDetectorSucceeded = details.last_correct_l1_batch >= CATCH_UP_BATCH_COUNT; + } + } + } + + if (!consistencyCheckerSucceeded) { + const status = health.components.consistency_checker?.status; + expect(status).to.be.oneOf([undefined, 'not_ready', 'ready']); + const details = health.components.consistency_checker?.details; + if (status === 'ready' && details !== undefined) { + console.log('Received consistency checker health details', details); + if (details.first_checked_batch !== undefined && details.last_checked_batch !== undefined) { + expect(details.first_checked_batch).to.equal(1); + consistencyCheckerSucceeded = details.last_checked_batch >= CATCH_UP_BATCH_COUNT; + } + } + } + } + + // If `externalNodeProcess` fails early, we'll trip these checks. + expect(externalNodeProcess.exitCode()).to.be.null; + expect(treeFetcherSucceeded, 'tree fetching failed').to.be.true; + expect(reorgDetectorSucceeded, 'reorg detection check failed').to.be.true; + }); + + step('get EN batch number', async () => { + externalNodeBatchNumber = await externalNode.getL1BatchNumber(); + console.log(`L1 batch number on EN: ${externalNodeBatchNumber}`); + expect(externalNodeBatchNumber).to.be.greaterThanOrEqual(CATCH_UP_BATCH_COUNT); + }); + + step('stop EN', async () => { + await externalNodeProcess.stopAndWait(); + }); + + step('generate new batches for 2nd phase if necessary', async () => { + let pastL1BatchNumber = await mainNode.getL1BatchNumber(); + while (pastL1BatchNumber < externalNodeBatchNumber + CATCH_UP_BATCH_COUNT) { + pastL1BatchNumber = await fundedWallet.generateL1Batch(); + } + }); + + step('restart EN', async () => { + externalNodeProcess = await NodeProcess.spawn( + externalNodeEnv, + externalNodeProcess.logs, + NodeComponents.WITH_TREE_FETCHER + ); + + let isNodeReady = false; + while (!isNodeReady) { + await sleep(1000); + const health = await getExternalNodeHealth(); + if (health === null) { + continue; + } + console.log('Node health', health); + isNodeReady = health.status === 'ready'; + } + }); + + step('wait for tree to catch up', async () => { + const mainNodeBatchNumber = await mainNode.getL1BatchNumber(); + expect(mainNodeBatchNumber).to.be.greaterThanOrEqual(externalNodeBatchNumber + CATCH_UP_BATCH_COUNT); + const catchUpBatchNumber = Math.min(mainNodeBatchNumber, externalNodeBatchNumber + CATCH_UP_BATCH_COUNT); + console.log(`Catching up to L1 batch #${catchUpBatchNumber}`); + + let reorgDetectorSucceeded = false; + let treeSucceeded = false; + let consistencyCheckerSucceeded = false; + + while (!treeSucceeded || !reorgDetectorSucceeded || !consistencyCheckerSucceeded) { + await sleep(1000); + const health = await getExternalNodeHealth(); + if (health === null) { + continue; + } + + if (!treeSucceeded) { + const status = health.components.tree?.status; + const details = health.components.tree?.details; + if (status === 'ready' && details !== undefined && details.next_l1_batch_number !== undefined) { + console.log('Received tree health details', details); + expect(details.min_l1_batch_number).to.be.equal(0); + treeSucceeded = details.next_l1_batch_number > catchUpBatchNumber; + } + } + + if (!reorgDetectorSucceeded) { + const status = health.components.reorg_detector?.status; + expect(status).to.be.oneOf([undefined, 'not_ready', 'ready']); + const details = health.components.reorg_detector?.details; + if (status === 'ready' && details !== undefined) { + console.log('Received reorg detector health details', details); + if (details.last_correct_l1_batch !== undefined) { + reorgDetectorSucceeded = details.last_correct_l1_batch >= catchUpBatchNumber; + } + } + } + + if (!consistencyCheckerSucceeded) { + const status = health.components.consistency_checker?.status; + expect(status).to.be.oneOf([undefined, 'not_ready', 'ready']); + const details = health.components.consistency_checker?.details; + if (status === 'ready' && details !== undefined) { + console.log('Received consistency checker health details', details); + if (details.first_checked_batch !== undefined && details.last_checked_batch !== undefined) { + consistencyCheckerSucceeded = details.last_checked_batch >= catchUpBatchNumber; + } + } + } + } + }); +}); diff --git a/core/tests/snapshot-recovery-test/tests/snapshot-recovery.test.ts b/core/tests/recovery-test/tests/snapshot-recovery.test.ts similarity index 63% rename from core/tests/snapshot-recovery-test/tests/snapshot-recovery.test.ts rename to core/tests/recovery-test/tests/snapshot-recovery.test.ts index 58275e5b397a..47350921d5a1 100644 --- a/core/tests/snapshot-recovery-test/tests/snapshot-recovery.test.ts +++ b/core/tests/recovery-test/tests/snapshot-recovery.test.ts @@ -1,13 +1,22 @@ import { expect } from 'chai'; -import fetch, { FetchError } from 'node-fetch'; import * as protobuf from 'protobufjs'; import * as zlib from 'zlib'; -import fs, { FileHandle } from 'node:fs/promises'; -import { ChildProcess, spawn, exec } from 'node:child_process'; +import fs from 'node:fs/promises'; import path from 'node:path'; -import { promisify } from 'node:util'; +import { ethers } from 'ethers'; import * as zksync from 'zksync-ethers'; +import { + getExternalNodeHealth, + sleep, + NodeComponents, + NodeProcess, + dropNodeDatabase, + dropNodeStorage, + executeCommandWithLogs, + FundedWallet +} from '../src'; + interface AllSnapshotsResponse { readonly snapshotsL1BatchNumbers: number[]; } @@ -39,55 +48,14 @@ interface TokenInfo { readonly l2_address: string; } -interface Health { - readonly status: string; - readonly details?: T; -} - -interface SnapshotRecoveryDetails { - readonly snapshot_l1_batch: number; - readonly snapshot_l2_block: number; - readonly factory_deps_recovered: boolean; - readonly tokens_recovered: boolean; - readonly storage_logs_chunks_left_to_process: number; -} - -interface ConsistencyCheckerDetails { - readonly first_checked_batch?: number; - readonly last_checked_batch?: number; -} - -interface ReorgDetectorDetails { - readonly last_correct_l1_batch?: number; - readonly last_correct_l2_block?: number; -} - -interface TreeDetails { - readonly min_l1_batch_number?: number | null; -} - -interface DbPrunerDetails { - readonly last_soft_pruned_l1_batch?: number; - readonly last_hard_pruned_l1_batch?: number; -} - -interface HealthCheckResponse { - readonly components: { - snapshot_recovery?: Health; - consistency_checker?: Health; - reorg_detector?: Health; - tree?: Health; - db_pruner?: Health; - tree_pruner?: Health<{}>; - }; -} - /** * Tests snapshot recovery and node state pruning. * * Assumptions: * * - Main node is run for the duration of the test. + * - "Rich wallet" 0x36615Cf349d7F6344891B1e7CA7C72883F5dc049 is funded on L1. This is always true if the environment + * was initialized via `zk init`. * - `ZKSYNC_ENV` variable is not set (checked at the start of the test). For this reason, * the test doesn't have a `zk` wrapper; it should be launched using `yarn`. */ @@ -98,6 +66,9 @@ describe('snapshot recovery', () => { const homeDir = process.env.ZKSYNC_HOME!!; + const disableTreeDuringPruning = process.env.DISABLE_TREE_DURING_PRUNING === 'true'; + console.log(`Tree is ${disableTreeDuringPruning ? 'disabled' : 'enabled'} during pruning`); + const externalNodeEnvProfile = 'ext-node' + (process.env.DEPLOYMENT_MODE === 'Validium' ? '-validium' : '') + @@ -112,31 +83,29 @@ describe('snapshot recovery', () => { let snapshotMetadata: GetSnapshotResponse; let mainNode: zksync.Provider; let externalNode: zksync.Provider; - let externalNodeLogs: FileHandle; - let externalNodeProcess: ChildProcess; + let externalNodeProcess: NodeProcess; - let fundedWallet: zksync.Wallet; + let fundedWallet: FundedWallet; before('prepare environment', async () => { expect(process.env.ZKSYNC_ENV, '`ZKSYNC_ENV` should not be set to allow running both server and EN components') .to.be.undefined; mainNode = new zksync.Provider('http://127.0.0.1:3050'); externalNode = new zksync.Provider('http://127.0.0.1:3060'); - await killExternalNode(); + await NodeProcess.stopAll('KILL'); }); before('create test wallet', async () => { - const testConfigPath = path.join(process.env.ZKSYNC_HOME!, `etc/test_config/constant/eth.json`); - const ethTestConfig = JSON.parse(await fs.readFile(testConfigPath, { encoding: 'utf-8' })); - const mnemonic = ethTestConfig.test_mnemonic as string; - fundedWallet = zksync.Wallet.fromMnemonic(mnemonic, "m/44'/60'/0'/0/0").connect(mainNode); + const ethRpcUrl = process.env.ETH_CLIENT_WEB3_URL ?? 'http://127.0.0.1:8545'; + console.log(`Using L1 RPC at ${ethRpcUrl}`); + const eth = new ethers.providers.JsonRpcProvider(ethRpcUrl); + fundedWallet = await FundedWallet.create(mainNode, eth); }); after(async () => { if (externalNodeProcess) { - externalNodeProcess.kill(); - await killExternalNode(); - await externalNodeLogs.close(); + await externalNodeProcess.stopAndWait('KILL'); + await externalNodeProcess.logs.close(); } }); @@ -156,18 +125,7 @@ describe('snapshot recovery', () => { } step('create snapshot', async () => { - const logs = await fs.open('snapshot-creator.log', 'w'); - const childProcess = spawn('zk run snapshots-creator', { - cwd: homeDir, - stdio: [null, logs.fd, logs.fd], - shell: true - }); - try { - await waitForProcess(childProcess); - } finally { - childProcess.kill(); - await logs.close(); - } + await executeCommandWithLogs('zk run snapshots-creator', 'snapshot-creator.log'); }); step('validate snapshot', async () => { @@ -217,41 +175,15 @@ describe('snapshot recovery', () => { }); step('drop external node database', async () => { - const childProcess = spawn('zk db reset', { - cwd: homeDir, - stdio: 'inherit', - shell: true, - env: externalNodeEnv - }); - try { - await waitForProcess(childProcess); - } finally { - childProcess.kill(); - } + await dropNodeDatabase(externalNodeEnv); }); step('drop external node storage', async () => { - const childProcess = spawn('zk clean --database', { - cwd: homeDir, - stdio: 'inherit', - shell: true, - env: externalNodeEnv - }); - try { - await waitForProcess(childProcess); - } finally { - childProcess.kill(); - } + await dropNodeStorage(externalNodeEnv); }); step('initialize external node', async () => { - externalNodeLogs = await fs.open('snapshot-recovery.log', 'w'); - externalNodeProcess = spawn('zk', externalNodeArgs(), { - cwd: homeDir, - stdio: [null, externalNodeLogs.fd, externalNodeLogs.fd], - shell: true, - env: externalNodeEnv - }); + externalNodeProcess = await NodeProcess.spawn(externalNodeEnv, 'snapshot-recovery.log'); let recoveryFinished = false; let consistencyCheckerSucceeded = false; @@ -314,7 +246,7 @@ describe('snapshot recovery', () => { } // If `externalNodeProcess` fails early, we'll trip these checks. - expect(externalNodeProcess.exitCode).to.be.null; + expect(externalNodeProcess.exitCode()).to.be.null; expect(consistencyCheckerSucceeded, 'consistency check failed').to.be.true; expect(reorgDetectorSucceeded, 'reorg detection check failed').to.be.true; }); @@ -348,9 +280,11 @@ describe('snapshot recovery', () => { step('restart EN', async () => { console.log('Stopping external node'); - await stopExternalNode(); - await waitForProcess(externalNodeProcess); + await externalNodeProcess.stopAndWait(); + const components = disableTreeDuringPruning + ? NodeComponents.WITH_TREE_FETCHER_AND_NO_TREE + : NodeComponents.WITH_TREE_FETCHER; const pruningParams = { EN_PRUNING_ENABLED: 'true', EN_PRUNING_REMOVAL_DELAY_SEC: '1', @@ -359,16 +293,12 @@ describe('snapshot recovery', () => { }; externalNodeEnv = { ...externalNodeEnv, ...pruningParams }; console.log('Starting EN with pruning params', pruningParams); - externalNodeProcess = spawn('zk', externalNodeArgs(), { - cwd: homeDir, - stdio: [null, externalNodeLogs.fd, externalNodeLogs.fd], - shell: true, - env: externalNodeEnv - }); + externalNodeProcess = await NodeProcess.spawn(externalNodeEnv, externalNodeProcess.logs, components); let isDbPrunerReady = false; - let isTreePrunerReady = false; - while (!isDbPrunerReady || !isTreePrunerReady) { + let isTreePrunerReady = disableTreeDuringPruning; // skip health checks if we don't run the tree + let isTreeFetcherReady = false; + while (!isDbPrunerReady || !isTreePrunerReady || !isTreeFetcherReady) { await sleep(1000); const health = await getExternalNodeHealth(); if (health === null) { @@ -387,31 +317,21 @@ describe('snapshot recovery', () => { expect(status).to.be.oneOf([undefined, 'not_ready', 'affected', 'ready']); isTreePrunerReady = status === 'ready'; } + if (!isTreeFetcherReady) { + console.log('Tree fetcher health', health.components.tree_data_fetcher); + const status = health.components.tree_data_fetcher?.status; + expect(status).to.be.oneOf([undefined, 'not_ready', 'affected', 'ready']); + isTreeFetcherReady = status === 'ready'; + } } }); // The logic below works fine if there is other transaction activity on the test network; we still // create *at least* `PRUNED_BATCH_COUNT + 1` L1 batches; thus, at least `PRUNED_BATCH_COUNT` of them // should be pruned eventually. - step(`generate ${PRUNED_BATCH_COUNT + 1} transactions`, async () => { - let pastL1BatchNumber = snapshotMetadata.l1BatchNumber; + step(`generate ${PRUNED_BATCH_COUNT + 1} L1 batches`, async () => { for (let i = 0; i < PRUNED_BATCH_COUNT + 1; i++) { - const transactionResponse = await fundedWallet.transfer({ - to: fundedWallet.address, - amount: 1, - token: zksync.utils.ETH_ADDRESS - }); - console.log('Generated a transaction from funded wallet', transactionResponse); - const receipt = await transactionResponse.wait(); - console.log('Got finalized transaction receipt', receipt); - - // Wait until an L1 batch number with the transaction is sealed. - let newL1BatchNumber: number; - while ((newL1BatchNumber = await mainNode.getL1BatchNumber()) <= pastL1BatchNumber) { - await sleep(1000); - } - console.log(`Sealed L1 batch #${newL1BatchNumber}`); - pastL1BatchNumber = newL1BatchNumber; + await fundedWallet.generateL1Batch(); } }); @@ -419,7 +339,7 @@ describe('snapshot recovery', () => { const expectedPrunedBatchNumber = snapshotMetadata.l1BatchNumber + PRUNED_BATCH_COUNT; console.log(`Waiting for L1 batch #${expectedPrunedBatchNumber} to be pruned`); let isDbPruned = false; - let isTreePruned = false; + let isTreePruned = disableTreeDuringPruning; while (!isDbPruned || !isTreePruned) { await sleep(1000); @@ -430,30 +350,19 @@ describe('snapshot recovery', () => { expect(dbPrunerHealth.status).to.be.equal('ready'); isDbPruned = dbPrunerHealth.details!.last_hard_pruned_l1_batch! >= expectedPrunedBatchNumber; - const treeHealth = health.components.tree!; - console.log('Tree health', treeHealth); - expect(treeHealth.status).to.be.equal('ready'); - const minTreeL1BatchNumber = treeHealth.details?.min_l1_batch_number; - // The batch number pruned from the tree is one less than `minTreeL1BatchNumber`. - isTreePruned = minTreeL1BatchNumber ? minTreeL1BatchNumber - 1 >= expectedPrunedBatchNumber : false; - } - }); -}); - -async function waitForProcess(childProcess: ChildProcess) { - await new Promise((resolve, reject) => { - childProcess.on('error', (error) => { - reject(error); - }); - childProcess.on('exit', (code) => { - if (code === 0) { - resolve(undefined); + if (disableTreeDuringPruning) { + expect(health.components.tree).to.be.undefined; } else { - reject(new Error(`Process exited with non-zero code: ${code}`)); + const treeHealth = health.components.tree!; + console.log('Tree health', treeHealth); + expect(treeHealth.status).to.be.equal('ready'); + const minTreeL1BatchNumber = treeHealth.details?.min_l1_batch_number; + // The batch number pruned from the tree is one less than `minTreeL1BatchNumber`. + isTreePruned = minTreeL1BatchNumber ? minTreeL1BatchNumber - 1 >= expectedPrunedBatchNumber : false; } - }); + } }); -} +}); async function decompressGzip(filePath: string): Promise { const readStream = (await fs.open(filePath)).createReadStream(); @@ -467,69 +376,3 @@ async function decompressGzip(filePath: string): Promise { readStream.pipe(gunzip); }); } - -async function sleep(millis: number) { - await new Promise((resolve) => setTimeout(resolve, millis)); -} - -async function getExternalNodeHealth() { - const EXTERNAL_NODE_HEALTH_URL = 'http://127.0.0.1:3081/health'; - - try { - const response: HealthCheckResponse = await fetch(EXTERNAL_NODE_HEALTH_URL).then((response) => response.json()); - return response; - } catch (e) { - let displayedError = e; - if (e instanceof FetchError && e.code === 'ECONNREFUSED') { - displayedError = '(connection refused)'; // Don't spam logs with "connection refused" messages - } - console.log( - `Request to EN health check server failed ${displayedError}, in CI you can see more details - in "Show snapshot-creator.log logs" and "Show contract_verifier.log logs" steps` - ); - return null; - } -} - -function externalNodeArgs() { - const enableConsensus = process.env.ENABLE_CONSENSUS === 'true'; - const args = ['external-node', '--']; - if (enableConsensus) { - args.push('--enable-consensus'); - } - return args; -} - -async function stopExternalNode() { - interface ChildProcessError extends Error { - readonly code: number | null; - } - - try { - await promisify(exec)('killall -q -INT zksync_external_node'); - } catch (err) { - const typedErr = err as ChildProcessError; - if (typedErr.code === 1) { - // No matching processes were found; this is fine. - } else { - throw err; - } - } -} - -async function killExternalNode() { - interface ChildProcessError extends Error { - readonly code: number | null; - } - - try { - await promisify(exec)('killall -q -KILL zksync_external_node'); - } catch (err) { - const typedErr = err as ChildProcessError; - if (typedErr.code === 1) { - // No matching processes were found; this is fine. - } else { - throw err; - } - } -} diff --git a/core/tests/snapshot-recovery-test/tsconfig.json b/core/tests/recovery-test/tsconfig.json similarity index 100% rename from core/tests/snapshot-recovery-test/tsconfig.json rename to core/tests/recovery-test/tsconfig.json diff --git a/core/tests/ts-integration/package.json b/core/tests/ts-integration/package.json index 4774864af8b0..1741f2b20557 100644 --- a/core/tests/ts-integration/package.json +++ b/core/tests/ts-integration/package.json @@ -31,6 +31,7 @@ "ts-node": "^10.1.0", "typescript": "^4.3.5", "zksync-ethers": "5.8.0-beta.5", - "elliptic": "^6.5.5" + "elliptic": "^6.5.5", + "yaml": "^2.4.2" } } diff --git a/core/tests/ts-integration/src/env.ts b/core/tests/ts-integration/src/env.ts index 363664694b3b..ada8a695e0aa 100644 --- a/core/tests/ts-integration/src/env.ts +++ b/core/tests/ts-integration/src/env.ts @@ -4,6 +4,7 @@ import * as ethers from 'ethers'; import * as zksync from 'zksync-ethers'; import { DataAvailabityMode, NodeMode, TestEnvironment } from './types'; import { Reporter } from './reporter'; +import * as yaml from 'yaml'; import { L2_BASE_TOKEN_ADDRESS } from 'zksync-ethers/build/utils'; /** @@ -14,16 +15,12 @@ import { L2_BASE_TOKEN_ADDRESS } from 'zksync-ethers/build/utils'; * This function is expected to be called *before* loading an environment via `loadTestEnvironment`, * because the latter expects server to be running and may throw otherwise. */ -export async function waitForServer() { +export async function waitForServer(l2NodeUrl: string) { const reporter = new Reporter(); // Server startup may take a lot of time on the staging. const attemptIntervalMs = 1000; const maxAttempts = 20 * 60; // 20 minutes - const l2NodeUrl = ensureVariable( - process.env.ZKSYNC_WEB3_API_URL || process.env.API_WEB3_JSON_RPC_HTTP_URL, - 'L2 node URL' - ); const l2Provider = new zksync.Provider(l2NodeUrl); reporter.startAction('Connecting to server'); @@ -45,25 +42,146 @@ export async function waitForServer() { throw new Error('Failed to wait for the server to start'); } +function getMainWalletPk(pathToHome: string, network: string): string { + if (network.toLowerCase() == 'localhost') { + const testConfigPath = path.join(pathToHome, `etc/test_config/constant`); + const ethTestConfig = JSON.parse(fs.readFileSync(`${testConfigPath}/eth.json`, { encoding: 'utf-8' })); + return ethers.Wallet.fromMnemonic(ethTestConfig.test_mnemonic as string, "m/44'/60'/0'/0/0").privateKey; + } else { + return ensureVariable(process.env.MASTER_WALLET_PK, 'Main wallet private key'); + } +} + +/* + Loads the environment for file based configs. + */ +async function loadTestEnvironmentFromFile(chain: string): Promise { + const pathToHome = path.join(__dirname, '../../../..'); + let ecosystem = loadEcosystem(pathToHome); + + let generalConfig = loadConfig(pathToHome, chain, 'general.yaml'); + let genesisConfig = loadConfig(pathToHome, chain, 'genesis.yaml'); + + const network = ecosystem.l1_network; + let mainWalletPK = getMainWalletPk(pathToHome, network); + const l2NodeUrl = generalConfig.api.web3_json_rpc.http_url; + + await waitForServer(l2NodeUrl); + + const l2Provider = new zksync.Provider(l2NodeUrl); + const baseTokenAddress = await l2Provider.getBaseTokenContractAddress(); + + const l1NodeUrl = ecosystem.l1_rpc_url; + const wsL2NodeUrl = generalConfig.api.web3_json_rpc.ws_url; + + const contractVerificationUrl = generalConfig.contract_verifier.url; + + const tokens = getTokensNew(pathToHome); + // wBTC is chosen because it has decimals different from ETH (8 instead of 18). + // Using this token will help us to detect decimals-related errors. + // but if it's not available, we'll use the first token from the list. + let token = tokens.tokens['wBTC']; + if (token === undefined) { + token = Object.values(tokens.tokens)[0]; + } + const weth = tokens.tokens['WETH']; + let baseToken; + + for (const key in tokens.tokens) { + const token = tokens.tokens[key]; + if (zksync.utils.isAddressEq(token.address, baseTokenAddress)) { + baseToken = token; + } + } + // `waitForServer` is expected to be executed. Otherwise this call may throw. + + const l2TokenAddress = await new zksync.Wallet( + mainWalletPK, + l2Provider, + ethers.getDefaultProvider(l1NodeUrl) + ).l2TokenAddress(token.address); + + const l2WethAddress = await new zksync.Wallet( + mainWalletPK, + l2Provider, + ethers.getDefaultProvider(l1NodeUrl) + ).l2TokenAddress(weth.address); + + const baseTokenAddressL2 = L2_BASE_TOKEN_ADDRESS; + const l2ChainId = parseInt(genesisConfig.l2_chain_id); + const l1BatchCommitDataGeneratorMode = genesisConfig.l1_batch_commit_data_generator_mode as DataAvailabityMode; + let minimalL2GasPrice = generalConfig.state_keeper.minimal_l2_gas_price; + // TODO add support for en + let nodeMode = NodeMode.Main; + + const validationComputationalGasLimit = parseInt(generalConfig.state_keeper.validation_computational_gas_limit); + // TODO set it properly + const priorityTxMaxGasLimit = 72000000; + const maxLogsLimit = parseInt(generalConfig.api.web3_json_rpc.req_entities_limit); + + return { + maxLogsLimit, + pathToHome, + priorityTxMaxGasLimit, + validationComputationalGasLimit, + nodeMode, + minimalL2GasPrice, + l1BatchCommitDataGeneratorMode, + l2ChainId, + network, + mainWalletPK, + l2NodeUrl, + l1NodeUrl, + wsL2NodeUrl, + contractVerificationUrl, + erc20Token: { + name: token.name, + symbol: token.symbol, + decimals: token.decimals, + l1Address: token.address, + l2Address: l2TokenAddress + }, + wethToken: { + name: weth.name, + symbol: weth.symbol, + decimals: weth.decimals, + l1Address: weth.address, + l2Address: l2WethAddress + }, + baseToken: { + name: baseToken?.name || token.name, + symbol: baseToken?.symbol || token.symbol, + decimals: baseToken?.decimals || token.decimals, + l1Address: baseToken?.address || token.address, + l2Address: baseTokenAddressL2 + } + }; +} + +export async function loadTestEnvironment(): Promise { + let chain = process.env.CHAIN_NAME; + + if (chain) { + return await loadTestEnvironmentFromFile(chain); + } + return await loadTestEnvironmentFromEnv(); +} + /** * Loads the test environment from the env variables. */ -export async function loadTestEnvironment(): Promise { +export async function loadTestEnvironmentFromEnv(): Promise { const network = process.env.CHAIN_ETH_NETWORK || 'localhost'; + const pathToHome = path.join(__dirname, '../../../../'); - let mainWalletPK; - if (network == 'localhost') { - const testConfigPath = path.join(process.env.ZKSYNC_HOME!, `etc/test_config/constant`); - const ethTestConfig = JSON.parse(fs.readFileSync(`${testConfigPath}/eth.json`, { encoding: 'utf-8' })); - mainWalletPK = ethers.Wallet.fromMnemonic(ethTestConfig.test_mnemonic as string, "m/44'/60'/0'/0/0").privateKey; - } else { - mainWalletPK = ensureVariable(process.env.MASTER_WALLET_PK, 'Main wallet private key'); - } + let mainWalletPK = getMainWalletPk(pathToHome, network); const l2NodeUrl = ensureVariable( process.env.ZKSYNC_WEB3_API_URL || process.env.API_WEB3_JSON_RPC_HTTP_URL, 'L2 node URL' ); + + await waitForServer(l2NodeUrl); const l2Provider = new zksync.Provider(l2NodeUrl); const baseTokenAddress = await l2Provider.getBaseTokenContractAddress(); @@ -76,7 +194,6 @@ export async function loadTestEnvironment(): Promise { ? process.env.CONTRACT_VERIFIER_URL! : ensureVariable(process.env.CONTRACT_VERIFIER_URL, 'Contract verification API'); - const pathToHome = path.join(__dirname, '../../../../'); const tokens = getTokens(pathToHome, process.env.CHAIN_ETH_NETWORK || 'localhost'); // wBTC is chosen because it has decimals different from ETH (8 instead of 18). // Using this token will help us to detect decimals-related errors. @@ -177,6 +294,14 @@ function ensureVariable(value: string | undefined, variableName: string): string return value; } +interface TokensDict { + [key: string]: L1Token; +} + +type Tokens = { + tokens: TokensDict; +}; + type L1Token = { name: string; symbol: string; @@ -195,3 +320,56 @@ function getTokens(pathToHome: string, network: string): L1Token[] { }) ); } + +function getTokensNew(pathToHome: string): Tokens { + const configPath = path.join(pathToHome, '/configs/erc20.yaml'); + if (!fs.existsSync(configPath)) { + throw Error('Tokens config not found'); + } + + return yaml.parse( + fs.readFileSync(configPath, { + encoding: 'utf-8' + }), + { + customTags + } + ); +} + +function loadEcosystem(pathToHome: string): any { + const configPath = path.join(pathToHome, '/ZkStack.yaml'); + if (!fs.existsSync(configPath)) { + return []; + } + return yaml.parse( + fs.readFileSync(configPath, { + encoding: 'utf-8' + }) + ); +} + +function loadConfig(pathToHome: string, chainName: string, config: string): any { + const configPath = path.join(pathToHome, `/chains/${chainName}/configs/${config}`); + if (!fs.existsSync(configPath)) { + return []; + } + return yaml.parse( + fs.readFileSync(configPath, { + encoding: 'utf-8' + }) + ); +} + +function customTags(tags: yaml.Tags): yaml.Tags { + for (const tag of tags) { + // @ts-ignore + if (tag.format === 'HEX') { + // @ts-ignore + tag.resolve = (str, _onError, _opt) => { + return str; + }; + } + } + return tags; +} diff --git a/core/tests/ts-integration/src/jest-setup/global-setup.ts b/core/tests/ts-integration/src/jest-setup/global-setup.ts index b0e2c8bf56dc..f86961eb1dc1 100644 --- a/core/tests/ts-integration/src/jest-setup/global-setup.ts +++ b/core/tests/ts-integration/src/jest-setup/global-setup.ts @@ -1,4 +1,4 @@ -import { TestContextOwner, loadTestEnvironment, waitForServer } from '../index'; +import { TestContextOwner, loadTestEnvironment } from '../index'; declare global { var __ZKSYNC_TEST_CONTEXT_OWNER__: TestContextOwner; @@ -18,7 +18,6 @@ async function performSetup(_globalConfig: any, _projectConfig: any) { // Before starting any actual logic, we need to ensure that the server is running (it may not // be the case, for example, right after deployment on stage). - await waitForServer(); const testEnvironment = await loadTestEnvironment(); const testContextOwner = new TestContextOwner(testEnvironment); diff --git a/docker/proof-fri-gpu-compressor/Dockerfile b/docker/proof-fri-gpu-compressor/Dockerfile new file mode 100644 index 000000000000..ead48f6af6bb --- /dev/null +++ b/docker/proof-fri-gpu-compressor/Dockerfile @@ -0,0 +1,48 @@ +# Will work locally only after prior universal setup key download +FROM nvidia/cuda:12.2.0-devel-ubuntu22.04 as builder + +ARG DEBIAN_FRONTEND=noninteractive + +ARG CUDA_ARCH=89 +ENV CUDAARCHS=${CUDA_ARCH} + +RUN apt-get update && apt-get install -y curl clang openssl libssl-dev gcc g++ git \ + pkg-config build-essential libclang-dev && \ + rm -rf /var/lib/apt/lists/* + +ENV RUSTUP_HOME=/usr/local/rustup \ + CARGO_HOME=/usr/local/cargo \ + PATH=/usr/local/cargo/bin:$PATH + +RUN curl https://sh.rustup.rs -sSf | bash -s -- -y && \ + rustup install nightly-2023-08-21 && \ + rustup default nightly-2023-08-21 + +RUN curl -Lo cmake-3.24.2-linux-x86_64.sh https://github.com/Kitware/CMake/releases/download/v3.24.2/cmake-3.24.2-linux-x86_64.sh && \ + chmod +x cmake-3.24.2-linux-x86_64.sh && \ + ./cmake-3.24.2-linux-x86_64.sh --skip-license --prefix=/usr/local + +WORKDIR /usr/src/zksync +COPY . . + +RUN cd prover && \ + git clone https://github.com/matter-labs/era-bellman-cuda.git --branch main bellman-cuda && \ + cmake -Bbellman-cuda/build -Sbellman-cuda/ -DCMAKE_BUILD_TYPE=Release && \ + cmake --build bellman-cuda/build/ + +RUN cd prover && BELLMAN_CUDA_DIR=$PWD/bellman-cuda cargo build --features "gpu" --release --bin zksync_proof_fri_compressor + +FROM nvidia/cuda:12.2.0-runtime-ubuntu22.04 + +RUN apt-get update && apt-get install -y curl libpq5 ca-certificates && rm -rf /var/lib/apt/lists/* + +# copy VK required for proof wrapping +COPY prover/vk_setup_data_generator_server_fri/data/ /prover/vk_setup_data_generator_server_fri/data/ + +COPY setup_2\^24.key /setup_2\^24.key + +ENV CRS_FILE=/setup_2\^24.key + +COPY --from=builder /usr/src/zksync/prover/target/release/zksync_proof_fri_compressor /usr/bin/ + +ENTRYPOINT ["zksync_proof_fri_compressor"] diff --git a/docker/zk-environment/Dockerfile b/docker/zk-environment/Dockerfile index 6690d317d2a3..1ed60f4b95f1 100644 --- a/docker/zk-environment/Dockerfile +++ b/docker/zk-environment/Dockerfile @@ -125,6 +125,10 @@ RUN wget -c https://sourceware.org/pub/valgrind/valgrind-3.20.0.tar.bz2 && \ cd valgrind-3.20.0 && ./configure && make && make install && \ cd ../ && rm -rf valgrind-3.20.0.tar.bz2 && rm -rf valgrind-3.20.0 +# Install foundry +RUN cargo install --git https://github.com/foundry-rs/foundry \ + --profile local --locked forge cast + # Setup the environment ENV ZKSYNC_HOME=/usr/src/zksync ENV PATH="${ZKSYNC_HOME}/bin:${PATH}" diff --git a/docs/guides/external-node/docker-compose-examples/mainnet-external-node-docker-compose.yml b/docs/guides/external-node/docker-compose-examples/mainnet-external-node-docker-compose.yml index 8cd329c9d40c..f99a0b2e491c 100644 --- a/docs/guides/external-node/docker-compose-examples/mainnet-external-node-docker-compose.yml +++ b/docs/guides/external-node/docker-compose-examples/mainnet-external-node-docker-compose.yml @@ -46,7 +46,7 @@ services: - POSTGRES_PASSWORD=notsecurepassword - PGPORT=5430 external-node: - image: "matterlabs/external-node:2.0-v24.0.0" + image: "matterlabs/external-node:2.0-v24.2.0" depends_on: postgres: condition: service_healthy @@ -76,7 +76,7 @@ services: EN_SNAPSHOTS_RECOVERY_ENABLED: "true" EN_SNAPSHOTS_OBJECT_STORE_BUCKET_BASE_URL: "zksync-era-mainnet-external-node-snapshots" EN_SNAPSHOTS_OBJECT_STORE_MODE: "GCSAnonymousReadOnly" - RUST_LOG: "zksync_core=info,zksync_core::metadata_calculator=debug,zksync_dal=info,zksync_db_connection=info,zksync_eth_client=info,zksync_merkle_tree=info,zksync_storage=info,zksync_state=debug,zksync_types=info,vm=info,zksync_external_node=info,zksync_utils=debug,zksync_snapshots_applier=info" + RUST_LOG: "warn,zksync=info,zksync_core::metadata_calculator=debug,zksync_state=debug,zksync_utils=debug,zksync_web3_decl::client=error" volumes: mainnet-postgres: {} diff --git a/docs/guides/external-node/docker-compose-examples/testnet-external-node-docker-compose.yml b/docs/guides/external-node/docker-compose-examples/testnet-external-node-docker-compose.yml index c1893a670f25..f0fc51be2796 100644 --- a/docs/guides/external-node/docker-compose-examples/testnet-external-node-docker-compose.yml +++ b/docs/guides/external-node/docker-compose-examples/testnet-external-node-docker-compose.yml @@ -46,7 +46,7 @@ services: - POSTGRES_PASSWORD=notsecurepassword - PGPORT=5430 external-node: - image: "matterlabs/external-node:2.0-v24.0.0" + image: "matterlabs/external-node:2.0-v24.2.0" depends_on: postgres: condition: service_healthy @@ -76,7 +76,7 @@ services: EN_SNAPSHOTS_RECOVERY_ENABLED: "true" EN_SNAPSHOTS_OBJECT_STORE_BUCKET_BASE_URL: "zksync-era-boojnet-external-node-snapshots" EN_SNAPSHOTS_OBJECT_STORE_MODE: "GCSAnonymousReadOnly" - RUST_LOG: "zksync_core=info,zksync_core::metadata_calculator=debug,zksync_dal=info,zksync_db_connection=info,zksync_eth_client=info,zksync_merkle_tree=info,zksync_storage=info,zksync_state=debug,zksync_types=info,vm=info,zksync_external_node=info,zksync_utils=debug,zksync_snapshots_applier=info" + RUST_LOG: "warn,zksync=info,zksync_core::metadata_calculator=debug,zksync_state=debug,zksync_utils=debug,zksync_web3_decl::client=error" volumes: testnet-postgres: {} diff --git a/docs/guides/setup-dev.md b/docs/guides/setup-dev.md index a27cdd3ea593..f096a2f8a270 100644 --- a/docs/guides/setup-dev.md +++ b/docs/guides/setup-dev.md @@ -27,6 +27,10 @@ cargo install sqlx-cli --version 0.7.3 sudo systemctl stop postgresql # Start docker. sudo systemctl start docker + +# Foundry +curl -L https://foundry.paradigm.xyz | bash +foundryup --branch master ``` ## Supported operating systems @@ -257,6 +261,11 @@ enable nix-ld. Go to the zksync folder and run `nix develop --impure`. After it finishes, you are in a shell that has all the dependencies. +## Foundry + +[Foundry](https://book.getfoundry.sh/getting-started/installation) can be utilized for deploying smart contracts. For +commands related to deployment, you can pass flags for Foundry integration. + ## Environment Edit the lines below and add them to your shell profile file (e.g. `~/.bash_profile`, `~/.zshrc`): diff --git a/etc/env/base/contracts.toml b/etc/env/base/contracts.toml index 1820c9e57c22..40563e3e9870 100644 --- a/etc/env/base/contracts.toml +++ b/etc/env/base/contracts.toml @@ -55,10 +55,10 @@ MAX_NUMBER_OF_HYPERCHAINS = 100 L1_SHARED_BRIDGE_PROXY_ADDR = "0xFC073319977e314F251EAE6ae6bE76B0B3BAeeCF" L2_SHARED_BRIDGE_ADDR = "0xFC073319977e314F251EAE6ae6bE76B0B3BAeeCF" L2_SHARED_BRIDGE_IMPL_ADDR = "0xFC073319977e314F251EAE6ae6bE76B0B3BAeeCF" -FRI_RECURSION_LEAF_LEVEL_VK_HASH = "0xffb19d007c67b9000b40b372e7a7a55a47d11c92588515598d6cad4052c75ebb" +FRI_RECURSION_LEAF_LEVEL_VK_HASH = "0xcc4ac1853353538a166f5c2dde2c24e7e6c461dce8e3dc47d81e9139e1719456" FRI_RECURSION_NODE_LEVEL_VK_HASH = "0xf520cd5b37e74e19fdb369c8d676a04dce8a19457497ac6686d2bb95d94109c8" FRI_RECURSION_SCHEDULER_LEVEL_VK_HASH = "0x712bb009b5d5dc81c79f827ca0abff87b43506a8efed6028a818911d4b1b521f" -SNARK_WRAPPER_VK_HASH = "0x1e2d8304351d4667f0e13b0c51b30538f4dc6ece2c457babd03a9f3a1ec523b3" +SNARK_WRAPPER_VK_HASH = "0xb45190a52235abe353afd606a9144728f807804f5282df9247e27c56e817ccd6" SHARED_BRIDGE_UPGRADE_STORAGE_SWITCH = 0 ERA_CHAIN_ID = 9 ERA_DIAMOND_PROXY_ADDR = "0x0000000000000000000000000000000000000000" diff --git a/etc/env/base/eth_sender.toml b/etc/env/base/eth_sender.toml index 902efeca1f9d..31fe626c87f2 100644 --- a/etc/env/base/eth_sender.toml +++ b/etc/env/base/eth_sender.toml @@ -46,8 +46,6 @@ max_single_tx_gas = 6000000 # Max acceptable fee for sending tx to L1 max_acceptable_priority_fee_in_gwei = 100000000000 -proof_loading_mode="FriProofFromGcs" - pubdata_sending_mode = "Blobs" [eth_sender.gas_adjuster] diff --git a/etc/env/base/fri_proof_compressor.toml b/etc/env/base/fri_proof_compressor.toml index bda943391f06..9d26fe876896 100644 --- a/etc/env/base/fri_proof_compressor.toml +++ b/etc/env/base/fri_proof_compressor.toml @@ -1,10 +1,10 @@ [fri_proof_compressor] -compression_mode=1 -prometheus_listener_port=3321 -prometheus_pushgateway_url="http://127.0.0.1:9091" -prometheus_push_interval_ms=100 -generation_timeout_in_secs=3600 -max_attempts=5 -universal_setup_path="../keys/setup/setup_2^26.key" -universal_setup_download_url="https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2^26.key" -verify_wrapper_proof=true +compression_mode = 1 +prometheus_listener_port = 3321 +prometheus_pushgateway_url = "http://127.0.0.1:9091" +prometheus_push_interval_ms = 100 +generation_timeout_in_secs = 3600 +max_attempts = 5 +universal_setup_path = "../keys/setup/setup_2^24.key" +universal_setup_download_url = "https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2^24.key" +verify_wrapper_proof = true diff --git a/etc/env/configs/ext-node.toml b/etc/env/configs/ext-node.toml index eef24cf60370..eb07aa387542 100644 --- a/etc/env/configs/ext-node.toml +++ b/etc/env/configs/ext-node.toml @@ -58,13 +58,21 @@ warn,\ zksync_consensus_bft=info,\ zksync_consensus_network=info,\ zksync_consensus_storage=info,\ +zksync_commitment_generator=info,\ zksync_core=debug,\ zksync_dal=info,\ zksync_db_connection=info,\ zksync_health_check=debug,\ zksync_eth_client=info,\ +zksync_state_keeper=info,\ +zksync_node_sync=info,\ zksync_storage=info,\ +zksync_metadata_calculator=info,\ zksync_merkle_tree=info,\ +zksync_node_api_server=info,\ +zksync_node_db_pruner=info,\ +zksync_reorg_detector=info,\ +zksync_consistency_checker=info,\ zksync_state=debug,\ zksync_utils=debug,\ zksync_types=info,\ diff --git a/etc/env/file_based/general.yaml b/etc/env/file_based/general.yaml index d31e694594de..d59da18d1266 100644 --- a/etc/env/file_based/general.yaml +++ b/etc/env/file_based/general.yaml @@ -131,7 +131,6 @@ eth: aggregated_proof_sizes: [ 1,4 ] max_aggregated_tx_gas: 4000000 max_acceptable_priority_fee_in_gwei: 100000000000 - proof_loading_mode: OLD_PROOF_FROM_DB pubdata_sending_mode: BLOBS gas_adjuster: default_priority_fee_per_gas: 1000000000 @@ -200,8 +199,8 @@ proof_compressor: prometheus_push_interval_ms: 100 generation_timeout_in_secs: 3600 max_attempts: 5 - universal_setup_path: keys/setup/setup_2^26.key - universal_setup_download_url: https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2^26.key + universal_setup_path: keys/setup/setup_2^24.key + universal_setup_download_url: https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2^24.key verify_wrapper_proof: true prover_group: group_0: diff --git a/infrastructure/zk/src/docker.ts b/infrastructure/zk/src/docker.ts index fc98e8ad02a3..6d0edf1f4cdf 100644 --- a/infrastructure/zk/src/docker.ts +++ b/infrastructure/zk/src/docker.ts @@ -15,6 +15,7 @@ const IMAGES = [ 'witness-vector-generator', 'prover-fri-gateway', 'proof-fri-compressor', + 'proof-fri-gpu-compressor', 'snapshots-creator', 'verified-sources-fetcher' ]; @@ -79,6 +80,7 @@ function defaultTagList(image: string, imageTagSha: string, imageTagShaTS: strin 'witness-vector-generator', 'prover-fri-gateway', 'proof-fri-compressor', + 'proof-fri-gpu-compressor', 'snapshots-creator' ].includes(image) ? ['latest', 'latest2.0', `2.0-${imageTagSha}`, `${imageTagSha}`, `2.0-${imageTagShaTS}`, `${imageTagShaTS}`] diff --git a/infrastructure/zk/src/prover_setup.ts b/infrastructure/zk/src/prover_setup.ts index 586844856a7c..361ae44b8fa0 100644 --- a/infrastructure/zk/src/prover_setup.ts +++ b/infrastructure/zk/src/prover_setup.ts @@ -23,7 +23,6 @@ export async function setupProver(proverType: ProverType) { if (proverType == ProverType.GPU || proverType == ProverType.CPU) { env.modify('PROVER_TYPE', proverType, process.env.ENV_FILE!); env.modify('ETH_SENDER_SENDER_PROOF_SENDING_MODE', 'OnlyRealProofs', process.env.ENV_FILE!); - env.modify('ETH_SENDER_SENDER_PROOF_LOADING_MODE', 'FriProofFromGcs', process.env.ENV_FILE!); env.modify('FRI_PROVER_GATEWAY_API_POLL_DURATION_SECS', '120', process.env.ENV_FILE!); await setupArtifactsMode(); if (!process.env.CI) { diff --git a/infrastructure/zk/src/server.ts b/infrastructure/zk/src/server.ts index 896cb97fe340..923097f5c604 100644 --- a/infrastructure/zk/src/server.ts +++ b/infrastructure/zk/src/server.ts @@ -6,12 +6,12 @@ import * as path from 'path'; import * as db from './database'; import * as env from './env'; -export async function server(rebuildTree: boolean, uring: boolean, components?: string) { +export async function server(rebuildTree: boolean, uring: boolean, components?: string, useNodeFramework?: boolean) { let options = ''; if (uring) { options += '--features=rocksdb/io-uring'; } - if (rebuildTree || components) { + if (rebuildTree || components || useNodeFramework) { options += ' --'; } if (rebuildTree) { @@ -21,6 +21,9 @@ export async function server(rebuildTree: boolean, uring: boolean, components?: if (components) { options += ` --components=${components}`; } + if (useNodeFramework) { + options += ' --use-node-framework'; + } await utils.spawn(`cargo run --bin zksync_server --release ${options}`); } @@ -79,12 +82,13 @@ export const serverCommand = new Command('server') .option('--uring', 'enables uring support for RocksDB') .option('--components ', 'comma-separated list of components to run') .option('--chain-name ', 'environment name') + .option('--use-node-framework', 'use node framework for server') .action(async (cmd: Command) => { cmd.chainName ? env.reload(cmd.chainName) : env.load(); if (cmd.genesis) { await genesisFromSources(); } else { - await server(cmd.rebuildTree, cmd.uring, cmd.components); + await server(cmd.rebuildTree, cmd.uring, cmd.components, cmd.useNodeFramework); } }); diff --git a/infrastructure/zk/src/status.ts b/infrastructure/zk/src/status.ts index 1ad437b85540..d2f1ca08f71f 100644 --- a/infrastructure/zk/src/status.ts +++ b/infrastructure/zk/src/status.ts @@ -190,11 +190,6 @@ export async function statusProver() { main_pool = new Pool({ connectionString: process.env.DATABASE_URL }); prover_pool = new Pool({ connectionString: process.env.DATABASE_PROVER_URL }); - if (process.env.ETH_SENDER_SENDER_PROOF_LOADING_MODE != 'FriProofFromGcs') { - console.log(`${redStart}Can only show status for FRI provers.${resetColor}`); - return; - } - // Fetch the first and most recent sealed batch numbers const stateKeeperStatus = ( await queryAndReturnRows(main_pool, 'select min(number), max(number) from l1_batches') diff --git a/package.json b/package.json index 2bf96f4716a4..cdbc8acee00e 100644 --- a/package.json +++ b/package.json @@ -14,7 +14,7 @@ "infrastructure/zk", "infrastructure/local-setup-preparation", "core/tests/revert-test", - "core/tests/snapshot-recovery-test", + "core/tests/recovery-test", "core/tests/upgrade-test", "core/tests/ts-integration", "infrastructure/protocol-upgrade" @@ -30,7 +30,7 @@ "l2-contracts": "yarn workspace l2-contracts", "revert-test": "yarn workspace revert-test", "upgrade-test": "yarn workspace upgrade-test", - "snapshot-recovery-test": "yarn workspace snapshot-recovery-test", + "recovery-test": "yarn workspace recovery-test", "ts-integration": "yarn workspace ts-integration", "zk": "yarn workspace zk" }, diff --git a/prover/CHANGELOG.md b/prover/CHANGELOG.md index 4313c0a4fc08..eb727013603e 100644 --- a/prover/CHANGELOG.md +++ b/prover/CHANGELOG.md @@ -1,5 +1,19 @@ # Changelog +## [14.3.0](https://github.com/matter-labs/zksync-era/compare/prover-v14.2.0...prover-v14.3.0) (2024-05-23) + + +### Features + +* **config:** remove zksync home ([#2022](https://github.com/matter-labs/zksync-era/issues/2022)) ([d08fe81](https://github.com/matter-labs/zksync-era/commit/d08fe81f4ec6c3aaeb5ad98351e44a63e5b100be)) +* **prover_cli:** add general status for batch command ([#1953](https://github.com/matter-labs/zksync-era/issues/1953)) ([7b0df3b](https://github.com/matter-labs/zksync-era/commit/7b0df3b22f04f1fdead308ec30572f565b34dd5c)) +* **prover:** add GPU feature for compressor ([#1838](https://github.com/matter-labs/zksync-era/issues/1838)) ([e9a2213](https://github.com/matter-labs/zksync-era/commit/e9a2213985928cd3804a3855ccfde6a7d99da238)) + + +### Bug Fixes + +* **prover:** Fix path to vk_setup_data_generator_server_fri ([#2025](https://github.com/matter-labs/zksync-era/issues/2025)) ([dbe4d6f](https://github.com/matter-labs/zksync-era/commit/dbe4d6f1724a458e61ab56cd94d17e1ecfa4c207)) + ## [14.2.0](https://github.com/matter-labs/zksync-era/compare/prover-v14.1.1...prover-v14.2.0) (2024-05-17) diff --git a/prover/Cargo.lock b/prover/Cargo.lock index c13e06fd3021..89cb099cfa3c 100644 --- a/prover/Cargo.lock +++ b/prover/Cargo.lock @@ -444,6 +444,29 @@ dependencies = [ "serde", ] +[[package]] +name = "bindgen" +version = "0.59.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2bd2a9a458e8f4304c52c43ebb0cfbd520289f8379a52e329a38afda99bf8eb8" +dependencies = [ + "bitflags 1.3.2", + "cexpr", + "clang-sys", + "clap 2.34.0", + "env_logger 0.9.3", + "lazy_static", + "lazycell", + "log", + "peeking_take_while", + "proc-macro2 1.0.78", + "quote 1.0.35", + "regex", + "rustc-hash", + "shlex", + "which", +] + [[package]] name = "bindgen" version = "0.65.1" @@ -855,6 +878,20 @@ dependencies = [ "windows-targets 0.48.5", ] +[[package]] +name = "circuit_definitions" +version = "0.1.0" +source = "git+https://github.com/matter-labs/era-zkevm_test_harness?branch=gpu-wrapper#ea0d54f6d5d7d3302a4a6594150a2ca809e6677b" +dependencies = [ + "crossbeam 0.8.4", + "derivative", + "seq-macro", + "serde", + "snark_wrapper", + "zk_evm 1.4.0", + "zkevm_circuits 1.4.0 (git+https://github.com/matter-labs/era-zkevm_circuits.git?branch=main)", +] + [[package]] name = "circuit_definitions" version = "1.5.0" @@ -878,7 +915,7 @@ dependencies = [ "derivative", "serde", "zk_evm 1.4.0", - "zkevm_circuits 1.4.0", + "zkevm_circuits 1.4.0 (git+https://github.com/matter-labs/era-zkevm_circuits.git?branch=v1.4.0)", ] [[package]] @@ -2250,6 +2287,17 @@ version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" +[[package]] +name = "futures-locks" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45ec6fe3675af967e67c5536c0b9d44e34e6c52f86bedc4ea49c5317b8e94d06" +dependencies = [ + "futures-channel", + "futures-task", + "tokio", +] + [[package]] name = "futures-macro" version = "0.3.30" @@ -2455,6 +2503,35 @@ dependencies = [ "async-trait", ] +[[package]] +name = "gpu-ffi" +version = "0.1.0" +source = "git+https://github.com/matter-labs/era-heavy-ops-service.git?rev=3d33e06#3d33e069d9d263f3a9626d235ac6dc6c49179965" +dependencies = [ + "bindgen 0.59.2", + "crossbeam 0.7.3", + "derivative", + "futures 0.3.30", + "futures-locks", + "num_cpus", +] + +[[package]] +name = "gpu-prover" +version = "0.1.0" +source = "git+https://github.com/matter-labs/era-heavy-ops-service.git?rev=3d33e06#3d33e069d9d263f3a9626d235ac6dc6c49179965" +dependencies = [ + "bit-vec", + "cfg-if 1.0.0", + "crossbeam 0.7.3", + "franklin-crypto 0.0.5 (git+https://github.com/matter-labs/franklin-crypto?branch=snark_wrapper)", + "gpu-ffi", + "itertools 0.10.5", + "num_cpus", + "rand 0.4.6", + "serde", +] + [[package]] name = "group" version = "0.12.1" @@ -4556,7 +4633,7 @@ version = "0.1.0" dependencies = [ "anyhow", "bincode", - "circuit_definitions", + "circuit_definitions 1.5.0", "clap 4.4.6", "colored", "dialoguer", @@ -5680,7 +5757,7 @@ dependencies = [ "blake2 0.10.6 (registry+https://github.com/rust-lang/crates.io-index)", "boojum", "boojum-cuda", - "circuit_definitions", + "circuit_definitions 1.5.0", "cudart", "cudart-sys", "derivative", @@ -6991,7 +7068,7 @@ version = "0.1.0" dependencies = [ "anyhow", "bincode", - "circuit_definitions", + "circuit_definitions 1.5.0", "clap 4.4.6", "hex", "itertools 0.10.5", @@ -7013,6 +7090,7 @@ dependencies = [ "zksync_env_config", "zksync_prover_fri_types", "zksync_types", + "zksync_utils", ] [[package]] @@ -7372,6 +7450,16 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "wrapper-prover" +version = "0.1.0" +source = "git+https://github.com/matter-labs/era-heavy-ops-service.git?rev=3d33e06#3d33e069d9d263f3a9626d235ac6dc6c49179965" +dependencies = [ + "circuit_definitions 0.1.0", + "gpu-prover", + "zkevm_test_harness 1.4.0", +] + [[package]] name = "wyz" version = "0.5.1" @@ -7501,7 +7589,7 @@ dependencies = [ [[package]] name = "zk_evm" version = "1.5.0" -source = "git+https://github.com/matter-labs/era-zk_evm.git?branch=v1.5.0#c42da1512334c3d95869198e41ee4f0da68812b4" +source = "git+https://github.com/matter-labs/era-zk_evm.git?branch=v1.5.0#9bbf7ffd2c38ee8b9667e96eaf0c111037fe976f" dependencies = [ "anyhow", "lazy_static", @@ -7586,6 +7674,27 @@ dependencies = [ "zkevm_opcode_defs 1.5.0", ] +[[package]] +name = "zkevm_circuits" +version = "1.4.0" +source = "git+https://github.com/matter-labs/era-zkevm_circuits.git?branch=main#fb3e2574b5c890342518fc930c145443f039a105" +dependencies = [ + "arrayvec 0.7.4", + "bincode", + "boojum", + "cs_derive 0.1.0 (git+https://github.com/matter-labs/era-boojum?branch=main)", + "derivative", + "hex", + "itertools 0.10.5", + "rand 0.4.6", + "rand 0.8.5", + "seq-macro", + "serde", + "serde_json", + "smallvec", + "zkevm_opcode_defs 1.3.2", +] + [[package]] name = "zkevm_circuits" version = "1.4.0" @@ -7730,13 +7839,36 @@ dependencies = [ "zkevm-assembly 1.3.2", ] +[[package]] +name = "zkevm_test_harness" +version = "1.4.0" +source = "git+https://github.com/matter-labs/era-zkevm_test_harness?branch=gpu-wrapper#ea0d54f6d5d7d3302a4a6594150a2ca809e6677b" +dependencies = [ + "bincode", + "circuit_definitions 0.1.0", + "codegen", + "crossbeam 0.8.4", + "derivative", + "env_logger 0.9.3", + "hex", + "rand 0.4.6", + "rayon", + "serde", + "serde_json", + "smallvec", + "structopt", + "test-log", + "tracing", + "zkevm-assembly 1.3.2", +] + [[package]] name = "zkevm_test_harness" version = "1.5.0" source = "git+https://github.com/matter-labs/era-zkevm_test_harness.git?branch=v1.5.0#ecb08797ced36fcc7d3696ffd2ec6a2d534b9395" dependencies = [ "bincode", - "circuit_definitions", + "circuit_definitions 1.5.0", "circuit_sequencer_api 0.1.50", "codegen", "crossbeam 0.8.4", @@ -7779,7 +7911,7 @@ dependencies = [ [[package]] name = "zksync_concurrency" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "once_cell", @@ -7810,7 +7942,7 @@ dependencies = [ [[package]] name = "zksync_consensus_crypto" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "blst", @@ -7831,7 +7963,7 @@ dependencies = [ [[package]] name = "zksync_consensus_roles" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "bit-vec", @@ -7852,7 +7984,7 @@ dependencies = [ [[package]] name = "zksync_consensus_storage" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "async-trait", @@ -7870,7 +8002,7 @@ dependencies = [ [[package]] name = "zksync_consensus_utils" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "rand 0.8.5", "thiserror", @@ -8069,6 +8201,7 @@ dependencies = [ "vise", "vk_setup_data_generator_server_fri", "vlog", + "wrapper-prover", "zkevm_test_harness 1.3.3", "zkevm_test_harness 1.5.0", "zksync_config", @@ -8084,7 +8217,7 @@ dependencies = [ [[package]] name = "zksync_protobuf" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "bit-vec", @@ -8104,7 +8237,7 @@ dependencies = [ [[package]] name = "zksync_protobuf_build" version = "0.1.0" -source = "git+https://github.com/matter-labs/era-consensus.git?rev=92ecb2d5d65e3bc4a883dacd18d0640e86576c8c#92ecb2d5d65e3bc4a883dacd18d0640e86576c8c" +source = "git+https://github.com/matter-labs/era-consensus.git?rev=3e6f101ee4124308c4c974caaa259d524549b0c6#3e6f101ee4124308c4c974caaa259d524549b0c6" dependencies = [ "anyhow", "heck 0.5.0", @@ -8123,7 +8256,7 @@ version = "0.1.0" dependencies = [ "anyhow", "async-trait", - "circuit_definitions", + "circuit_definitions 1.5.0", "ctrlc", "futures 0.3.30", "local-ip-address", @@ -8178,7 +8311,7 @@ dependencies = [ name = "zksync_prover_fri_types" version = "0.1.0" dependencies = [ - "circuit_definitions", + "circuit_definitions 1.5.0", "serde", "zksync_object_store", "zksync_types", @@ -8318,8 +8451,10 @@ dependencies = [ "hex", "itertools 0.10.5", "num", + "once_cell", "reqwest", "serde", + "serde_json", "thiserror", "tokio", "tracing", @@ -8355,7 +8490,7 @@ dependencies = [ "anyhow", "async-trait", "bincode", - "circuit_definitions", + "circuit_definitions 1.5.0", "const-decoder", "ctrlc", "futures 0.3.30", diff --git a/prover/Cargo.toml b/prover/Cargo.toml index 3a958aad30fd..ca1f97d75b83 100644 --- a/prover/Cargo.toml +++ b/prover/Cargo.toml @@ -4,7 +4,6 @@ members = [ "prover_fri_utils", "prover_fri_types", "prover_dal", - # binaries "witness_generator", "vk_setup_data_generator_server_fri", @@ -90,6 +89,9 @@ zksync_utils = { path = "../core/lib/utils" } zksync_eth_client = { path = "../core/lib/eth_client" } zksync_contracts = { path = "../core/lib/contracts" } +wrapper_prover = { package = "wrapper-prover", git = "https://github.com/matter-labs/era-heavy-ops-service.git", rev = "3d33e06" } + + # for `perf` profiling [profile.perf] inherits = "release" diff --git a/prover/proof_fri_compressor/Cargo.toml b/prover/proof_fri_compressor/Cargo.toml index 7e602d754c18..dd1aad902da3 100644 --- a/prover/proof_fri_compressor/Cargo.toml +++ b/prover/proof_fri_compressor/Cargo.toml @@ -39,3 +39,8 @@ bincode.workspace = true reqwest = { workspace = true, features = ["blocking"] } serde_json.workspace = true serde = { workspace = true, features = ["derive"] } +wrapper_prover = { workspace = true, optional = true } + +[features] +gpu = ["wrapper_prover"] + diff --git a/prover/proof_fri_compressor/src/compressor.rs b/prover/proof_fri_compressor/src/compressor.rs index 6f933aaf4a2c..c85162ccdfe0 100644 --- a/prover/proof_fri_compressor/src/compressor.rs +++ b/prover/proof_fri_compressor/src/compressor.rs @@ -5,14 +5,20 @@ use async_trait::async_trait; use circuit_sequencer_api::proof::FinalProof; use prover_dal::{ConnectionPool, Prover, ProverDal}; use tokio::task::JoinHandle; -use zkevm_test_harness::proof_wrapper_utils::{wrap_proof, WrapperConfig}; +#[cfg(feature = "gpu")] +use wrapper_prover::{Bn256, GPUWrapperConfigs, WrapperProver, DEFAULT_WRAPPER_CONFIG}; +#[cfg(not(feature = "gpu"))] +use zkevm_test_harness::proof_wrapper_utils::WrapperConfig; +#[allow(unused_imports)] +use zkevm_test_harness::proof_wrapper_utils::{get_trusted_setup, wrap_proof}; +#[cfg(not(feature = "gpu"))] +use zkevm_test_harness_1_3_3::bellman::bn256::Bn256; use zkevm_test_harness_1_3_3::{ abstract_zksync_circuit::concrete_circuits::{ ZkSyncCircuit, ZkSyncProof, ZkSyncVerificationKey, }, - bellman::{ - bn256::Bn256, - plonk::better_better_cs::{proof::Proof, setup::VerificationKey as SnarkVerificationKey}, + bellman::plonk::better_better_cs::{ + proof::Proof, setup::VerificationKey as SnarkVerificationKey, }, witness::oracle::VmWitnessOracle, }; @@ -62,9 +68,30 @@ impl ProofCompressor { } } + fn verify_proof(keystore: Keystore, serialized_proof: Vec) -> anyhow::Result<()> { + let proof: Proof>> = + bincode::deserialize(&serialized_proof) + .expect("Failed to deserialize proof with ZkSyncCircuit"); + // We're fetching the key as String and deserializing it here + // as we don't want to include the old version of prover in the main libraries. + let existing_vk_serialized = keystore + .load_snark_verification_key() + .context("get_snark_vk()")?; + let existing_vk = serde_json::from_str::< + SnarkVerificationKey>>, + >(&existing_vk_serialized)?; + + let vk = ZkSyncVerificationKey::from_verification_key_and_numeric_type(0, existing_vk); + let scheduler_proof = ZkSyncProof::from_proof_and_numeric_type(0, proof.clone()); + match vk.verify_proof(&scheduler_proof) { + true => tracing::info!("Compressed proof verified successfully"), + false => anyhow::bail!("Compressed proof verification failed "), + } + Ok(()) + } pub fn compress_proof( proof: ZkSyncRecursionLayerProof, - compression_mode: u8, + _compression_mode: u8, verify_wrapper_proof: bool, ) -> anyhow::Result { let keystore = Keystore::default(); @@ -73,35 +100,36 @@ impl ProofCompressor { ZkSyncRecursionLayerStorageType::SchedulerCircuit as u8, ) .context("get_recursiver_layer_vk_for_circuit_type()")?; - let config = WrapperConfig::new(compression_mode); - let (wrapper_proof, _) = wrap_proof(proof, scheduler_vk, config); - let inner = wrapper_proof.into_inner(); + #[cfg(feature = "gpu")] + let wrapper_proof = { + let crs = get_trusted_setup(); + let wrapper_config = DEFAULT_WRAPPER_CONFIG; + let mut prover = WrapperProver::::new(&crs, wrapper_config).unwrap(); + + prover + .generate_setup_data(scheduler_vk.into_inner()) + .unwrap(); + prover.generate_proofs(proof.into_inner()).unwrap(); + + prover.get_wrapper_proof().unwrap() + }; + #[cfg(not(feature = "gpu"))] + let wrapper_proof = { + let config = WrapperConfig::new(_compression_mode); + + let (wrapper_proof, _) = wrap_proof(proof, scheduler_vk, config); + wrapper_proof.into_inner() + }; + // (Re)serialization should always succeed. - let serialized = bincode::serialize(&inner) + let serialized = bincode::serialize(&wrapper_proof) .expect("Failed to serialize proof with ZkSyncSnarkWrapperCircuit"); if verify_wrapper_proof { // If we want to verify the proof, we have to deserialize it, with proper type. // So that we can pass it into `from_proof_and_numeric_type` method below. - let proof: Proof>> = - bincode::deserialize(&serialized) - .expect("Failed to deserialize proof with ZkSyncCircuit"); - // We're fetching the key as String and deserializing it here - // as we don't want to include the old version of prover in the main libraries. - let existing_vk_serialized = keystore - .load_snark_verification_key() - .context("get_snark_vk()")?; - let existing_vk = serde_json::from_str::< - SnarkVerificationKey>>, - >(&existing_vk_serialized)?; - - let vk = ZkSyncVerificationKey::from_verification_key_and_numeric_type(0, existing_vk); - let scheduler_proof = ZkSyncProof::from_proof_and_numeric_type(0, proof.clone()); - match vk.verify_proof(&scheduler_proof) { - true => tracing::info!("Compressed proof verified successfully"), - false => anyhow::bail!("Compressed proof verification failed "), - } + Self::verify_proof(keystore, serialized.clone())?; } // For sending to L1, we can use the `FinalProof` type, that has a generic circuit inside, that is not used for serialization. diff --git a/prover/proof_fri_compressor/src/main.rs b/prover/proof_fri_compressor/src/main.rs index d303c62804b2..1d261cd6b352 100644 --- a/prover/proof_fri_compressor/src/main.rs +++ b/prover/proof_fri_compressor/src/main.rs @@ -1,3 +1,5 @@ +#![feature(generic_const_exprs)] + use std::{env, time::Duration}; use anyhow::Context as _; diff --git a/prover/prover_cli/Cargo.toml b/prover/prover_cli/Cargo.toml index 8b4e131caa2d..272baaf9491a 100644 --- a/prover/prover_cli/Cargo.toml +++ b/prover/prover_cli/Cargo.toml @@ -12,7 +12,7 @@ categories.workspace = true [dependencies] dialoguer.workspace = true tokio = { workspace = true, features = ["rt-multi-thread", "macros"] } -clap = { workspace = true, features = ["derive"] } +clap = { workspace = true, features = ["derive", "env"] } tracing-subscriber = { workspace = true, features = ["env-filter"] } tracing.workspace = true bincode.workspace = true diff --git a/prover/prover_cli/src/cli.rs b/prover/prover_cli/src/cli.rs index 6d05fe3c97fc..bbcf5ac8b98d 100644 --- a/prover/prover_cli/src/cli.rs +++ b/prover/prover_cli/src/cli.rs @@ -1,7 +1,7 @@ use clap::{command, Args, Parser, Subcommand}; use zksync_types::url::SensitiveUrl; -use crate::commands::{self, delete, get_file_info, requeue, restart}; +use crate::commands::{self, config, delete, get_file_info, requeue, restart}; pub const VERSION_STRING: &str = env!("CARGO_PKG_VERSION"); @@ -14,14 +14,13 @@ struct ProverCLI { config: ProverCLIConfig, } -// Note: This is a temporary solution for the configuration of the CLI. In the -// future, we should have an `config` command to set the configuration in a -// `.config` file. +// Note: this is set via the `config` command. Values are taken from the file pointed +// by the env var `PLI__CONFIG` or from `$ZKSYNC_HOME/etc/pliconfig` if unset. #[derive(Args)] pub struct ProverCLIConfig { #[clap( - long, - default_value = "postgres://postgres:notsecurepassword@localhost/prover_local" + default_value = "postgres://postgres:notsecurepassword@localhost/prover_local", + env("PLI__DB_URL") )] pub db_url: SensitiveUrl, } @@ -29,6 +28,7 @@ pub struct ProverCLIConfig { #[derive(Subcommand)] enum ProverCommand { FileInfo(get_file_info::Args), + Config(ProverCLIConfig), Delete(delete::Args), #[command(subcommand)] Status(commands::StatusCommand), @@ -40,6 +40,7 @@ pub async fn start() -> anyhow::Result<()> { let ProverCLI { command, config } = ProverCLI::parse(); match command { ProverCommand::FileInfo(args) => get_file_info::run(args).await?, + ProverCommand::Config(cfg) => config::run(cfg).await?, ProverCommand::Delete(args) => delete::run(args).await?, ProverCommand::Status(cmd) => cmd.run(config).await?, ProverCommand::Requeue(args) => requeue::run(args, config).await?, diff --git a/prover/prover_cli/src/commands/config.rs b/prover/prover_cli/src/commands/config.rs new file mode 100644 index 000000000000..4b5f2421c7a4 --- /dev/null +++ b/prover/prover_cli/src/commands/config.rs @@ -0,0 +1,7 @@ +use crate::{cli::ProverCLIConfig, config}; + +pub async fn run(cfg: ProverCLIConfig) -> anyhow::Result<()> { + let envfile = config::get_envfile()?; + config::update_envfile(&envfile, "PLI__DB_URL", cfg.db_url.expose_str())?; + Ok(()) +} diff --git a/prover/prover_cli/src/commands/mod.rs b/prover/prover_cli/src/commands/mod.rs index cd76c6aff960..34291d91ce60 100644 --- a/prover/prover_cli/src/commands/mod.rs +++ b/prover/prover_cli/src/commands/mod.rs @@ -1,3 +1,4 @@ +pub(crate) mod config; pub(crate) mod delete; pub(crate) mod get_file_info; pub(crate) mod requeue; diff --git a/prover/prover_cli/src/commands/status/batch.rs b/prover/prover_cli/src/commands/status/batch.rs index 389437f17ac7..6f52170444a5 100644 --- a/prover/prover_cli/src/commands/status/batch.rs +++ b/prover/prover_cli/src/commands/status/batch.rs @@ -35,6 +35,22 @@ pub(crate) async fn run(args: Args, config: ProverCLIConfig) -> anyhow::Result<( "== {} ==", format!("Batch {} Status", batch_data.batch_number).bold() ); + + if let Status::Custom(msg) = batch_data.compressor.witness_generator_jobs_status() { + if msg.contains("Sent to server") { + println!("> Proof sent to server ✅"); + return Ok(()); + } + } + + let basic_witness_generator_status = batch_data + .basic_witness_generator + .witness_generator_jobs_status(); + if matches!(basic_witness_generator_status, Status::JobsNotFound) { + println!("> No batch found. 🚫"); + return Ok(()); + } + if !args.verbose { display_batch_status(batch_data); } else { diff --git a/prover/prover_cli/src/config/mod.rs b/prover/prover_cli/src/config/mod.rs new file mode 100644 index 000000000000..452e1ad9ce01 --- /dev/null +++ b/prover/prover_cli/src/config/mod.rs @@ -0,0 +1,49 @@ +use std::io::Write; + +pub fn get_envfile() -> anyhow::Result { + if let Ok(envfile) = std::env::var("PLI__CONFIG") { + return Ok(envfile); + } + Ok(std::env::var("ZKSYNC_HOME").map(|home| home + "/etc/pliconfig")?) +} + +pub fn load_envfile(path: impl AsRef) -> anyhow::Result<()> { + std::fs::read_to_string(path)? + .lines() + .filter(|l| !l.starts_with('#')) + .filter_map(|l| l.split_once('=')) + .for_each(|(k, v)| std::env::set_var(k, v)); + + Ok(()) +} + +pub fn update_envfile( + path: impl AsRef + std::marker::Copy, + key: impl AsRef, + value: impl AsRef, +) -> anyhow::Result<()> { + let prefix = format!("{}=", key.as_ref()); + let kv = format!("{}={}", key.as_ref(), value.as_ref()); + let swapfile = path.as_ref().with_extension(".swp"); + let mut out = std::io::BufWriter::new(std::fs::File::create_new(&swapfile)?); + let mut found = false; + + std::fs::read_to_string(path)? + .lines() + .map(|l| { + if l.starts_with(&prefix) { + found = true; + kv.clone() + } else { + l.to_string() + } + }) + .try_for_each(|l| writeln!(&mut out, "{}", l))?; + if !found { + writeln!(&mut out, "{}", kv)?; + } + out.flush()?; + std::fs::rename(swapfile, path)?; + + Ok(()) +} diff --git a/prover/prover_cli/src/examples/pliconfig b/prover/prover_cli/src/examples/pliconfig new file mode 100644 index 000000000000..5a870cd031d7 --- /dev/null +++ b/prover/prover_cli/src/examples/pliconfig @@ -0,0 +1,2 @@ +### PLI__DB_URL: full URL for connecting to the prover DB +PLI__DB_URL=postgres://postgres:notsecurepassword@localhost/prover_local_default_from_env diff --git a/prover/prover_cli/src/lib.rs b/prover/prover_cli/src/lib.rs index 3ef8b313f0c2..3a441e45bdeb 100644 --- a/prover/prover_cli/src/lib.rs +++ b/prover/prover_cli/src/lib.rs @@ -1,2 +1,3 @@ pub mod cli; -mod commands; +pub mod commands; +pub mod config; diff --git a/prover/prover_cli/src/main.rs b/prover/prover_cli/src/main.rs index 4bc0908a4f83..b393fad6a31b 100644 --- a/prover/prover_cli/src/main.rs +++ b/prover/prover_cli/src/main.rs @@ -1,10 +1,19 @@ -use prover_cli::cli; +use prover_cli::{cli, config}; #[tokio::main] async fn main() { tracing_subscriber::fmt() .with_max_level(tracing::Level::ERROR) .init(); + + config::get_envfile() + .and_then(config::load_envfile) + .inspect_err(|err| { + tracing::error!("{err:?}"); + std::process::exit(1); + }) + .unwrap(); + match cli::start().await { Ok(_) => {} Err(err) => { diff --git a/prover/prover_fri/README.md b/prover/prover_fri/README.md index 9ec6cb870c75..5f0a26cfdd49 100644 --- a/prover/prover_fri/README.md +++ b/prover/prover_fri/README.md @@ -16,9 +16,9 @@ will pull jobs from the database and do their part of the pipeline, loading inte ```mermaid flowchart LR - A["Operator"] --> |Produces block| F[Prover Gateway] - F --> |Inserts into DB| B["Postgres DB"] - B --> |Retrieves proven block \nafter compression| F + A["Operator"] -->|Produces block| F[Prover Gateway] + F -->|Inserts into DB| B["Postgres DB"] + B -->|Retrieves proven block \nafter compression| F B --> C["Witness"] C --- C1["Basic Circuits"] C --- C2["Leaf Aggregation"] @@ -27,9 +27,9 @@ flowchart LR C --- C5["Scheduler"] C --> B B --> D["Vector Generator/Prover"] - D --> |Proven Block| B + D -->|Proven Block| B B --> G["Compressor"] - G --> |Compressed block| B + G -->|Compressed block| B ``` ## Prerequisites @@ -60,9 +60,10 @@ installation as a pre-requisite, alongside these machine specs: Note that it will produce a first l1 batch that can be proven (should be batch 0). -3. Generate the GPU setup data (no need to regenerate if it's already there). This will consume around 20GB of disk. You - need to be in the `prover/` directory (for all commands from here onwards, you need to be in the `prover/` directory) - and run: +3. Generate the GPU setup data (no need to regenerate if it's already there). If you want to use this with the GPU + compressors, you need to change the key in the file from `setup_2^26.key` to `setup_2^24.key`. This will consume + around 20GB of disk. You need to be in the `prover/` directory (for all commands from here onwards, you need to be in + the `prover/` directory) and run: ```console ./setup.sh gpu @@ -167,6 +168,43 @@ Machine specs: zk f cargo run --release --bin zksync_proof_fri_compressor ``` +## Running GPU compressors + +There is an option to run compressors with the GPU, which will significantly improve the performance. + +1. The hardware setup should be the same as for GPU prover +2. Install and compile `era-bellman-cuda` library + + ```console + git clone https://github.com/matter-labs/bellman-cuda.git --branch dev bellman-cuda + cmake -Bbellman-cuda/build -Sbellman-cuda/ -DCMAKE_BUILD_TYPE=Release + cmake --build bellman-cuda/build/ + ``` + +3. Set path of library as environmental variable + + ```console + export BELLMAN_CUDA_DIR=$PWD/bellman-cuda + ``` + +4. GPU compressor uses `setup_2^24.key`. Download it by using: + + ```console + wget https://storage.googleapis.com/matterlabs-setup-keys-us/setup-keys/setup_2^24.key + ``` + +5. Set the env variable with it's path: + + ```console + export CRS_FILE=$PWD/setup_2^24.key + ``` + +6. Run the compressor using: + + ```console + zk f cargo run ---features "gpu" --release --bin zksync_proof_fri_compressor + ``` + ## Checking the status of the prover Once everything is running (either with the CPU or GPU prover), the server should have at least three blocks, and you diff --git a/prover/setup.sh b/prover/setup.sh index e755f5ae433a..2d546c1f8bd6 100755 --- a/prover/setup.sh +++ b/prover/setup.sh @@ -14,7 +14,6 @@ if [[ -z "${ZKSYNC_HOME}" ]]; then fi sed -i.backup 's/^proof_sending_mode=.*$/proof_sending_mode="OnlyRealProofs"/' ../etc/env/base/eth_sender.toml -sed -i.backup 's/^proof_loading_mode=.*$/proof_loading_mode="FriProofFromGcs"/' ../etc/env/base/eth_sender.toml rm ../etc/env/base/eth_sender.toml.backup sed -i.backup 's/^setup_data_path=.*$/setup_data_path="vk_setup_data_generator_server_fri\/data\/"/' ../etc/env/base/fri_prover.toml rm ../etc/env/base/fri_prover.toml.backup diff --git a/prover/vk_setup_data_generator_server_fri/Cargo.toml b/prover/vk_setup_data_generator_server_fri/Cargo.toml index c7309ee98f37..bda9dafe3de6 100644 --- a/prover/vk_setup_data_generator_server_fri/Cargo.toml +++ b/prover/vk_setup_data_generator_server_fri/Cargo.toml @@ -22,9 +22,10 @@ path = "src/lib.rs" [dependencies] vlog.workspace = true zksync_types.workspace = true +zksync_utils.workspace = true zksync_prover_fri_types.workspace = true zkevm_test_harness.workspace = true -circuit_definitions = { workspace = true, features = [ "log_tracing" ] } +circuit_definitions = { workspace = true, features = ["log_tracing"] } shivini = { workspace = true, optional = true } zksync_config.workspace = true zksync_env_config.workspace = true diff --git a/prover/vk_setup_data_generator_server_fri/src/keystore.rs b/prover/vk_setup_data_generator_server_fri/src/keystore.rs index 21ca42ba3a38..d1ba66e1fd2a 100644 --- a/prover/vk_setup_data_generator_server_fri/src/keystore.rs +++ b/prover/vk_setup_data_generator_server_fri/src/keystore.rs @@ -1,7 +1,7 @@ use std::{ fs::{self, File}, io::Read, - path::Path, + path::{Path, PathBuf}, }; use anyhow::Context as _; @@ -23,7 +23,7 @@ use zksync_types::basic_fri_types::AggregationRound; #[cfg(feature = "gpu")] use crate::GoldilocksGpuProverSetupData; -use crate::{GoldilocksProverSetupData, VkCommitments}; +use crate::{utils::core_workspace_dir_or_current_dir, GoldilocksProverSetupData, VkCommitments}; pub enum ProverServiceDataType { VerificationKey, @@ -38,23 +38,19 @@ pub enum ProverServiceDataType { /// - large setup keys, used during proving. pub struct Keystore { /// Directory to store all the small keys. - basedir: String, + basedir: PathBuf, /// Directory to store large setup keys. setup_data_path: Option, } -fn get_base_path_from_env() -> String { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| "/".into()); - format!( - "{}/prover/vk_setup_data_generator_server_fri/data", - zksync_home - ) +fn get_base_path() -> PathBuf { + core_workspace_dir_or_current_dir().join("prover/vk_setup_data_generator_server_fri/data") } impl Default for Keystore { fn default() -> Self { Self { - basedir: get_base_path_from_env(), + basedir: get_base_path(), setup_data_path: Some( FriProverConfig::from_env() .expect("FriProverConfig::from_env()") @@ -67,20 +63,20 @@ impl Default for Keystore { impl Keystore { /// Base-dir is the location of smaller keys (like verification keys and finalization hints). /// Setup data path is used for the large setup keys. - pub fn new(basedir: String, setup_data_path: String) -> Self { + pub fn new(basedir: PathBuf, setup_data_path: String) -> Self { Keystore { basedir, setup_data_path: Some(setup_data_path), } } - pub fn new_with_optional_setup_path(basedir: String, setup_data_path: Option) -> Self { + pub fn new_with_optional_setup_path(basedir: PathBuf, setup_data_path: Option) -> Self { Keystore { basedir, setup_data_path, } } - pub fn get_base_path(&self) -> &str { + pub fn get_base_path(&self) -> &PathBuf { &self.basedir } @@ -88,43 +84,49 @@ impl Keystore { &self, key: ProverServiceDataKey, service_data_type: ProverServiceDataType, - ) -> String { + ) -> PathBuf { let name = key.name(); match service_data_type { ProverServiceDataType::VerificationKey => { - format!("{}/verification_{}_key.json", self.basedir, name) - } - ProverServiceDataType::SetupData => { - format!( - "{}/setup_{}_data.bin", - self.setup_data_path - .as_ref() - .expect("Setup data path not set"), - name - ) - } - ProverServiceDataType::FinalizationHints => { - format!("{}/finalization_hints_{}.bin", self.basedir, name) - } - ProverServiceDataType::SnarkVerificationKey => { - format!("{}/snark_verification_{}_key.json", self.basedir, name) + self.basedir.join(format!("verification_{}_key.json", name)) } + ProverServiceDataType::SetupData => PathBuf::from(format!( + "{}/setup_{}_data.bin", + self.setup_data_path + .as_ref() + .expect("Setup data path not set"), + name + )), + ProverServiceDataType::FinalizationHints => self + .basedir + .join(format!("finalization_hints_{}.bin", name)), + ProverServiceDataType::SnarkVerificationKey => self + .basedir + .join(format!("snark_verification_{}_key.json", name)), } } - fn load_json_from_file Deserialize<'a>>(filepath: String) -> anyhow::Result { + fn load_json_from_file Deserialize<'a>>( + filepath: impl AsRef + std::fmt::Debug, + ) -> anyhow::Result { let text = std::fs::read_to_string(&filepath) - .with_context(|| format!("Failed reading verification key from path: {filepath}"))?; - serde_json::from_str::(&text) - .with_context(|| format!("Failed deserializing verification key from path: {filepath}")) + .with_context(|| format!("Failed reading verification key from path: {filepath:?}"))?; + serde_json::from_str::(&text).with_context(|| { + format!("Failed deserializing verification key from path: {filepath:?}") + }) } - fn save_json_pretty(filepath: String, data: &T) -> anyhow::Result<()> { + fn save_json_pretty( + filepath: impl AsRef + std::fmt::Debug, + data: &T, + ) -> anyhow::Result<()> { std::fs::write(&filepath, serde_json::to_string_pretty(data).unwrap()) - .with_context(|| format!("writing to '{filepath}' failed")) + .with_context(|| format!("writing to '{filepath:?}' failed")) } - fn load_bincode_from_file Deserialize<'a>>(filepath: String) -> anyhow::Result { - let mut file = File::open(filepath.clone()) + fn load_bincode_from_file Deserialize<'a>>( + filepath: impl AsRef + std::fmt::Debug, + ) -> anyhow::Result { + let mut file = File::open(&filepath) .with_context(|| format!("Failed reading setup-data from path: {filepath:?}"))?; let mut buffer = Vec::new(); file.read_to_end(&mut buffer).with_context(|| { @@ -166,7 +168,7 @@ impl Keystore { ProverServiceDataKey::new(vk.numeric_circuit_type(), AggregationRound::BasicCircuits), ProverServiceDataType::VerificationKey, ); - tracing::info!("saving basic verification key to: {}", filepath); + tracing::info!("saving basic verification key to: {:?}", filepath); Self::save_json_pretty(filepath, &vk) } @@ -178,7 +180,7 @@ impl Keystore { ProverServiceDataKey::new_recursive(vk.numeric_circuit_type()), ProverServiceDataType::VerificationKey, ); - tracing::info!("saving recursive layer verification key to: {}", filepath); + tracing::info!("saving recursive layer verification key to: {:?}", filepath); Self::save_json_pretty(filepath, &vk) } @@ -193,7 +195,7 @@ impl Keystore { ) -> anyhow::Result<()> { let filepath = self.get_file_path(key.clone(), ProverServiceDataType::FinalizationHints); - tracing::info!("saving finalization hints for {:?} to: {}", key, filepath); + tracing::info!("saving finalization hints for {:?} to: {:?}", key, filepath); let serialized = bincode::serialize(&hint).context("Failed to serialize finalization hints")?; fs::write(filepath, serialized).context("Failed to write finalization hints to file") @@ -227,8 +229,9 @@ impl Keystore { ProverServiceDataKey::snark(), ProverServiceDataType::SnarkVerificationKey, ); - std::fs::read_to_string(&filepath) - .with_context(|| format!("Failed reading Snark verification key from path: {filepath}")) + std::fs::read_to_string(&filepath).with_context(|| { + format!("Failed reading Snark verification key from path: {filepath:?}") + }) } pub fn save_snark_verification_key(&self, vk: ZkSyncSnarkWrapperVK) -> anyhow::Result<()> { @@ -236,7 +239,7 @@ impl Keystore { ProverServiceDataKey::snark(), ProverServiceDataType::SnarkVerificationKey, ); - tracing::info!("saving snark verification key to: {}", filepath); + tracing::info!("saving snark verification key to: {:?}", filepath); Self::save_json_pretty(filepath, &vk.into_inner()) } @@ -256,7 +259,7 @@ impl Keystore { file.read_to_end(&mut buffer).with_context(|| { format!("Failed reading setup-data to buffer from path: {filepath:?}") })?; - tracing::info!("loading {:?} setup data from path: {}", key, filepath); + tracing::info!("loading {:?} setup data from path: {:?}", key, filepath); bincode::deserialize::(&buffer).with_context(|| { format!("Failed deserializing setup-data at path: {filepath:?} for circuit: {key:?}") }) @@ -275,7 +278,7 @@ impl Keystore { file.read_to_end(&mut buffer).with_context(|| { format!("Failed reading setup-data to buffer from path: {filepath:?}") })?; - tracing::info!("loading {:?} setup data from path: {}", key, filepath); + tracing::info!("loading {:?} setup data from path: {:?}", key, filepath); bincode::deserialize::(&buffer).with_context(|| { format!("Failed deserializing setup-data at path: {filepath:?} for circuit: {key:?}") }) @@ -291,7 +294,7 @@ impl Keystore { serialized_setup_data: &Vec, ) -> anyhow::Result<()> { let filepath = self.get_file_path(key.clone(), ProverServiceDataType::SetupData); - tracing::info!("saving {:?} setup data to: {}", key, filepath); + tracing::info!("saving {:?} setup data to: {:?}", key, filepath); std::fs::write(filepath.clone(), serialized_setup_data) .with_context(|| format!("Failed saving setup-data at path: {filepath:?}")) } @@ -440,12 +443,9 @@ impl Keystore { } pub fn load_commitments(&self) -> anyhow::Result { - Self::load_json_from_file(format!("{}/commitments.json", self.get_base_path())) + Self::load_json_from_file(self.get_base_path().join("commitments.json")) } pub fn save_commitments(&self, commitments: &VkCommitments) -> anyhow::Result<()> { - Self::save_json_pretty( - format!("{}/commitments.json", self.get_base_path()), - &commitments, - ) + Self::save_json_pretty(self.get_base_path().join("commitments.json"), &commitments) } } diff --git a/prover/vk_setup_data_generator_server_fri/src/main.rs b/prover/vk_setup_data_generator_server_fri/src/main.rs index ce3a0799baa0..4cf7aa1abb30 100644 --- a/prover/vk_setup_data_generator_server_fri/src/main.rs +++ b/prover/vk_setup_data_generator_server_fri/src/main.rs @@ -158,7 +158,7 @@ fn print_stats(digests: HashMap) -> anyhow::Result<()> { fn keystore_from_optional_path(path: Option, setup_path: Option) -> Keystore { if let Some(path) = path { - return Keystore::new_with_optional_setup_path(path, setup_path); + return Keystore::new_with_optional_setup_path(path.into(), setup_path); } if setup_path.is_some() { panic!("--setup_path must not be set when --path is not set"); diff --git a/prover/vk_setup_data_generator_server_fri/src/tests.rs b/prover/vk_setup_data_generator_server_fri/src/tests.rs index 41aba88f784c..39b5f7a44fb8 100644 --- a/prover/vk_setup_data_generator_server_fri/src/tests.rs +++ b/prover/vk_setup_data_generator_server_fri/src/tests.rs @@ -63,15 +63,6 @@ proptest! { } -// Test `get_base_path` method -#[test] -fn test_get_base_path() { - let keystore = Keystore::default(); - - let base_path = keystore.get_base_path(); - assert!(!base_path.is_empty(), "Base path should not be empty"); -} - // Test `ProverServiceDataKey::new` method #[test] fn test_proverservicedatakey_new() { diff --git a/prover/vk_setup_data_generator_server_fri/src/utils.rs b/prover/vk_setup_data_generator_server_fri/src/utils.rs index 555204eb9e2b..0dff2f36cec7 100644 --- a/prover/vk_setup_data_generator_server_fri/src/utils.rs +++ b/prover/vk_setup_data_generator_server_fri/src/utils.rs @@ -1,3 +1,5 @@ +use std::path::PathBuf; + use anyhow::Context as _; use circuit_definitions::{ circuit_definitions::aux_layer::ZkSyncSnarkWrapperCircuit, @@ -20,6 +22,7 @@ use zksync_prover_fri_types::circuit_definitions::{ }, }; use zksync_types::H256; +use zksync_utils::locate_workspace; use crate::keystore::Keystore; @@ -112,23 +115,28 @@ pub fn calculate_snark_vk_hash(keystore: &Keystore) -> anyhow::Result { Ok(H256::from_slice(&computed_vk_hash)) } +/// Returns workspace of the core component, we assume that prover is one folder deeper. +/// Or fallback to current dir +pub fn core_workspace_dir_or_current_dir() -> PathBuf { + locate_workspace() + .map(|a| a.join("..")) + .unwrap_or_else(|| PathBuf::from(".")) +} + #[cfg(test)] mod tests { - use std::{env, path::PathBuf, str::FromStr}; + use std::{path::PathBuf, str::FromStr}; use super::*; #[test] fn test_keyhash_generation() { - let mut path_to_input = PathBuf::from(env::var("CARGO_MANIFEST_DIR").unwrap()); + let mut path_to_input = PathBuf::from(std::env::var("CARGO_MANIFEST_DIR").unwrap()); path_to_input.push("historical_data"); for version in 18..=22 { let basepath = path_to_input.join(format!("{}", version)); - let keystore = Keystore::new_with_optional_setup_path( - basepath.as_os_str().to_str().unwrap().to_string(), - None, - ); + let keystore = Keystore::new_with_optional_setup_path(basepath, None); let expected = H256::from_str(&keystore.load_commitments().unwrap().snark_wrapper).unwrap(); diff --git a/prover/vk_setup_data_generator_server_fri/src/vk_commitment_helper.rs b/prover/vk_setup_data_generator_server_fri/src/vk_commitment_helper.rs index 9a6c074b1d23..bf568e06157b 100644 --- a/prover/vk_setup_data_generator_server_fri/src/vk_commitment_helper.rs +++ b/prover/vk_setup_data_generator_server_fri/src/vk_commitment_helper.rs @@ -1,8 +1,10 @@ -use std::fs; +use std::{fs, path::PathBuf}; use anyhow::Context as _; use toml_edit::{Document, Item, Value}; +use crate::utils::core_workspace_dir_or_current_dir; + pub fn get_toml_formatted_value(string_value: String) -> Item { let mut value = Value::from(string_value); value.decor_mut().set_prefix(""); @@ -17,11 +19,10 @@ pub fn write_contract_toml(contract_doc: Document) -> anyhow::Result<()> { pub fn read_contract_toml() -> anyhow::Result { let path = get_contract_toml_path(); let toml_data = std::fs::read_to_string(path.clone()) - .with_context(|| format!("contract.toml file does not exist on path {path}"))?; + .with_context(|| format!("contract.toml file does not exist on path {path:?}"))?; toml_data.parse::().context("invalid config file") } -pub fn get_contract_toml_path() -> String { - let zksync_home = std::env::var("ZKSYNC_HOME").unwrap_or_else(|_| "/".into()); - format!("{}/etc/env/base/contracts.toml", zksync_home) +pub fn get_contract_toml_path() -> PathBuf { + core_workspace_dir_or_current_dir().join("etc/env/base/contracts.toml") } diff --git a/yarn.lock b/yarn.lock index 0e1ad2630d7c..b7e2b98c431e 100644 --- a/yarn.lock +++ b/yarn.lock @@ -10944,6 +10944,11 @@ yaml@^2.4.1: resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.4.1.tgz#2e57e0b5e995292c25c75d2658f0664765210eed" integrity sha512-pIXzoImaqmfOrL7teGUBt/T7ZDnyeGBWyXQBvOVhLkWLN37GXv8NMLK406UY6dS51JfcQHsmcW5cJ441bHg6Lg== +yaml@^2.4.2: + version "2.4.2" + resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.4.2.tgz#7a2b30f2243a5fc299e1f14ca58d475ed4bc5362" + integrity sha512-B3VqDZ+JAg1nZpaEmWtTXUlBneoGx6CPM9b0TENK6aoSu5t73dItudwdgmi6tHlIZZId4dZ9skcAQ2UbcyAeVA== + yargs-parser@20.2.4: version "20.2.4" resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.4.tgz#b42890f14566796f85ae8e3a25290d205f154a54" diff --git a/zk_toolbox/Cargo.lock b/zk_toolbox/Cargo.lock new file mode 100644 index 000000000000..2492caf89780 --- /dev/null +++ b/zk_toolbox/Cargo.lock @@ -0,0 +1,4552 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 3 + +[[package]] +name = "Inflector" +version = "0.11.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fe438c63458706e03479442743baae6c88256498e6431708f6dfc520a26515d3" +dependencies = [ + "lazy_static", + "regex", +] + +[[package]] +name = "addr2line" +version = "0.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a30b2e23b9e17a9f90641c7ab1549cd9b44f296d3ccbf309d2863cfe398a0cb" +dependencies = [ + "gimli", +] + +[[package]] +name = "adler" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" + +[[package]] +name = "aes" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures", +] + +[[package]] +name = "ahash" +version = "0.8.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011" +dependencies = [ + "cfg-if", + "getrandom", + "once_cell", + "version_check", + "zerocopy", +] + +[[package]] +name = "aho-corasick" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2969dcb958b36655471fc61f7e416fa76033bdd4bfed0678d8fee1e2d07a1f0" +dependencies = [ + "memchr", +] + +[[package]] +name = "allocator-api2" +version = "0.2.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c6cb57a04249c6480766f7f7cef5467412af1490f8d1e243141daddada3264f" + +[[package]] +name = "anstream" +version = "0.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96b09b5178381e0874812a9b157f7fe84982617e48f71f4e3235482775e5b540" +dependencies = [ + "anstyle", + "anstyle-parse", + "anstyle-query", + "anstyle-wincon", + "colorchoice", + "utf8parse", +] + +[[package]] +name = "anstyle" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8901269c6307e8d93993578286ac0edf7f195079ffff5ebdeea6a59ffb7e36bc" + +[[package]] +name = "anstyle-parse" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c75ac65da39e5fe5ab759307499ddad880d724eed2f6ce5b5e8a26f4f387928c" +dependencies = [ + "utf8parse", +] + +[[package]] +name = "anstyle-query" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e28923312444cdd728e4738b3f9c9cac739500909bb3d3c94b43551b16517648" +dependencies = [ + "windows-sys 0.52.0", +] + +[[package]] +name = "anstyle-wincon" +version = "3.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1cd54b81ec8d6180e24654d0b371ad22fc3dd083b6ff8ba325b72e00c87660a7" +dependencies = [ + "anstyle", + "windows-sys 0.52.0", +] + +[[package]] +name = "anyhow" +version = "1.0.82" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f538837af36e6f6a9be0faa67f9a314f8119e4e4b5867c6ab40ed60360142519" + +[[package]] +name = "arrayvec" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96d30a06541fbafbc7f82ed10c06164cfbd2c401138f6addd8404629c4b16711" + +[[package]] +name = "ascii-canvas" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8824ecca2e851cec16968d54a01dd372ef8f95b244fb84b84e70128be347c3c6" +dependencies = [ + "term", +] + +[[package]] +name = "async-trait" +version = "0.1.77" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c980ee35e870bd1a4d2c8294d4c04d0499e67bca1e4b5cefcc693c2fa00caea9" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "async_io_stream" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6d7b9decdf35d8908a7e3ef02f64c5e9b1695e230154c0e8de3969142d9b94c" +dependencies = [ + "futures", + "pharos", + "rustc_version", +] + +[[package]] +name = "atoi" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f28d99ec8bfea296261ca1af174f24225171fea9664ba9003cbebee704810528" +dependencies = [ + "num-traits", +] + +[[package]] +name = "auto_impl" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c87f3f15e7794432337fc718554eaa4dc8f04c9677a950ffe366f20a162ae42" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "autocfg" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" + +[[package]] +name = "backtrace" +version = "0.3.69" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2089b7e3f35b9dd2d0ed921ead4f6d318c27680d4a5bd167b3ee120edb105837" +dependencies = [ + "addr2line", + "cc", + "cfg-if", + "libc", + "miniz_oxide", + "object", + "rustc-demangle", +] + +[[package]] +name = "base16ct" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c7f02d4ea65f2c1853089ffd8d2787bdbc63de2f0d29dedbcf8ccdfa0ccd4cf" + +[[package]] +name = "base64" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" + +[[package]] +name = "base64" +version = "0.21.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" + +[[package]] +name = "base64ct" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c3c1a368f70d6cf7302d78f8f7093da241fb8e8807c05cc9e51a125895a6d5b" + +[[package]] +name = "bech32" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d86b93f97252c47b41663388e6d155714a9d0c398b99f1005cbc5f978b29f445" + +[[package]] +name = "bit-set" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0700ddab506f33b20a03b13996eccd309a48e5ff77d0d95926aa0210fb4e95f1" +dependencies = [ + "bit-vec", +] + +[[package]] +name = "bit-vec" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "349f9b6a179ed607305526ca489b34ad0a41aed5f7980fa90eb03160b69598fb" + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bitflags" +version = "2.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed570934406eb16438a4e976b1b4500774099c13b8cb96eec99f620f05090ddf" +dependencies = [ + "serde", +] + +[[package]] +name = "bitvec" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bc2832c24239b0141d5674bb9174f9d68a8b5b3f2753311927c172ca46f7e9c" +dependencies = [ + "funty", + "radium", + "tap", + "wyz", +] + +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array", +] + +[[package]] +name = "bs58" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f5353f36341f7451062466f0b755b96ac3a9547e4d7f6b70d603fc721a7d7896" +dependencies = [ + "sha2", + "tinyvec", +] + +[[package]] +name = "bumpalo" +version = "3.15.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ea184aa71bb362a1157c896979544cc23974e08fd265f29ea96b59f0b4a555b" + +[[package]] +name = "byte-slice-cast" +version = "1.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3ac9f8b63eca6fd385229b3675f6cc0dc5c8a5c8a54a59d4f52ffd670d87b0c" + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + +[[package]] +name = "bytes" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2bd12c1caf447e69cd4528f47f94d203fd2582878ecb9e9465484c4148a8223" +dependencies = [ + "serde", +] + +[[package]] +name = "bzip2" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bdb116a6ef3f6c3698828873ad02c3014b3c85cadb88496095628e3ef1e347f8" +dependencies = [ + "bzip2-sys", + "libc", +] + +[[package]] +name = "bzip2-sys" +version = "0.1.11+1.0.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "736a955f3fa7875102d57c82b8cac37ec45224a07fd32d58f9f7a186b6cd4cdc" +dependencies = [ + "cc", + "libc", + "pkg-config", +] + +[[package]] +name = "camino" +version = "1.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c59e92b5a388f549b863a7bea62612c09f24c8393560709a54558a9abdfb3b9c" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo-platform" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "694c8807f2ae16faecc43dc17d74b3eb042482789fd0eb64b39a2e04e087053f" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo_metadata" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d886547e41f740c616ae73108f6eb70afe6d940c7bc697cb30f13daec073037" +dependencies = [ + "camino", + "cargo-platform", + "semver", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "cc" +version = "1.0.88" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02f341c093d19155a6e41631ce5971aac4e9a868262212153124c15fa22d1cdc" +dependencies = [ + "libc", +] + +[[package]] +name = "cfg-if" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" + +[[package]] +name = "chrono" +version = "0.4.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5bc015644b92d5890fab7489e49d21f879d5c990186827d42ec511919404f38b" +dependencies = [ + "num-traits", +] + +[[package]] +name = "cipher" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad" +dependencies = [ + "crypto-common", + "inout", +] + +[[package]] +name = "clap" +version = "4.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c918d541ef2913577a0f9566e9ce27cb35b6df072075769e0b26cb5a554520da" +dependencies = [ + "clap_builder", + "clap_derive", +] + +[[package]] +name = "clap_builder" +version = "4.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f3e7391dad68afb0c2ede1bf619f579a3dc9c2ec67f089baa397123a2f3d1eb" +dependencies = [ + "anstream", + "anstyle", + "clap_lex", + "strsim", + "terminal_size", +] + +[[package]] +name = "clap_derive" +version = "4.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "307bc0538d5f0f83b8248db3087aa92fe504e4691294d0c96c0eabc33f47ba47" +dependencies = [ + "heck", + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "clap_lex" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "98cc8fbded0c607b7ba9dd60cd98df59af97e84d24e49c8557331cfc26d301ce" + +[[package]] +name = "cliclack" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4febf49beeedc40528e4956995631f1bbdb4d8804ef940b44351f393a996c739" +dependencies = [ + "console", + "indicatif", + "once_cell", + "textwrap", + "zeroize", +] + +[[package]] +name = "coins-bip32" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b6be4a5df2098cd811f3194f64ddb96c267606bffd9689ac7b0160097b01ad3" +dependencies = [ + "bs58", + "coins-core", + "digest", + "hmac", + "k256", + "serde", + "sha2", + "thiserror", +] + +[[package]] +name = "coins-bip39" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3db8fba409ce3dc04f7d804074039eb68b960b0829161f8e06c95fea3f122528" +dependencies = [ + "bitvec", + "coins-bip32", + "hmac", + "once_cell", + "pbkdf2 0.12.2", + "rand", + "sha2", + "thiserror", +] + +[[package]] +name = "coins-core" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5286a0843c21f8367f7be734f89df9b822e0321d8bcce8d6e735aadff7d74979" +dependencies = [ + "base64 0.21.7", + "bech32", + "bs58", + "digest", + "generic-array", + "hex", + "ripemd", + "serde", + "serde_derive", + "sha2", + "sha3", + "thiserror", +] + +[[package]] +name = "colorchoice" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "acbf1af155f9b9ef647e42cdc158db4b64a1b61f743629225fde6f3e0be2a7c7" + +[[package]] +name = "common" +version = "0.1.0" +dependencies = [ + "anyhow", + "clap", + "cliclack", + "console", + "ethers", + "futures", + "once_cell", + "serde", + "serde_json", + "serde_yaml", + "sqlx", + "strum 0.26.2", + "strum_macros 0.26.2", + "toml", + "url", + "xshell", +] + +[[package]] +name = "console" +version = "0.15.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e1f83fc076bd6dd27517eacdf25fef6c4dfe5f1d7448bafaaf3a26f13b5e4eb" +dependencies = [ + "encode_unicode", + "lazy_static", + "libc", + "unicode-width", + "windows-sys 0.52.0", +] + +[[package]] +name = "const-hex" +version = "1.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ba00838774b4ab0233e355d26710fbfc8327a05c017f6dc4873f876d1f79f78" +dependencies = [ + "cfg-if", + "cpufeatures", + "hex", + "proptest", + "serde", +] + +[[package]] +name = "const-oid" +version = "0.9.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2459377285ad874054d797f3ccebf984978aa39129f6eafde5cdc8315b612f8" + +[[package]] +name = "constant_time_eq" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "245097e9a4535ee1e3e3931fcfcd55a796a44c643e8596ff6566d68f09b87bbc" + +[[package]] +name = "core-foundation" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "core-foundation-sys" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06ea2b9bc92be3c2baa9334a323ebca2d6f074ff852cd1d7b11064035cd3868f" + +[[package]] +name = "cpufeatures" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53fe5e26ff1b7aef8bca9c6080520cfb8d9333c7568e1829cef191a9723e5504" +dependencies = [ + "libc", +] + +[[package]] +name = "crc" +version = "3.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69e6e4d7b33a94f0991c26729976b10ebde1d34c3ee82408fb536164fa10d636" +dependencies = [ + "crc-catalog", +] + +[[package]] +name = "crc-catalog" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19d374276b40fb8bbdee95aef7c7fa6b5316ec764510eb64b8dd0e2ed0d7e7f5" + +[[package]] +name = "crc32fast" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b3855a8a784b474f333699ef2bbca9db2c4a1f6d9088a90a2d25b1eb53111eaa" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "crossbeam-deque" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d" +dependencies = [ + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-queue" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df0346b5d5e76ac2fe4e327c5fd1118d6be7c51dfb18f9b7922923f287471e35" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "248e3bacc7dc6baa3b21e405ee045c3047101a49145e7e9eca583ab4c2ca5345" + +[[package]] +name = "crunchy" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" + +[[package]] +name = "crypto-bigint" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0dc92fb57ca44df6db8059111ab3af99a63d5d0f8375d9972e319a379c6bab76" +dependencies = [ + "generic-array", + "rand_core", + "subtle", + "zeroize", +] + +[[package]] +name = "crypto-common" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" +dependencies = [ + "generic-array", + "typenum", +] + +[[package]] +name = "ctr" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0369ee1ad671834580515889b80f2ea915f23b8be8d0daa4bbaf2ac5c7590835" +dependencies = [ + "cipher", +] + +[[package]] +name = "data-encoding" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e962a19be5cfc3f3bf6dd8f61eb50107f356ad6270fbb3ed41476571db78be5" + +[[package]] +name = "der" +version = "0.7.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fffa369a668c8af7dbf8b5e56c9f744fbd399949ed171606040001947de40b1c" +dependencies = [ + "const-oid", + "pem-rfc7468", + "zeroize", +] + +[[package]] +name = "deranged" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4" +dependencies = [ + "powerfmt", +] + +[[package]] +name = "derive_more" +version = "0.99.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4fb810d30a7c1953f91334de7244731fc3f3c10d7fe163338a35b9f640960321" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "diff" +version = "0.1.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56254986775e3233ffa9c4d7d3faaf6d36a2c09d30b20687e9f88bc8bafc16c8" + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer", + "const-oid", + "crypto-common", + "subtle", +] + +[[package]] +name = "dirs" +version = "5.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44c45a9d03d6676652bcb5e724c7e988de1acad23a711b5217ab9cbecbec2225" +dependencies = [ + "dirs-sys", +] + +[[package]] +name = "dirs-next" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b98cf8ebf19c3d1b223e151f99a4f9f0690dca41414773390fc824184ac833e1" +dependencies = [ + "cfg-if", + "dirs-sys-next", +] + +[[package]] +name = "dirs-sys" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "520f05a5cbd335fae5a99ff7a6ab8627577660ee5cfd6a94a6a929b52ff0321c" +dependencies = [ + "libc", + "option-ext", + "redox_users", + "windows-sys 0.48.0", +] + +[[package]] +name = "dirs-sys-next" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ebda144c4fe02d1f7ea1a7d9641b6fc6b580adcfa024ae48797ecdeb6825b4d" +dependencies = [ + "libc", + "redox_users", + "winapi", +] + +[[package]] +name = "dotenvy" +version = "0.15.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b" + +[[package]] +name = "dunce" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56ce8c6da7551ec6c462cbaf3bfbc75131ebbfa1c944aeaa9dab51ca1c5f0c3b" + +[[package]] +name = "ecdsa" +version = "0.16.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee27f32b5c5292967d2d4a9d7f1e0b0aed2c15daded5a60300e4abb9d8020bca" +dependencies = [ + "der", + "digest", + "elliptic-curve", + "rfc6979", + "signature", + "spki", +] + +[[package]] +name = "either" +version = "1.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11157ac094ffbdde99aa67b23417ebdd801842852b500e395a45a9c0aac03e4a" +dependencies = [ + "serde", +] + +[[package]] +name = "elliptic-curve" +version = "0.13.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5e6043086bf7973472e0c7dff2142ea0b680d30e18d9cc40f267efbf222bd47" +dependencies = [ + "base16ct", + "crypto-bigint", + "digest", + "ff", + "generic-array", + "group", + "pkcs8", + "rand_core", + "sec1", + "subtle", + "zeroize", +] + +[[package]] +name = "ena" +version = "0.14.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c533630cf40e9caa44bd91aadc88a75d75a4c3a12b4cfde353cbed41daa1e1f1" +dependencies = [ + "log", +] + +[[package]] +name = "encode_unicode" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a357d28ed41a50f9c765dbfe56cbc04a64e53e5fc58ba79fbc34c10ef3df831f" + +[[package]] +name = "encoding_rs" +version = "0.8.33" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7268b386296a025e474d5140678f75d6de9493ae55a5d709eeb9dd08149945e1" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "enr" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fe81b5c06ecfdbc71dd845216f225f53b62a10cb8a16c946836a3467f701d05b" +dependencies = [ + "base64 0.21.7", + "bytes", + "hex", + "k256", + "log", + "rand", + "rlp", + "serde", + "sha3", + "zeroize", +] + +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + +[[package]] +name = "errno" +version = "0.3.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a258e46cdc063eb8519c00b9fc845fc47bcfca4130e2f08e88665ceda8474245" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "etcetera" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "136d1b5283a1ab77bd9257427ffd09d8667ced0570b6f938942bc7568ed5b943" +dependencies = [ + "cfg-if", + "home", + "windows-sys 0.48.0", +] + +[[package]] +name = "eth-keystore" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fda3bf123be441da5260717e0661c25a2fd9cb2b2c1d20bf2e05580047158ab" +dependencies = [ + "aes", + "ctr", + "digest", + "hex", + "hmac", + "pbkdf2 0.11.0", + "rand", + "scrypt", + "serde", + "serde_json", + "sha2", + "sha3", + "thiserror", + "uuid 0.8.2", +] + +[[package]] +name = "ethabi" +version = "18.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7413c5f74cc903ea37386a8965a936cbeb334bd270862fdece542c1b2dcbc898" +dependencies = [ + "ethereum-types", + "hex", + "once_cell", + "regex", + "serde", + "serde_json", + "sha3", + "thiserror", + "uint", +] + +[[package]] +name = "ethbloom" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c22d4b5885b6aa2fe5e8b9329fb8d232bf739e434e6b87347c63bdd00c120f60" +dependencies = [ + "crunchy", + "fixed-hash", + "impl-codec", + "impl-rlp", + "impl-serde", + "scale-info", + "tiny-keccak", +] + +[[package]] +name = "ethereum-types" +version = "0.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02d215cbf040552efcbe99a38372fe80ab9d00268e20012b79fcd0f073edd8ee" +dependencies = [ + "ethbloom", + "fixed-hash", + "impl-codec", + "impl-rlp", + "impl-serde", + "primitive-types", + "scale-info", + "uint", +] + +[[package]] +name = "ethers" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c7cd562832e2ff584fa844cd2f6e5d4f35bbe11b28c7c9b8df957b2e1d0c701" +dependencies = [ + "ethers-addressbook", + "ethers-contract", + "ethers-core", + "ethers-etherscan", + "ethers-middleware", + "ethers-providers", + "ethers-signers", + "ethers-solc", +] + +[[package]] +name = "ethers-addressbook" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "35dc9a249c066d17e8947ff52a4116406163cf92c7f0763cb8c001760b26403f" +dependencies = [ + "ethers-core", + "once_cell", + "serde", + "serde_json", +] + +[[package]] +name = "ethers-contract" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43304317c7f776876e47f2f637859f6d0701c1ec7930a150f169d5fbe7d76f5a" +dependencies = [ + "const-hex", + "ethers-contract-abigen", + "ethers-contract-derive", + "ethers-core", + "ethers-providers", + "futures-util", + "once_cell", + "pin-project", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "ethers-contract-abigen" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9f96502317bf34f6d71a3e3d270defaa9485d754d789e15a8e04a84161c95eb" +dependencies = [ + "Inflector", + "const-hex", + "dunce", + "ethers-core", + "ethers-etherscan", + "eyre", + "prettyplease", + "proc-macro2", + "quote", + "regex", + "reqwest", + "serde", + "serde_json", + "syn 2.0.51", + "toml", + "walkdir", +] + +[[package]] +name = "ethers-contract-derive" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "452ff6b0a64507ce8d67ffd48b1da3b42f03680dcf5382244e9c93822cbbf5de" +dependencies = [ + "Inflector", + "const-hex", + "ethers-contract-abigen", + "ethers-core", + "proc-macro2", + "quote", + "serde_json", + "syn 2.0.51", +] + +[[package]] +name = "ethers-core" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aab3cef6cc1c9fd7f787043c81ad3052eff2b96a3878ef1526aa446311bdbfc9" +dependencies = [ + "arrayvec", + "bytes", + "cargo_metadata", + "chrono", + "const-hex", + "elliptic-curve", + "ethabi", + "generic-array", + "k256", + "num_enum", + "once_cell", + "open-fastrlp", + "rand", + "rlp", + "serde", + "serde_json", + "strum 0.25.0", + "syn 2.0.51", + "tempfile", + "thiserror", + "tiny-keccak", + "unicode-xid", +] + +[[package]] +name = "ethers-etherscan" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "16d45b981f5fa769e1d0343ebc2a44cfa88c9bc312eb681b676318b40cef6fb1" +dependencies = [ + "chrono", + "ethers-core", + "reqwest", + "semver", + "serde", + "serde_json", + "thiserror", + "tracing", +] + +[[package]] +name = "ethers-middleware" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "145211f34342487ef83a597c1e69f0d3e01512217a7c72cc8a25931854c7dca0" +dependencies = [ + "async-trait", + "auto_impl", + "ethers-contract", + "ethers-core", + "ethers-etherscan", + "ethers-providers", + "ethers-signers", + "futures-channel", + "futures-locks", + "futures-util", + "instant", + "reqwest", + "serde", + "serde_json", + "thiserror", + "tokio", + "tracing", + "tracing-futures", + "url", +] + +[[package]] +name = "ethers-providers" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fb6b15393996e3b8a78ef1332d6483c11d839042c17be58decc92fa8b1c3508a" +dependencies = [ + "async-trait", + "auto_impl", + "base64 0.21.7", + "bytes", + "const-hex", + "enr", + "ethers-core", + "futures-core", + "futures-timer", + "futures-util", + "hashers", + "http", + "instant", + "jsonwebtoken", + "once_cell", + "pin-project", + "reqwest", + "serde", + "serde_json", + "thiserror", + "tokio", + "tokio-tungstenite", + "tracing", + "tracing-futures", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "ws_stream_wasm", +] + +[[package]] +name = "ethers-signers" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b3b125a103b56aef008af5d5fb48191984aa326b50bfd2557d231dc499833de3" +dependencies = [ + "async-trait", + "coins-bip32", + "coins-bip39", + "const-hex", + "elliptic-curve", + "eth-keystore", + "ethers-core", + "rand", + "sha2", + "thiserror", + "tracing", +] + +[[package]] +name = "ethers-solc" +version = "2.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d21df08582e0a43005018a858cc9b465c5fff9cf4056651be64f844e57d1f55f" +dependencies = [ + "cfg-if", + "const-hex", + "dirs", + "dunce", + "ethers-core", + "glob", + "home", + "md-5", + "num_cpus", + "once_cell", + "path-slash", + "rayon", + "regex", + "semver", + "serde", + "serde_json", + "solang-parser", + "svm-rs", + "thiserror", + "tiny-keccak", + "tokio", + "tracing", + "walkdir", + "yansi", +] + +[[package]] +name = "event-listener" +version = "2.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0206175f82b8d6bf6652ff7d71a1e27fd2e4efde587fd368662814d6ec1d9ce0" + +[[package]] +name = "eyre" +version = "0.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7cd915d99f24784cdc19fd37ef22b97e3ff0ae756c7e492e9fbfe897d61e2aec" +dependencies = [ + "indenter", + "once_cell", +] + +[[package]] +name = "fastrand" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "25cbce373ec4653f1a01a31e8a5e5ec0c622dc27ff9c4e6606eefef5cbbed4a5" + +[[package]] +name = "ff" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ded41244b729663b1e574f1b4fb731469f69f79c17667b5d776b16cda0479449" +dependencies = [ + "rand_core", + "subtle", +] + +[[package]] +name = "finl_unicode" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fcfdc7a0362c9f4444381a9e697c79d435fe65b52a37466fc2c1184cee9edc6" + +[[package]] +name = "fixed-hash" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "835c052cb0c08c1acf6ffd71c022172e18723949c8282f2b9f27efbc51e64534" +dependencies = [ + "byteorder", + "rand", + "rustc-hex", + "static_assertions", +] + +[[package]] +name = "fixedbitset" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ce7134b9999ecaf8bcd65542e436736ef32ddca1b3e06094cb6ec5755203b80" + +[[package]] +name = "flate2" +version = "1.0.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46303f565772937ffe1d394a4fac6f411c6013172fadde9dcdb1e147a086940e" +dependencies = [ + "crc32fast", + "miniz_oxide", +] + +[[package]] +name = "flume" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55ac459de2512911e4b674ce33cf20befaba382d05b62b008afc1c8b57cbf181" +dependencies = [ + "futures-core", + "futures-sink", + "spin 0.9.8", +] + +[[package]] +name = "fnv" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" + +[[package]] +name = "form_urlencoded" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456" +dependencies = [ + "percent-encoding", +] + +[[package]] +name = "fs2" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9564fc758e15025b46aa6643b1b77d047d1a56a1aea6e01002ac0c7026876213" +dependencies = [ + "libc", + "winapi", +] + +[[package]] +name = "funty" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c" + +[[package]] +name = "futures" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "645c6916888f6cb6350d2550b80fb63e734897a8498abe35cfb732b6487804b0" +dependencies = [ + "futures-channel", + "futures-core", + "futures-executor", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-channel" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eac8f7d7865dcb88bd4373ab671c8cf4508703796caa2b1985a9ca867b3fcb78" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[package]] +name = "futures-core" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dfc6580bb841c5a68e9ef15c77ccc837b40a7504914d52e47b8b0e9bbda25a1d" + +[[package]] +name = "futures-executor" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a576fc72ae164fca6b9db127eaa9a9dda0d61316034f33a0a0d4eda41f02b01d" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-intrusive" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d930c203dd0b6ff06e0201a4a2fe9149b43c684fd4420555b26d21b1a02956f" +dependencies = [ + "futures-core", + "lock_api", + "parking_lot", +] + +[[package]] +name = "futures-io" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" + +[[package]] +name = "futures-locks" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45ec6fe3675af967e67c5536c0b9d44e34e6c52f86bedc4ea49c5317b8e94d06" +dependencies = [ + "futures-channel", + "futures-task", +] + +[[package]] +name = "futures-macro" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "futures-sink" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9fb8e00e87438d937621c1c6269e53f536c14d3fbd6a042bb24879e57d474fb5" + +[[package]] +name = "futures-task" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38d84fa142264698cdce1a9f9172cf383a0c82de1bddcf3092901442c4097004" + +[[package]] +name = "futures-timer" +version = "3.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f288b0a4f20f9a56b5d1da57e2227c661b7b16168e2f72365f57b63326e29b24" +dependencies = [ + "gloo-timers", + "send_wrapper 0.4.0", +] + +[[package]] +name = "futures-util" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "pin-utils", + "slab", +] + +[[package]] +name = "fxhash" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c" +dependencies = [ + "byteorder", +] + +[[package]] +name = "generic-array" +version = "0.14.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" +dependencies = [ + "typenum", + "version_check", + "zeroize", +] + +[[package]] +name = "getrandom" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "190092ea657667030ac6a35e305e62fc4dd69fd98ac98631e5d3a2b1575a12b5" +dependencies = [ + "cfg-if", + "libc", + "wasi", +] + +[[package]] +name = "gimli" +version = "0.28.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4271d37baee1b8c7e4b708028c57d816cf9d2434acb33a549475f78c181f6253" + +[[package]] +name = "glob" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b" + +[[package]] +name = "gloo-timers" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b995a66bb87bebce9a0f4a95aed01daca4872c050bfcb21653361c03bc35e5c" +dependencies = [ + "futures-channel", + "futures-core", + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "group" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0f9ef7462f7c099f518d754361858f86d8a07af53ba9af0fe635bbccb151a63" +dependencies = [ + "ff", + "rand_core", + "subtle", +] + +[[package]] +name = "h2" +version = "0.3.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81fe527a889e1532da5c525686d96d4c2e74cdd345badf8dfef9f6b39dd5f5e8" +dependencies = [ + "bytes", + "fnv", + "futures-core", + "futures-sink", + "futures-util", + "http", + "indexmap", + "slab", + "tokio", + "tokio-util", + "tracing", +] + +[[package]] +name = "hashbrown" +version = "0.14.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "290f1a1d9242c78d09ce40a5e87e7554ee637af1351968159f4952f028f75604" +dependencies = [ + "ahash", + "allocator-api2", +] + +[[package]] +name = "hashers" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2bca93b15ea5a746f220e56587f71e73c6165eab783df9e26590069953e3c30" +dependencies = [ + "fxhash", +] + +[[package]] +name = "hashlink" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8094feaf31ff591f651a2664fb9cfd92bba7a60ce3197265e9482ebe753c8f7" +dependencies = [ + "hashbrown", +] + +[[package]] +name = "heck" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8" +dependencies = [ + "unicode-segmentation", +] + +[[package]] +name = "hermit-abi" +version = "0.3.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "379dada1584ad501b383485dd706b8afb7a70fcbc7f4da7d780638a5a6124a60" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "hkdf" +version = "0.12.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b5f8eb2ad728638ea2c7d47a21db23b7b58a72ed6a38256b8a1849f15fbbdf7" +dependencies = [ + "hmac", +] + +[[package]] +name = "hmac" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" +dependencies = [ + "digest", +] + +[[package]] +name = "home" +version = "0.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3d1354bf6b7235cb4a0576c2619fd4ed18183f689b12b006a0ee7329eeff9a5" +dependencies = [ + "windows-sys 0.52.0", +] + +[[package]] +name = "http" +version = "0.2.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8947b1a6fad4393052c7ba1f4cd97bed3e953a95c79c92ad9b051a04611d9fbb" +dependencies = [ + "bytes", + "fnv", + "itoa", +] + +[[package]] +name = "http-body" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2" +dependencies = [ + "bytes", + "http", + "pin-project-lite", +] + +[[package]] +name = "httparse" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d897f394bad6a705d5f4104762e116a75639e470d80901eed05a860a95cb1904" + +[[package]] +name = "httpdate" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" + +[[package]] +name = "human-panic" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4c5d0e9120f6bca6120d142c7ede1ba376dd6bf276d69dd3dbe6cbeb7824179" +dependencies = [ + "anstream", + "anstyle", + "backtrace", + "os_info", + "serde", + "serde_derive", + "toml", + "uuid 1.8.0", +] + +[[package]] +name = "hyper" +version = "0.14.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf96e135eb83a2a8ddf766e426a841d8ddd7449d5f00d34ea02b41d2f19eef80" +dependencies = [ + "bytes", + "futures-channel", + "futures-core", + "futures-util", + "h2", + "http", + "http-body", + "httparse", + "httpdate", + "itoa", + "pin-project-lite", + "socket2", + "tokio", + "tower-service", + "tracing", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.24.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec3efd23720e2049821a693cbc7e65ea87c72f1c58ff2f9522ff332b1491e590" +dependencies = [ + "futures-util", + "http", + "hyper", + "rustls", + "tokio", + "tokio-rustls", +] + +[[package]] +name = "idna" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +dependencies = [ + "unicode-bidi", + "unicode-normalization", +] + +[[package]] +name = "impl-codec" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba6a270039626615617f3f36d15fc827041df3b78c439da2cadfa47455a77f2f" +dependencies = [ + "parity-scale-codec", +] + +[[package]] +name = "impl-rlp" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f28220f89297a075ddc7245cd538076ee98b01f2a9c23a53a4f1105d5a322808" +dependencies = [ + "rlp", +] + +[[package]] +name = "impl-serde" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebc88fc67028ae3db0c853baa36269d398d5f45b6982f95549ff5def78c935cd" +dependencies = [ + "serde", +] + +[[package]] +name = "impl-trait-for-tuples" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11d7a9f6330b71fea57921c9b61c47ee6e84f72d394754eff6163ae67e7395eb" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "indenter" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce23b50ad8242c51a442f3ff322d56b02f08852c77e4c0b4d3fd684abc89c683" + +[[package]] +name = "indexmap" +version = "2.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "233cf39063f058ea2caae4091bf4a3ef70a653afbc026f5c4a4135d114e3c177" +dependencies = [ + "equivalent", + "hashbrown", +] + +[[package]] +name = "indicatif" +version = "0.17.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "763a5a8f45087d6bcea4222e7b72c291a054edf80e4ef6efd2a4979878c7bea3" +dependencies = [ + "console", + "instant", + "number_prefix", + "portable-atomic", + "unicode-width", +] + +[[package]] +name = "inout" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a0c10553d664a4d0bcff9f4215d0aac67a639cc68ef660840afe309b807bc9f5" +dependencies = [ + "generic-array", +] + +[[package]] +name = "instant" +version = "0.1.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "ipnet" +version = "2.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f518f335dce6725a761382244631d86cf0ccb2863413590b31338feb467f9c3" + +[[package]] +name = "is-terminal" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f23ff5ef2b80d608d61efee834934d862cd92461afc0560dedf493e4c033738b" +dependencies = [ + "hermit-abi", + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "itertools" +version = "0.10.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b0fd2260e829bddf4cb6ea802289de2f86d6a7a690192fbe91b3f46e0f2c8473" +dependencies = [ + "either", +] + +[[package]] +name = "itertools" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1c173a5686ce8bfa551b3563d0c2170bf24ca44da99c7ca4bfdab5418c3fe57" +dependencies = [ + "either", +] + +[[package]] +name = "itertools" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba291022dbbd398a455acf126c1e341954079855bc60dfdda641363bd6922569" +dependencies = [ + "either", +] + +[[package]] +name = "itoa" +version = "1.0.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1a46d1a171d865aa5f83f92695765caa047a9b4cbae2cbf37dbd613a793fd4c" + +[[package]] +name = "js-sys" +version = "0.3.68" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "406cda4b368d531c842222cf9d2600a9a4acce8d29423695379c6868a143a9ee" +dependencies = [ + "wasm-bindgen", +] + +[[package]] +name = "jsonwebtoken" +version = "8.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6971da4d9c3aa03c3d8f3ff0f4155b534aad021292003895a469716b2a230378" +dependencies = [ + "base64 0.21.7", + "pem", + "ring 0.16.20", + "serde", + "serde_json", + "simple_asn1", +] + +[[package]] +name = "k256" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cadb76004ed8e97623117f3df85b17aaa6626ab0b0831e6573f104df16cd1bcc" +dependencies = [ + "cfg-if", + "ecdsa", + "elliptic-curve", + "once_cell", + "sha2", + "signature", +] + +[[package]] +name = "keccak" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ecc2af9a1119c51f12a14607e783cb977bde58bc069ff0c3da1095e635d70654" +dependencies = [ + "cpufeatures", +] + +[[package]] +name = "lalrpop" +version = "0.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da4081d44f4611b66c6dd725e6de3169f9f63905421e8626fcb86b6a898998b8" +dependencies = [ + "ascii-canvas", + "bit-set", + "diff", + "ena", + "is-terminal", + "itertools 0.10.5", + "lalrpop-util", + "petgraph", + "regex", + "regex-syntax 0.7.5", + "string_cache", + "term", + "tiny-keccak", + "unicode-xid", +] + +[[package]] +name = "lalrpop-util" +version = "0.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f35c735096c0293d313e8f2a641627472b83d01b937177fe76e5e2708d31e0d" + +[[package]] +name = "lazy_static" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" +dependencies = [ + "spin 0.5.2", +] + +[[package]] +name = "libc" +version = "0.2.153" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c198f91728a82281a64e1f4f9eeb25d82cb32a5de251c6bd1b5154d63a8e7bd" + +[[package]] +name = "libm" +version = "0.2.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ec2a862134d2a7d32d7983ddcdd1c4923530833c9f2ea1a44fc5fa473989058" + +[[package]] +name = "libredox" +version = "0.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85c833ca1e66078851dba29046874e38f08b2c883700aa29a03ddd3b23814ee8" +dependencies = [ + "bitflags 2.4.2", + "libc", + "redox_syscall", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.27.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf4e226dcd58b4be396f7bd3c20da8fdee2911400705297ba7d2d7cc2c30f716" +dependencies = [ + "cc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "linux-raw-sys" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "01cda141df6706de531b6c46c3a33ecca755538219bd484262fa09410c13539c" + +[[package]] +name = "lock_api" +version = "0.4.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c168f8615b12bc01f9c17e2eb0cc07dcae1940121185446edc3744920e8ef45" +dependencies = [ + "autocfg", + "scopeguard", +] + +[[package]] +name = "log" +version = "0.4.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5e6163cb8c49088c2c36f57875e58ccd8c87c7427f7fbd50ea6710b2f3f2e8f" + +[[package]] +name = "md-5" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d89e7ee0cfbedfc4da3340218492196241d89eefb6dab27de5df917a6d2e78cf" +dependencies = [ + "cfg-if", + "digest", +] + +[[package]] +name = "memchr" +version = "2.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "523dc4f511e55ab87b694dc30d0f820d60906ef06413f93d4d7a1385599cc149" + +[[package]] +name = "mime" +version = "0.3.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" + +[[package]] +name = "minimal-lexical" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a" + +[[package]] +name = "miniz_oxide" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d811f3e15f28568be3407c8e7fdb6514c1cda3cb30683f15b6a1a1dc4ea14a7" +dependencies = [ + "adler", +] + +[[package]] +name = "mio" +version = "0.8.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" +dependencies = [ + "libc", + "wasi", + "windows-sys 0.48.0", +] + +[[package]] +name = "new_debug_unreachable" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e4a24736216ec316047a1fc4252e27dabb04218aa4a3f37c6e7ddbf1f9782b54" + +[[package]] +name = "nom" +version = "7.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d273983c5a657a70a3e8f2a01329822f3b8c8172b73826411a55751e404a0a4a" +dependencies = [ + "memchr", + "minimal-lexical", +] + +[[package]] +name = "num-bigint" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "608e7659b5c3d7cba262d894801b9ec9d00de989e8a82bd4bef91d08da45cdc0" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-bigint-dig" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc84195820f291c7697304f3cbdadd1cb7199c0efc917ff5eafd71225c136151" +dependencies = [ + "byteorder", + "lazy_static", + "libm", + "num-integer", + "num-iter", + "num-traits", + "rand", + "smallvec", + "zeroize", +] + +[[package]] +name = "num-conv" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9" + +[[package]] +name = "num-integer" +version = "0.1.46" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f" +dependencies = [ + "num-traits", +] + +[[package]] +name = "num-iter" +version = "0.1.44" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d869c01cc0c455284163fd0092f1f93835385ccab5a98a0dcc497b2f8bf055a9" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-traits" +version = "0.2.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da0df0e5185db44f69b44f26786fe401b6c293d1907744beaa7fa62b2e5a517a" +dependencies = [ + "autocfg", + "libm", +] + +[[package]] +name = "num_cpus" +version = "1.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" +dependencies = [ + "hermit-abi", + "libc", +] + +[[package]] +name = "num_enum" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02339744ee7253741199f897151b38e72257d13802d4ee837285cc2990a90845" +dependencies = [ + "num_enum_derive", +] + +[[package]] +name = "num_enum_derive" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "681030a937600a36906c185595136d26abfebb4aa9c65701cefcaf8578bb982b" +dependencies = [ + "proc-macro-crate 3.1.0", + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "number_prefix" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "830b246a0e5f20af87141b25c173cd1b609bd7779a4617d6ec582abaf90870f3" + +[[package]] +name = "object" +version = "0.32.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6a622008b6e321afc04970976f62ee297fdbaa6f95318ca343e3eebb9648441" +dependencies = [ + "memchr", +] + +[[package]] +name = "once_cell" +version = "1.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" + +[[package]] +name = "open-fastrlp" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "786393f80485445794f6043fd3138854dd109cc6c4bd1a6383db304c9ce9b9ce" +dependencies = [ + "arrayvec", + "auto_impl", + "bytes", + "ethereum-types", + "open-fastrlp-derive", +] + +[[package]] +name = "open-fastrlp-derive" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "003b2be5c6c53c1cfeb0a238b8a1c3915cd410feb684457a36c10038f764bb1c" +dependencies = [ + "bytes", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "option-ext" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" + +[[package]] +name = "os_info" +version = "3.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae99c7fa6dd38c7cafe1ec085e804f8f555a2f8659b0dbe03f1f9963a9b51092" +dependencies = [ + "log", + "serde", + "windows-sys 0.52.0", +] + +[[package]] +name = "parity-scale-codec" +version = "3.6.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "881331e34fa842a2fb61cc2db9643a8fedc615e47cfcc52597d1af0db9a7e8fe" +dependencies = [ + "arrayvec", + "bitvec", + "byte-slice-cast", + "impl-trait-for-tuples", + "parity-scale-codec-derive", + "serde", +] + +[[package]] +name = "parity-scale-codec-derive" +version = "3.6.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be30eaf4b0a9fba5336683b38de57bb86d179a35862ba6bfcf57625d006bde5b" +dependencies = [ + "proc-macro-crate 2.0.0", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "parking_lot" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c42a9226546d68acdd9c0a280d17ce19bfe27a46bf68784e4066115788d008e" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-targets 0.48.5", +] + +[[package]] +name = "password-hash" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7676374caaee8a325c9e7a2ae557f216c5563a171d6997b0ef8a65af35147700" +dependencies = [ + "base64ct", + "rand_core", + "subtle", +] + +[[package]] +name = "paste" +version = "1.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de3145af08024dea9fa9914f381a17b8fc6034dfb00f3a84013f7ff43f29ed4c" + +[[package]] +name = "path-slash" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e91099d4268b0e11973f036e885d652fb0b21fedcf69738c627f94db6a44f42" + +[[package]] +name = "pbkdf2" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "83a0692ec44e4cf1ef28ca317f14f8f07da2d95ec3fa01f86e4467b725e60917" +dependencies = [ + "digest", + "hmac", + "password-hash", + "sha2", +] + +[[package]] +name = "pbkdf2" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8ed6a7761f76e3b9f92dfb0a60a6a6477c61024b775147ff0973a02653abaf2" +dependencies = [ + "digest", + "hmac", +] + +[[package]] +name = "pem" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8835c273a76a90455d7344889b0964598e3316e2a79ede8e36f16bdcf2228b8" +dependencies = [ + "base64 0.13.1", +] + +[[package]] +name = "pem-rfc7468" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "88b39c9bfcfc231068454382784bb460aae594343fb030d46e9f50a645418412" +dependencies = [ + "base64ct", +] + +[[package]] +name = "percent-encoding" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" + +[[package]] +name = "petgraph" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1d3afd2628e69da2be385eb6f2fd57c8ac7977ceeff6dc166ff1657b0e386a9" +dependencies = [ + "fixedbitset", + "indexmap", +] + +[[package]] +name = "pharos" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e9567389417feee6ce15dd6527a8a1ecac205ef62c2932bcf3d9f6fc5b78b414" +dependencies = [ + "futures", + "rustc_version", +] + +[[package]] +name = "phf" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ade2d8b8f33c7333b51bcf0428d37e217e9f32192ae4772156f65063b8ce03dc" +dependencies = [ + "phf_macros", + "phf_shared 0.11.2", +] + +[[package]] +name = "phf_generator" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48e4cc64c2ad9ebe670cb8fd69dd50ae301650392e81c05f9bfcb2d5bdbc24b0" +dependencies = [ + "phf_shared 0.11.2", + "rand", +] + +[[package]] +name = "phf_macros" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3444646e286606587e49f3bcf1679b8cef1dc2c5ecc29ddacaffc305180d464b" +dependencies = [ + "phf_generator", + "phf_shared 0.11.2", + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "phf_shared" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6796ad771acdc0123d2a88dc428b5e38ef24456743ddb1744ed628f9815c096" +dependencies = [ + "siphasher", +] + +[[package]] +name = "phf_shared" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90fcb95eef784c2ac79119d1dd819e162b5da872ce6f3c3abe1e8ca1c082f72b" +dependencies = [ + "siphasher", +] + +[[package]] +name = "pin-project" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0302c4a0442c456bd56f841aee5c3bfd17967563f6fadc9ceb9f9c23cf3807e0" +dependencies = [ + "pin-project-internal", +] + +[[package]] +name = "pin-project-internal" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "266c042b60c9c76b8d53061e52b2e0d1116abc57cefc8c5cd671619a56ac3690" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "pin-project-lite" +version = "0.2.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8afb450f006bf6385ca15ef45d71d2288452bc3683ce2e2cacc0d18e4be60b58" + +[[package]] +name = "pin-utils" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" + +[[package]] +name = "pkcs1" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8ffb9f10fa047879315e6625af03c164b16962a5368d724ed16323b68ace47f" +dependencies = [ + "der", + "pkcs8", + "spki", +] + +[[package]] +name = "pkcs8" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f950b2377845cebe5cf8b5165cb3cc1a5e0fa5cfa3e1f7f55707d8fd82e0a7b7" +dependencies = [ + "der", + "spki", +] + +[[package]] +name = "pkg-config" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d231b230927b5e4ad203db57bbcbee2802f6bce620b1e4a9024a07d94e2907ec" + +[[package]] +name = "portable-atomic" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7170ef9988bc169ba16dd36a7fa041e5c4cbeb6a35b76d4c03daded371eae7c0" + +[[package]] +name = "powerfmt" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" + +[[package]] +name = "ppv-lite86" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de" + +[[package]] +name = "precomputed-hash" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "925383efa346730478fb4838dbe9137d2a47675ad789c546d150a6e1dd4ab31c" + +[[package]] +name = "prettyplease" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a41cf62165e97c7f814d2221421dbb9afcbcdb0a88068e5ea206e19951c2cbb5" +dependencies = [ + "proc-macro2", + "syn 2.0.51", +] + +[[package]] +name = "primitive-types" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b34d9fd68ae0b74a41b21c03c2f62847aa0ffea044eee893b4c140b37e244e2" +dependencies = [ + "fixed-hash", + "impl-codec", + "impl-rlp", + "impl-serde", + "scale-info", + "uint", +] + +[[package]] +name = "proc-macro-crate" +version = "1.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f4c021e1093a56626774e81216a4ce732a735e5bad4868a03f3ed65ca0c3919" +dependencies = [ + "once_cell", + "toml_edit 0.19.15", +] + +[[package]] +name = "proc-macro-crate" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8" +dependencies = [ + "toml_edit 0.20.7", +] + +[[package]] +name = "proc-macro-crate" +version = "3.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d37c51ca738a55da99dc0c4a34860fd675453b8b36209178c2249bb13651284" +dependencies = [ + "toml_edit 0.21.1", +] + +[[package]] +name = "proc-macro2" +version = "1.0.78" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2422ad645d89c99f8f3e6b88a9fdeca7fabeac836b1002371c4367c8f984aae" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "proptest" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31b476131c3c86cb68032fdc5cb6d5a1045e3e42d96b69fa599fd77701e1f5bf" +dependencies = [ + "bitflags 2.4.2", + "lazy_static", + "num-traits", + "rand", + "rand_chacha", + "rand_xorshift", + "regex-syntax 0.8.2", + "unarray", +] + +[[package]] +name = "quote" +version = "1.0.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "291ec9ab5efd934aaf503a6466c5d5251535d108ee747472c3977cc5acc868ef" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "radium" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc33ff2d4973d518d823d61aa239014831e521c75da58e3df4840d3f47749d09" + +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha", + "rand_core", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core", +] + +[[package]] +name = "rand_core" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" +dependencies = [ + "getrandom", +] + +[[package]] +name = "rand_xorshift" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d25bf25ec5ae4a3f1b92f929810509a2f53d7dca2f50b794ff57e3face536c8f" +dependencies = [ + "rand_core", +] + +[[package]] +name = "rayon" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa7237101a77a10773db45d62004a272517633fbcc3df19d96455ede1122e051" +dependencies = [ + "either", + "rayon-core", +] + +[[package]] +name = "rayon-core" +version = "1.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2" +dependencies = [ + "crossbeam-deque", + "crossbeam-utils", +] + +[[package]] +name = "redox_syscall" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" +dependencies = [ + "bitflags 1.3.2", +] + +[[package]] +name = "redox_users" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a18479200779601e498ada4e8c1e1f50e3ee19deb0259c25825a98b5603b2cb4" +dependencies = [ + "getrandom", + "libredox", + "thiserror", +] + +[[package]] +name = "regex" +version = "1.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b62dbe01f0b06f9d8dc7d49e05a0785f153b00b2c227856282f671e0318c9b15" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata", + "regex-syntax 0.8.2", +] + +[[package]] +name = "regex-automata" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5bb987efffd3c6d0d8f5f89510bb458559eab11e4f869acb20bf845e016259cd" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax 0.8.2", +] + +[[package]] +name = "regex-syntax" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dbb5fb1acd8a1a18b3dd5be62d25485eb770e05afb408a9627d14d451bae12da" + +[[package]] +name = "regex-syntax" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c08c74e62047bb2de4ff487b251e4a92e24f48745648451635cec7d591162d9f" + +[[package]] +name = "reqwest" +version = "0.11.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6920094eb85afde5e4a138be3f2de8bbdf28000f0029e72c45025a56b042251" +dependencies = [ + "base64 0.21.7", + "bytes", + "encoding_rs", + "futures-core", + "futures-util", + "h2", + "http", + "http-body", + "hyper", + "hyper-rustls", + "ipnet", + "js-sys", + "log", + "mime", + "once_cell", + "percent-encoding", + "pin-project-lite", + "rustls", + "rustls-pemfile", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper", + "system-configuration", + "tokio", + "tokio-rustls", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "webpki-roots", + "winreg", +] + +[[package]] +name = "rfc6979" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8dd2a808d456c4a54e300a23e9f5a67e122c3024119acbfd73e3bf664491cb2" +dependencies = [ + "hmac", + "subtle", +] + +[[package]] +name = "ring" +version = "0.16.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3053cf52e236a3ed746dfc745aa9cacf1b791d846bdaf412f60a8d7d6e17c8fc" +dependencies = [ + "cc", + "libc", + "once_cell", + "spin 0.5.2", + "untrusted 0.7.1", + "web-sys", + "winapi", +] + +[[package]] +name = "ring" +version = "0.17.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c17fa4cb658e3583423e915b9f3acc01cceaee1860e33d59ebae66adc3a2dc0d" +dependencies = [ + "cc", + "cfg-if", + "getrandom", + "libc", + "spin 0.9.8", + "untrusted 0.9.0", + "windows-sys 0.52.0", +] + +[[package]] +name = "ripemd" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd124222d17ad93a644ed9d011a40f4fb64aa54275c08cc216524a9ea82fb09f" +dependencies = [ + "digest", +] + +[[package]] +name = "rlp" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb919243f34364b6bd2fc10ef797edbfa75f33c252e7998527479c6d6b47e1ec" +dependencies = [ + "bytes", + "rlp-derive", + "rustc-hex", +] + +[[package]] +name = "rlp-derive" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e33d7b2abe0c340d8797fe2907d3f20d3b5ea5908683618bfe80df7f621f672a" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "rsa" +version = "0.9.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d0e5124fcb30e76a7e79bfee683a2746db83784b86289f6251b54b7950a0dfc" +dependencies = [ + "const-oid", + "digest", + "num-bigint-dig", + "num-integer", + "num-traits", + "pkcs1", + "pkcs8", + "rand_core", + "signature", + "spki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustc-demangle" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d626bb9dae77e28219937af045c257c28bfd3f69333c512553507f5f9798cb76" + +[[package]] +name = "rustc-hex" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e75f6a532d0fd9f7f13144f392b6ad56a32696bfcd9c78f797f16bbb6f072d6" + +[[package]] +name = "rustc_version" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa0f585226d2e68097d4f95d113b15b83a82e819ab25717ec0590d9584ef366" +dependencies = [ + "semver", +] + +[[package]] +name = "rustix" +version = "0.38.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ea3e1a662af26cd7a3ba09c0297a31af215563ecf42817c98df621387f4e949" +dependencies = [ + "bitflags 2.4.2", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.52.0", +] + +[[package]] +name = "rustls" +version = "0.21.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f56a14d1f48b391359b22f731fd4bd7e43c97f3c50eee276f3aa09c94784d3e" +dependencies = [ + "log", + "ring 0.17.8", + "rustls-webpki", + "sct", +] + +[[package]] +name = "rustls-pemfile" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c74cae0a4cf6ccbbf5f359f08efdf8ee7e1dc532573bf0db71968cb56b1448c" +dependencies = [ + "base64 0.21.7", +] + +[[package]] +name = "rustls-webpki" +version = "0.101.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b6275d1ee7a1cd780b64aca7726599a1dbc893b1e64144529e55c3c2f745765" +dependencies = [ + "ring 0.17.8", + "untrusted 0.9.0", +] + +[[package]] +name = "rustversion" +version = "1.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ffc183a10b4478d04cbbbfc96d0873219d962dd5accaff2ffbd4ceb7df837f4" + +[[package]] +name = "ryu" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e86697c916019a8588c99b5fac3cead74ec0b4b819707a682fd4d23fa0ce1ba1" + +[[package]] +name = "salsa20" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97a22f5af31f73a954c10289c93e8a50cc23d971e80ee446f1f6f7137a088213" +dependencies = [ + "cipher", +] + +[[package]] +name = "same-file" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502" +dependencies = [ + "winapi-util", +] + +[[package]] +name = "scale-info" +version = "2.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f7d66a1128282b7ef025a8ead62a4a9fcf017382ec53b8ffbf4d7bf77bd3c60" +dependencies = [ + "cfg-if", + "derive_more", + "parity-scale-codec", + "scale-info-derive", +] + +[[package]] +name = "scale-info-derive" +version = "2.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "abf2c68b89cafb3b8d918dd07b42be0da66ff202cf1155c5739a4e0c1ea0dc19" +dependencies = [ + "proc-macro-crate 1.3.1", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + +[[package]] +name = "scrypt" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f9e24d2b632954ded8ab2ef9fea0a0c769ea56ea98bddbafbad22caeeadf45d" +dependencies = [ + "hmac", + "pbkdf2 0.11.0", + "salsa20", + "sha2", +] + +[[package]] +name = "sct" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da046153aa2352493d6cb7da4b6e5c0c057d8a1d0a9aa8560baffdd945acd414" +dependencies = [ + "ring 0.17.8", + "untrusted 0.9.0", +] + +[[package]] +name = "sec1" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3e97a565f76233a6003f9f5c54be1d9c5bdfa3eccfb189469f11ec4901c47dc" +dependencies = [ + "base16ct", + "der", + "generic-array", + "pkcs8", + "subtle", + "zeroize", +] + +[[package]] +name = "semver" +version = "1.0.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92d43fe69e652f3df9bdc2b85b2854a0825b86e4fb76bc44d945137d053639ca" +dependencies = [ + "serde", +] + +[[package]] +name = "send_wrapper" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f638d531eccd6e23b980caf34876660d38e265409d8e99b397ab71eb3612fad0" + +[[package]] +name = "send_wrapper" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd0b0ec5f1c1ca621c432a25813d8d60c88abe6d3e08a3eb9cf37d97a0fe3d73" + +[[package]] +name = "serde" +version = "1.0.197" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3fb1c873e1b9b056a4dc4c0c198b24c3ffa059243875552b2bd0933b1aee4ce2" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde_derive" +version = "1.0.197" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7eb0b34b42edc17f6b7cac84a52a1c5f0e1bb2227e997ca9011ea3dd34e8610b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "serde_json" +version = "1.0.114" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c5f09b1bd632ef549eaa9f60a1f8de742bdbc698e6cee2095fc84dde5f549ae0" +dependencies = [ + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "serde_spanned" +version = "0.6.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eb3622f419d1296904700073ea6cc23ad690adbd66f13ea683df73298736f0c1" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "serde_yaml" +version = "0.9.34+deprecated" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a8b1a1a2ebf674015cc02edccce75287f1a0130d394307b36743c2f5d504b47" +dependencies = [ + "indexmap", + "itoa", + "ryu", + "serde", + "unsafe-libyaml", +] + +[[package]] +name = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "sha2" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "sha3" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60" +dependencies = [ + "digest", + "keccak", +] + +[[package]] +name = "signal-hook-registry" +version = "1.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d8229b473baa5980ac72ef434c4415e70c4b5e71b423043adb4ba059f89c99a1" +dependencies = [ + "libc", +] + +[[package]] +name = "signature" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77549399552de45a898a580c1b41d445bf730df867cc44e6c0233bbc4b8329de" +dependencies = [ + "digest", + "rand_core", +] + +[[package]] +name = "simple_asn1" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "adc4e5204eb1910f40f9cfa375f6f05b68c3abac4b6fd879c8ff5e7ae8a0a085" +dependencies = [ + "num-bigint", + "num-traits", + "thiserror", + "time", +] + +[[package]] +name = "siphasher" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d" + +[[package]] +name = "slab" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67" +dependencies = [ + "autocfg", +] + +[[package]] +name = "smallvec" +version = "1.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6ecd384b10a64542d77071bd64bd7b231f4ed5940fba55e98c3de13824cf3d7" + +[[package]] +name = "smawk" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7c388c1b5e93756d0c740965c41e8822f866621d41acbdf6336a6a168f8840c" + +[[package]] +name = "socket2" +version = "0.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05ffd9c0a93b7543e062e759284fcf5f5e3b098501104bfbdde4d404db792871" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "solang-parser" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c425ce1c59f4b154717592f0bdf4715c3a1d55058883622d3157e1f0908a5b26" +dependencies = [ + "itertools 0.11.0", + "lalrpop", + "lalrpop-util", + "phf", + "thiserror", + "unicode-xid", +] + +[[package]] +name = "spin" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e63cff320ae2c57904679ba7cb63280a3dc4613885beafb148ee7bf9aa9042d" + +[[package]] +name = "spin" +version = "0.9.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" +dependencies = [ + "lock_api", +] + +[[package]] +name = "spki" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d91ed6c858b01f942cd56b37a94b3e0a1798290327d1236e4d9cf4eaca44d29d" +dependencies = [ + "base64ct", + "der", +] + +[[package]] +name = "sqlformat" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce81b7bd7c4493975347ef60d8c7e8b742d4694f4c49f93e0a12ea263938176c" +dependencies = [ + "itertools 0.12.1", + "nom", + "unicode_categories", +] + +[[package]] +name = "sqlx" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c9a2ccff1a000a5a59cd33da541d9f2fdcd9e6e8229cc200565942bff36d0aaa" +dependencies = [ + "sqlx-core", + "sqlx-macros", + "sqlx-mysql", + "sqlx-postgres", + "sqlx-sqlite", +] + +[[package]] +name = "sqlx-core" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24ba59a9342a3d9bab6c56c118be528b27c9b60e490080e9711a04dccac83ef6" +dependencies = [ + "ahash", + "atoi", + "byteorder", + "bytes", + "crc", + "crossbeam-queue", + "either", + "event-listener", + "futures-channel", + "futures-core", + "futures-intrusive", + "futures-io", + "futures-util", + "hashlink", + "hex", + "indexmap", + "log", + "memchr", + "once_cell", + "paste", + "percent-encoding", + "serde", + "serde_json", + "sha2", + "smallvec", + "sqlformat", + "thiserror", + "tokio", + "tokio-stream", + "tracing", + "url", +] + +[[package]] +name = "sqlx-macros" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ea40e2345eb2faa9e1e5e326db8c34711317d2b5e08d0d5741619048a803127" +dependencies = [ + "proc-macro2", + "quote", + "sqlx-core", + "sqlx-macros-core", + "syn 1.0.109", +] + +[[package]] +name = "sqlx-macros-core" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5833ef53aaa16d860e92123292f1f6a3d53c34ba8b1969f152ef1a7bb803f3c8" +dependencies = [ + "dotenvy", + "either", + "heck", + "hex", + "once_cell", + "proc-macro2", + "quote", + "serde", + "serde_json", + "sha2", + "sqlx-core", + "sqlx-mysql", + "sqlx-postgres", + "sqlx-sqlite", + "syn 1.0.109", + "tempfile", + "tokio", + "url", +] + +[[package]] +name = "sqlx-mysql" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1ed31390216d20e538e447a7a9b959e06ed9fc51c37b514b46eb758016ecd418" +dependencies = [ + "atoi", + "base64 0.21.7", + "bitflags 2.4.2", + "byteorder", + "bytes", + "crc", + "digest", + "dotenvy", + "either", + "futures-channel", + "futures-core", + "futures-io", + "futures-util", + "generic-array", + "hex", + "hkdf", + "hmac", + "itoa", + "log", + "md-5", + "memchr", + "once_cell", + "percent-encoding", + "rand", + "rsa", + "serde", + "sha1", + "sha2", + "smallvec", + "sqlx-core", + "stringprep", + "thiserror", + "tracing", + "whoami", +] + +[[package]] +name = "sqlx-postgres" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c824eb80b894f926f89a0b9da0c7f435d27cdd35b8c655b114e58223918577e" +dependencies = [ + "atoi", + "base64 0.21.7", + "bitflags 2.4.2", + "byteorder", + "crc", + "dotenvy", + "etcetera", + "futures-channel", + "futures-core", + "futures-io", + "futures-util", + "hex", + "hkdf", + "hmac", + "home", + "itoa", + "log", + "md-5", + "memchr", + "once_cell", + "rand", + "serde", + "serde_json", + "sha2", + "smallvec", + "sqlx-core", + "stringprep", + "thiserror", + "tracing", + "whoami", +] + +[[package]] +name = "sqlx-sqlite" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b244ef0a8414da0bed4bb1910426e890b19e5e9bccc27ada6b797d05c55ae0aa" +dependencies = [ + "atoi", + "flume", + "futures-channel", + "futures-core", + "futures-executor", + "futures-intrusive", + "futures-util", + "libsqlite3-sys", + "log", + "percent-encoding", + "serde", + "sqlx-core", + "tracing", + "url", + "urlencoding", +] + +[[package]] +name = "static_assertions" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" + +[[package]] +name = "string_cache" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f91138e76242f575eb1d3b38b4f1362f10d3a43f47d182a5b359af488a02293b" +dependencies = [ + "new_debug_unreachable", + "once_cell", + "parking_lot", + "phf_shared 0.10.0", + "precomputed-hash", +] + +[[package]] +name = "stringprep" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb41d74e231a107a1b4ee36bd1214b11285b77768d2e3824aedafa988fd36ee6" +dependencies = [ + "finl_unicode", + "unicode-bidi", + "unicode-normalization", +] + +[[package]] +name = "strsim" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ee073c9e4cd00e28217186dbe12796d692868f432bf2e97ee73bed0c56dfa01" + +[[package]] +name = "strum" +version = "0.25.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "290d54ea6f91c969195bdbcd7442c8c2a2ba87da8bf60a7ee86a235d4bc1e125" +dependencies = [ + "strum_macros 0.25.3", +] + +[[package]] +name = "strum" +version = "0.26.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d8cec3501a5194c432b2b7976db6b7d10ec95c253208b45f83f7136aa985e29" + +[[package]] +name = "strum_macros" +version = "0.25.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23dc1fa9ac9c169a78ba62f0b841814b7abae11bdd047b9c58f893439e309ea0" +dependencies = [ + "heck", + "proc-macro2", + "quote", + "rustversion", + "syn 2.0.51", +] + +[[package]] +name = "strum_macros" +version = "0.26.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6cf59daf282c0a494ba14fd21610a0325f9f90ec9d1231dea26bcb1d696c946" +dependencies = [ + "heck", + "proc-macro2", + "quote", + "rustversion", + "syn 2.0.51", +] + +[[package]] +name = "subtle" +version = "2.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6bdef32e8150c2a081110b42772ffe7d7c9032b606bc226c8260fd97e0976601" + +[[package]] +name = "svm-rs" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11297baafe5fa0c99d5722458eac6a5e25c01eb1b8e5cd137f54079093daa7a4" +dependencies = [ + "dirs", + "fs2", + "hex", + "once_cell", + "reqwest", + "semver", + "serde", + "serde_json", + "sha2", + "thiserror", + "url", + "zip", +] + +[[package]] +name = "syn" +version = "1.0.109" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "syn" +version = "2.0.51" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ab617d94515e94ae53b8406c628598680aa0c9587474ecbe58188f7b345d66c" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "sync_wrapper" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" + +[[package]] +name = "system-configuration" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba3a3adc5c275d719af8cb4272ea1c4a6d668a777f37e115f6d11ddbc1c8e0e7" +dependencies = [ + "bitflags 1.3.2", + "core-foundation", + "system-configuration-sys", +] + +[[package]] +name = "system-configuration-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75fb188eb626b924683e3b95e3a48e63551fcfb51949de2f06a9d91dbee93c9" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "tap" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369" + +[[package]] +name = "tempfile" +version = "3.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85b77fafb263dd9d05cbeac119526425676db3784113aa9295c88498cbf8bff1" +dependencies = [ + "cfg-if", + "fastrand", + "rustix", + "windows-sys 0.52.0", +] + +[[package]] +name = "term" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c59df8ac95d96ff9bede18eb7300b0fda5e5d8d90960e76f8e14ae765eedbf1f" +dependencies = [ + "dirs-next", + "rustversion", + "winapi", +] + +[[package]] +name = "terminal_size" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21bebf2b7c9e0a515f6e0f8c51dc0f8e4696391e6f1ff30379559f8365fb0df7" +dependencies = [ + "rustix", + "windows-sys 0.48.0", +] + +[[package]] +name = "textwrap" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23d434d3f8967a09480fb04132ebe0a3e088c173e6d0ee7897abbdf4eab0f8b9" +dependencies = [ + "smawk", + "unicode-linebreak", + "unicode-width", +] + +[[package]] +name = "thiserror" +version = "1.0.57" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e45bcbe8ed29775f228095caf2cd67af7a4ccf756ebff23a306bf3e8b47b24b" +dependencies = [ + "thiserror-impl", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.57" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a953cb265bef375dae3de6663da4d3804eee9682ea80d8e2542529b73c531c81" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "time" +version = "0.3.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8248b6521bb14bc45b4067159b9b6ad792e2d6d754d6c41fb50e29fefe38749" +dependencies = [ + "deranged", + "itoa", + "num-conv", + "powerfmt", + "serde", + "time-core", + "time-macros", +] + +[[package]] +name = "time-core" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3" + +[[package]] +name = "time-macros" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ba3a3ef41e6672a2f0f001392bb5dcd3ff0a9992d618ca761a11c3121547774" +dependencies = [ + "num-conv", + "time-core", +] + +[[package]] +name = "tiny-keccak" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2c9d3793400a45f954c52e73d068316d76b6f4e36977e3fcebb13a2721e80237" +dependencies = [ + "crunchy", +] + +[[package]] +name = "tinyvec" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" +dependencies = [ + "tinyvec_macros", +] + +[[package]] +name = "tinyvec_macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" + +[[package]] +name = "tokio" +version = "1.37.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1adbebffeca75fcfd058afa480fb6c0b81e165a0323f9c9d39c9697e37c46787" +dependencies = [ + "backtrace", + "bytes", + "libc", + "mio", + "num_cpus", + "parking_lot", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "windows-sys 0.48.0", +] + +[[package]] +name = "tokio-macros" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b8a1e28f2deaa14e508979454cb3a223b10b938b45af148bc0986de36f1923b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "tokio-rustls" +version = "0.24.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081" +dependencies = [ + "rustls", + "tokio", +] + +[[package]] +name = "tokio-stream" +version = "0.1.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "267ac89e0bec6e691e5813911606935d77c476ff49024f98abcea3e7b15e37af" +dependencies = [ + "futures-core", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "tokio-tungstenite" +version = "0.20.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "212d5dcb2a1ce06d81107c3d0ffa3121fe974b73f068c8282cb1c32328113b6c" +dependencies = [ + "futures-util", + "log", + "rustls", + "tokio", + "tokio-rustls", + "tungstenite", + "webpki-roots", +] + +[[package]] +name = "tokio-util" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5419f34732d9eb6ee4c3578b7989078579b7f039cbbb9ca2c4da015749371e15" +dependencies = [ + "bytes", + "futures-core", + "futures-sink", + "pin-project-lite", + "tokio", + "tracing", +] + +[[package]] +name = "toml" +version = "0.8.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e9dd1545e8208b4a5af1aa9bbd0b4cf7e9ea08fabc5d0a5c67fcaafa17433aa3" +dependencies = [ + "serde", + "serde_spanned", + "toml_datetime", + "toml_edit 0.22.9", +] + +[[package]] +name = "toml_datetime" +version = "0.6.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3550f4e9685620ac18a50ed434eb3aec30db8ba93b0287467bca5826ea25baf1" +dependencies = [ + "serde", +] + +[[package]] +name = "toml_edit" +version = "0.19.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b5bb770da30e5cbfde35a2d7b9b8a2c4b8ef89548a7a6aeab5c9a576e3e7421" +dependencies = [ + "indexmap", + "toml_datetime", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.20.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70f427fce4d84c72b5b732388bf4a9f4531b53f74e2887e3ecb2481f68f66d81" +dependencies = [ + "indexmap", + "toml_datetime", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.21.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a8534fd7f78b5405e860340ad6575217ce99f38d4d5c8f2442cb5ecb50090e1" +dependencies = [ + "indexmap", + "toml_datetime", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.22.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e40bb779c5187258fd7aad0eb68cb8706a0a81fa712fbea808ab43c4b8374c4" +dependencies = [ + "indexmap", + "serde", + "serde_spanned", + "toml_datetime", + "winnow 0.6.2", +] + +[[package]] +name = "tower-service" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6bc1c9ce2b5135ac7f93c72918fc37feb872bdc6a5533a8b85eb4b86bfdae52" + +[[package]] +name = "tracing" +version = "0.1.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" +dependencies = [ + "log", + "pin-project-lite", + "tracing-attributes", + "tracing-core", +] + +[[package]] +name = "tracing-attributes" +version = "0.1.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "tracing-core" +version = "0.1.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" +dependencies = [ + "once_cell", +] + +[[package]] +name = "tracing-futures" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97d095ae15e245a057c8e8451bab9b3ee1e1f68e9ba2b4fbc18d0ac5237835f2" +dependencies = [ + "pin-project", + "tracing", +] + +[[package]] +name = "try-lock" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" + +[[package]] +name = "tungstenite" +version = "0.20.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e3dac10fd62eaf6617d3a904ae222845979aec67c615d1c842b4002c7666fb9" +dependencies = [ + "byteorder", + "bytes", + "data-encoding", + "http", + "httparse", + "log", + "rand", + "rustls", + "sha1", + "thiserror", + "url", + "utf-8", +] + +[[package]] +name = "typenum" +version = "1.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" + +[[package]] +name = "uint" +version = "0.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76f64bba2c53b04fcab63c01a7d7427eadc821e3bc48c34dc9ba29c501164b52" +dependencies = [ + "byteorder", + "crunchy", + "hex", + "static_assertions", +] + +[[package]] +name = "unarray" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eaea85b334db583fe3274d12b4cd1880032beab409c0d774be044d4480ab9a94" + +[[package]] +name = "unicode-bidi" +version = "0.3.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08f95100a766bf4f8f28f90d77e0a5461bbdb219042e7679bebe79004fed8d75" + +[[package]] +name = "unicode-ident" +version = "1.0.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" + +[[package]] +name = "unicode-linebreak" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b09c83c3c29d37506a3e260c08c03743a6bb66a9cd432c6934ab501a190571f" + +[[package]] +name = "unicode-normalization" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a56d1686db2308d901306f92a263857ef59ea39678a5458e7cb17f01415101f5" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "unicode-segmentation" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d4c87d22b6e3f4a18d4d40ef354e97c90fcb14dd91d7dc0aa9d8a1172ebf7202" + +[[package]] +name = "unicode-width" +version = "0.1.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e51733f11c9c4f72aa0c160008246859e340b00807569a0da0e7a1079b27ba85" + +[[package]] +name = "unicode-xid" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f962df74c8c05a667b5ee8bcf162993134c104e96440b663c8daa176dc772d8c" + +[[package]] +name = "unicode_categories" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "39ec24b3121d976906ece63c9daad25b85969647682eee313cb5779fdd69e14e" + +[[package]] +name = "unsafe-libyaml" +version = "0.2.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "673aac59facbab8a9007c7f6108d11f63b603f7cabff99fabf650fea5c32b861" + +[[package]] +name = "untrusted" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a156c684c91ea7d62626509bce3cb4e1d9ed5c4d978f7b4352658f96a4c26b4a" + +[[package]] +name = "untrusted" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" + +[[package]] +name = "url" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31e6302e3bb753d46e83516cae55ae196fc0c309407cf11ab35cc51a4c2a4633" +dependencies = [ + "form_urlencoded", + "idna", + "percent-encoding", + "serde", +] + +[[package]] +name = "urlencoding" +version = "2.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "daf8dba3b7eb870caf1ddeed7bc9d2a049f3cfdfae7cb521b087cc33ae4c49da" + +[[package]] +name = "utf-8" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" + +[[package]] +name = "utf8parse" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "711b9620af191e0cdc7468a8d14e709c3dcdb115b36f838e601583af800a370a" + +[[package]] +name = "uuid" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7" +dependencies = [ + "getrandom", + "serde", +] + +[[package]] +name = "uuid" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a183cf7feeba97b4dd1c0d46788634f6221d87fa961b305bed08c851829efcc0" +dependencies = [ + "getrandom", +] + +[[package]] +name = "vcpkg" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" + +[[package]] +name = "version_check" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" + +[[package]] +name = "walkdir" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d71d857dc86794ca4c280d616f7da00d2dbfd8cd788846559a6813e6aa4b54ee" +dependencies = [ + "same-file", + "winapi-util", +] + +[[package]] +name = "want" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" +dependencies = [ + "try-lock", +] + +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" + +[[package]] +name = "wasite" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8dad83b4f25e74f184f64c43b150b91efe7647395b42289f38e50566d82855b" + +[[package]] +name = "wasm-bindgen" +version = "0.2.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c1e124130aee3fb58c5bdd6b639a0509486b0338acaaae0c84a5124b0f588b7f" +dependencies = [ + "cfg-if", + "wasm-bindgen-macro", +] + +[[package]] +name = "wasm-bindgen-backend" +version = "0.2.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c9e7e1900c352b609c8488ad12639a311045f40a35491fb69ba8c12f758af70b" +dependencies = [ + "bumpalo", + "log", + "once_cell", + "proc-macro2", + "quote", + "syn 2.0.51", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.41" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "877b9c3f61ceea0e56331985743b13f3d25c406a7098d45180fb5f09bc19ed97" +dependencies = [ + "cfg-if", + "js-sys", + "wasm-bindgen", + "web-sys", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b30af9e2d358182b5c7449424f017eba305ed32a7010509ede96cdc4696c46ed" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "642f325be6301eb8107a83d12a8ac6c1e1c54345a7ef1a9261962dfefda09e66" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", + "wasm-bindgen-backend", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f186bd2dcf04330886ce82d6f33dd75a7bfcf69ecf5763b89fcde53b6ac9838" + +[[package]] +name = "web-sys" +version = "0.3.68" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96565907687f7aceb35bc5fc03770a8a0471d82e479f25832f54a0e3f4b28446" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "webpki-roots" +version = "0.25.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f20c57d8d7db6d3b86154206ae5d8fba62dd39573114de97c2cb0578251f8e1" + +[[package]] +name = "whoami" +version = "1.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a44ab49fad634e88f55bf8f9bb3abd2f27d7204172a112c7c9987e01c1c94ea9" +dependencies = [ + "redox_syscall", + "wasite", +] + +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-util" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f29e6f9198ba0d26b4c9f07dbe6f9ed633e1f3d5b8b414090084349e46a52596" +dependencies = [ + "winapi", +] + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + +[[package]] +name = "windows-sys" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" +dependencies = [ + "windows-targets 0.48.5", +] + +[[package]] +name = "windows-sys" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d" +dependencies = [ + "windows-targets 0.52.3", +] + +[[package]] +name = "windows-targets" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" +dependencies = [ + "windows_aarch64_gnullvm 0.48.5", + "windows_aarch64_msvc 0.48.5", + "windows_i686_gnu 0.48.5", + "windows_i686_msvc 0.48.5", + "windows_x86_64_gnu 0.48.5", + "windows_x86_64_gnullvm 0.48.5", + "windows_x86_64_msvc 0.48.5", +] + +[[package]] +name = "windows-targets" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d380ba1dc7187569a8a9e91ed34b8ccfc33123bbacb8c0aed2d1ad7f3ef2dc5f" +dependencies = [ + "windows_aarch64_gnullvm 0.52.3", + "windows_aarch64_msvc 0.52.3", + "windows_i686_gnu 0.52.3", + "windows_i686_msvc 0.52.3", + "windows_x86_64_gnu 0.52.3", + "windows_x86_64_gnullvm 0.52.3", + "windows_x86_64_msvc 0.52.3", +] + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68e5dcfb9413f53afd9c8f86e56a7b4d86d9a2fa26090ea2dc9e40fba56c6ec6" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8dab469ebbc45798319e69eebf92308e541ce46760b49b18c6b3fe5e8965b30f" + +[[package]] +name = "windows_i686_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + +[[package]] +name = "windows_i686_gnu" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a4e9b6a7cac734a8b4138a4e1044eac3404d8326b6c0f939276560687a033fb" + +[[package]] +name = "windows_i686_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + +[[package]] +name = "windows_i686_msvc" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "28b0ec9c422ca95ff34a78755cfa6ad4a51371da2a5ace67500cf7ca5f232c58" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "704131571ba93e89d7cd43482277d6632589b18ecf4468f591fbae0a8b101614" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42079295511643151e98d61c38c0acc444e52dd42ab456f7ccfd5152e8ecf21c" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0770833d60a970638e989b3fa9fd2bb1aaadcf88963d1659fd7d9990196ed2d6" + +[[package]] +name = "winnow" +version = "0.5.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876" +dependencies = [ + "memchr", +] + +[[package]] +name = "winnow" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a4191c47f15cc3ec71fcb4913cb83d58def65dd3787610213c649283b5ce178" +dependencies = [ + "memchr", +] + +[[package]] +name = "winreg" +version = "0.50.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "524e57b2c537c0f9b1e69f1965311ec12182b4122e45035b1508cd24d2adadb1" +dependencies = [ + "cfg-if", + "windows-sys 0.48.0", +] + +[[package]] +name = "ws_stream_wasm" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7999f5f4217fe3818726b66257a4475f71e74ffd190776ad053fa159e50737f5" +dependencies = [ + "async_io_stream", + "futures", + "js-sys", + "log", + "pharos", + "rustc_version", + "send_wrapper 0.6.0", + "thiserror", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "wyz" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05f360fc0b24296329c78fda852a1e9ae82de9cf7b27dae4b7f62f118f77b9ed" +dependencies = [ + "tap", +] + +[[package]] +name = "xshell" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6db0ab86eae739efd1b054a8d3d16041914030ac4e01cd1dca0cf252fd8b6437" +dependencies = [ + "xshell-macros", +] + +[[package]] +name = "xshell-macros" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d422e8e38ec76e2f06ee439ccc765e9c6a9638b9e7c9f2e8255e4d41e8bd852" + +[[package]] +name = "yansi" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09041cd90cf85f7f8b2df60c646f853b7f535ce68f85244eb6731cf89fa498ec" + +[[package]] +name = "zerocopy" +version = "0.7.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "74d4d3961e53fa4c9a25a8637fc2bfaf2595b3d3ae34875568a5cf64787716be" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.7.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ce1b18ccd8e73a9321186f97e46f9f04b778851177567b1975109d26a08d2a6" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "zeroize" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "525b4ec142c6b68a2d10f01f7bbf6755599ca3f81ea53b8431b7dd348f5fdb2d" +dependencies = [ + "zeroize_derive", +] + +[[package]] +name = "zeroize_derive" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.51", +] + +[[package]] +name = "zip" +version = "0.6.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "760394e246e4c28189f19d488c058bf16f564016aefac5d32bb1f3b51d5e9261" +dependencies = [ + "aes", + "byteorder", + "bzip2", + "constant_time_eq", + "crc32fast", + "crossbeam-utils", + "flate2", + "hmac", + "pbkdf2 0.11.0", + "sha1", + "time", + "zstd", +] + +[[package]] +name = "zk_inception" +version = "0.1.0" +dependencies = [ + "anyhow", + "clap", + "cliclack", + "common", + "console", + "ethers", + "human-panic", + "serde", + "serde_json", + "serde_yaml", + "strum 0.26.2", + "strum_macros 0.26.2", + "thiserror", + "tokio", + "toml", + "url", + "xshell", +] + +[[package]] +name = "zk_supervisor" +version = "0.1.0" +dependencies = [ + "human-panic", +] + +[[package]] +name = "zstd" +version = "0.11.2+zstd.1.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20cc960326ece64f010d2d2107537f26dc589a6573a316bd5b1dba685fa5fde4" +dependencies = [ + "zstd-safe", +] + +[[package]] +name = "zstd-safe" +version = "5.0.2+zstd.1.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d2a5585e04f9eea4b2a3d1eca508c4dee9592a89ef6f450c11719da0726f4db" +dependencies = [ + "libc", + "zstd-sys", +] + +[[package]] +name = "zstd-sys" +version = "2.0.9+zstd.1.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e16efa8a874a0481a574084d34cc26fdb3b99627480f785888deb6386506656" +dependencies = [ + "cc", + "pkg-config", +] diff --git a/zk_toolbox/Cargo.toml b/zk_toolbox/Cargo.toml new file mode 100644 index 000000000000..f2ade7a48294 --- /dev/null +++ b/zk_toolbox/Cargo.toml @@ -0,0 +1,43 @@ +[workspace] +members = ["crates/common", + "crates/zk_inception", + "crates/zk_supervisor", +] +resolver = "2" + +[workspace.package] +version = "0.1.0" +edition = "2021" +homepage = "https://zksync.io/" +license = "MIT OR Apache-2.0" +authors = ["The Matter Labs Team "] +exclude = ["./github"] +repository = "https://github.com/matter-labs/zk_toolbox/" +description = "ZK Toolbox is a set of tools for working with zk stack." +keywords = ["zk", "cryptography", "blockchain", "ZKStack", "zkSync"] + + +[workspace.dependencies] +# Local dependencies +common = { path = "crates/common" } + +# External dependencies +anyhow = "1.0.82" +clap = { version = "4.4", features = ["derive", "wrap_help"] } +cliclack = "0.2.5" +console = "0.15.8" +ethers = "2.0" +human-panic = "2.0" +once_cell = "1.19.0" +serde = { version = "1.0", features = ["derive"] } +serde_json = "1.0" +serde_yaml = "0.9" +sqlx = { version = "0.7.4", features = ["runtime-tokio", "migrate", "postgres"] } +strum = "0.26.2" +strum_macros = "0.26.2" +tokio = { version = "1.37", features = ["full"] } +toml = "0.8.12" +url = { version = "2.5.0", features = ["serde"] } +xshell = "0.2.6" +futures = "0.3.30" +thiserror = "1.0.57" diff --git a/zk_toolbox/README.md b/zk_toolbox/README.md new file mode 100644 index 000000000000..aed5fc15cbc9 --- /dev/null +++ b/zk_toolbox/README.md @@ -0,0 +1,108 @@ +# zk_toolbox + +Toolkit for creating and managing ZK Stack chains. + +## ZK Inception + +ZK Inception facilitates the creation and management of ZK Stacks. All commands are interactive, but you can also pass +all necessary arguments via the command line. + +### Dependencies + +Ensure you have followed +[these instructions](https://github.com/matter-labs/zksync-era/blob/main/docs/guides/setup-dev.md) to set up +dependencies on your machine (don't worry about the Environment section for now). + +### Installation + +Install zk_inception from git: + +```bash +cargo install --git https://github.com/matter-labs/zksync-era/ --locked zk_inception --force +``` + +Manually building from a local copy of the [ZkSync](https://github.com/matter-labs/zksync-era/) repository: + +```bash +cd zk_toolbox +cargo install --path ./crates/zk_inception --force --locked +``` + +### Foundry Integration + +Foundry is utilized for deploying smart contracts. For commands related to deployment, you can pass flags for Foundry +integration. + +### Ecosystem + +ZK Stack allows you to either create a new ecosystem or connect to an existing one. An ecosystem includes the components +that connects all ZK chains, like the BridgeHub, the shared bridges, and state transition managers. +[Learn more](https://docs.zksync.io/zk-stack/components/shared-bridges.html). + +To create a ZK Stack project, you must first create an ecosystem: + +```bash +zk_inception ecosystem create +``` + +If you chose to not start database & L1 containers after creating the ecosystem, you can later run +`zk_inception containers` + +All subsequent commands should be executed from within the ecosystem folder you created: + +```bash +cd `path/to/ecosystem/name` +``` + +If the ecosystem has never been deployed before, initialization is required: + +```bash +zk_inception ecosystem init +``` + +This command also initializes the first ZK chain. Note that the very first chain becomes the default one, but you can +override it with another by using the `--chain ` flag. + +To change the default ZK chain, use: + +```bash +zk_inception ecosystem change-default-chain +``` + +IMPORTANT: It is not yet possible to use an existing ecosystem and register a chain to it. this feature will be added in +the future. + +### ZK Chain + +Upon ecosystem creation, the first ZK chain is automatically generated. However, you can create additional chains and +switch between them: + +```bash +zk_inception chain create +``` + +Once created, contracts for the ZK chain must be deployed: + +```bash +zk_inception chain init +``` + +Initialization utilizes the ecosystem's governance to register it in the BridgeHub. + +If contracts were deployed by a third party (e.g., MatterLabs), you may need to run the genesis process locally: + +```bash +zk_inception chain genesis +``` + +This ensures proper initialization of the server. + +### Zk Server + +For running the chain: + +```bash +zk_inception server +``` + +You can specify the chain you are running by providing `--chain ` argument diff --git a/zk_toolbox/crates/common/Cargo.toml b/zk_toolbox/crates/common/Cargo.toml new file mode 100644 index 000000000000..588254e445f8 --- /dev/null +++ b/zk_toolbox/crates/common/Cargo.toml @@ -0,0 +1,29 @@ +[package] +name = "common" +version.workspace = true +edition.workspace = true +homepage.workspace = true +license.workspace = true +authors.workspace = true +exclude.workspace = true +repository.workspace = true +description.workspace = true +keywords.workspace = true + +[dependencies] +anyhow.workspace = true +clap.workspace = true +cliclack.workspace = true +console.workspace = true +ethers.workspace = true +once_cell.workspace = true +serde.workspace = true +serde_json.workspace = true +serde_yaml.workspace = true +sqlx.workspace = true +strum.workspace = true +strum_macros.workspace = true +toml.workspace = true +url.workspace = true +xshell.workspace = true +futures.workspace = true \ No newline at end of file diff --git a/zk_toolbox/crates/common/src/cmd.rs b/zk_toolbox/crates/common/src/cmd.rs new file mode 100644 index 000000000000..8b18c7733059 --- /dev/null +++ b/zk_toolbox/crates/common/src/cmd.rs @@ -0,0 +1,135 @@ +use anyhow::bail; +use console::style; + +use crate::{ + config::global_config, + logger::{self}, +}; + +/// A wrapper around [`xshell::Cmd`] that allows for improved error handling, +/// and verbose logging. +#[derive(Debug)] +pub struct Cmd<'a> { + inner: xshell::Cmd<'a>, + force_run: bool, +} + +impl<'a> Cmd<'a> { + /// Create a new `Cmd` instance. + pub fn new(cmd: xshell::Cmd<'a>) -> Self { + Self { + inner: cmd, + force_run: false, + } + } + + /// Run the command printing the output to the console. + pub fn with_force_run(mut self) -> Self { + self.force_run = true; + self + } + + /// Run the command without capturing its output. + pub fn run(&mut self) -> anyhow::Result<()> { + self.run_cmd()?; + Ok(()) + } + + /// Run the command and capture its output, logging the command + /// and its output if verbose selected. + fn run_cmd(&mut self) -> anyhow::Result<()> { + if global_config().verbose || self.force_run { + logger::debug(format!("Running: {}", self.inner)); + logger::new_empty_line(); + self.inner.run()?; + logger::new_empty_line(); + logger::new_line(); + } else { + // Command will be logged manually. + self.inner.set_quiet(true); + // Error will be handled manually. + self.inner.set_ignore_status(true); + let output = self.inner.output()?; + self.check_output_status(&output)?; + } + + if global_config().verbose { + logger::debug(format!("Command completed: {}", self.inner)); + } + + Ok(()) + } + + fn check_output_status(&self, output: &std::process::Output) -> anyhow::Result<()> { + if !output.status.success() { + logger::new_line(); + logger::error_note( + &format!("Command failed to run: {}", self.inner), + &log_output(output), + ); + bail!("Command failed to run: {}", self.inner); + } + + Ok(()) + } +} + +fn log_output(output: &std::process::Output) -> String { + let (status, stdout, stderr) = get_indented_output(output, 4, 120); + let status_header = style(" Status:").bold(); + let stdout_header = style(" Stdout:").bold(); + let stderr_header = style(" Stderr:").bold(); + + format!("{status_header}\n{status}\n{stdout_header}\n{stdout}\n{stderr_header}\n{stderr}") +} + +// Indent output and wrap text. +fn get_indented_output( + output: &std::process::Output, + indentation: usize, + wrap: usize, +) -> (String, String, String) { + let status = output.status.to_string(); + let stdout = String::from_utf8_lossy(&output.stdout); + let stderr = String::from_utf8_lossy(&output.stderr); + + let indent = |s: &str| { + s.lines() + .map(|l| format!("{:indent$}{}", "", l, indent = indentation)) + .collect::>() + .join("\n") + }; + let wrap_text_to_len = |s: &str| { + let mut result = String::new(); + + for original_line in s.split('\n') { + if original_line.trim().is_empty() { + result.push('\n'); + continue; + } + + let mut line = String::new(); + for word in original_line.split_whitespace() { + if line.len() + word.len() + 1 > wrap { + result.push_str(&line); + result.push('\n'); + line.clear(); + } + if !line.is_empty() { + line.push(' '); + } + line.push_str(word); + } + result.push_str(&line); + result.push('\n'); + } + + result + }; + + ( + indent(&wrap_text_to_len(&status)), + indent(&wrap_text_to_len(&stdout)), + indent(&wrap_text_to_len(&stderr)), + ) +} diff --git a/zk_toolbox/crates/common/src/config.rs b/zk_toolbox/crates/common/src/config.rs new file mode 100644 index 000000000000..9f3adc2e83d3 --- /dev/null +++ b/zk_toolbox/crates/common/src/config.rs @@ -0,0 +1,18 @@ +use once_cell::sync::OnceCell; + +static CONFIG: OnceCell = OnceCell::new(); + +pub fn init_global_config(config: GlobalConfig) { + CONFIG.set(config).unwrap(); +} + +pub fn global_config() -> &'static GlobalConfig { + CONFIG.get().expect("GlobalConfig not initialized") +} + +#[derive(Debug)] +pub struct GlobalConfig { + pub verbose: bool, + pub chain_name: Option, + pub ignore_prerequisites: bool, +} diff --git a/zk_toolbox/crates/common/src/db.rs b/zk_toolbox/crates/common/src/db.rs new file mode 100644 index 000000000000..b345fc119469 --- /dev/null +++ b/zk_toolbox/crates/common/src/db.rs @@ -0,0 +1,105 @@ +use std::{collections::HashMap, path::PathBuf}; + +use crate::{config::global_config, logger}; +use sqlx::{ + migrate::{Migrate, MigrateError, Migrator}, + Connection, PgConnection, +}; +use url::Url; +use xshell::Shell; + +pub async fn init_db(db_url: &Url, name: &str) -> anyhow::Result<()> { + // Connect to the database. + let mut connection = PgConnection::connect(db_url.as_ref()).await?; + + let query = format!("CREATE DATABASE {}", name); + // Create DB. + sqlx::query(&query).execute(&mut connection).await?; + + Ok(()) +} + +pub async fn drop_db_if_exists(db_url: &Url, name: &str) -> anyhow::Result<()> { + // Connect to the database. + let mut connection = PgConnection::connect(db_url.as_ref()).await?; + + let query = format!("DROP DATABASE IF EXISTS {}", name); + // DROP DB. + sqlx::query(&query).execute(&mut connection).await?; + + Ok(()) +} + +pub async fn migrate_db( + shell: &Shell, + migrations_folder: PathBuf, + db_url: &str, +) -> anyhow::Result<()> { + // Most of this file is copy-pasted from SQLx CLI: + // https://github.com/launchbadge/sqlx/blob/main/sqlx-cli/src/migrate.rs + // Warrants a refactoring if this tool makes it to production. + + if !shell.path_exists(&migrations_folder) { + anyhow::bail!("Migrations folder {migrations_folder:?} doesn't exist"); + } + let migrator = Migrator::new(migrations_folder).await?; + + let mut conn = PgConnection::connect(db_url).await?; + conn.ensure_migrations_table().await?; + + let version = conn.dirty_version().await?; + if let Some(version) = version { + anyhow::bail!(MigrateError::Dirty(version)); + } + + let applied_migrations: HashMap<_, _> = conn + .list_applied_migrations() + .await? + .into_iter() + .map(|m| (m.version, m)) + .collect(); + + if global_config().verbose { + logger::debug("Migrations result:") + } + + for migration in migrator.iter() { + if migration.migration_type.is_down_migration() { + // Skipping down migrations + continue; + } + + match applied_migrations.get(&migration.version) { + Some(applied_migration) => { + if migration.checksum != applied_migration.checksum { + anyhow::bail!(MigrateError::VersionMismatch(migration.version)); + } + } + None => { + let skip = false; + + let elapsed = conn.apply(migration).await?; + let text = if skip { "Skipped" } else { "Applied" }; + + if global_config().verbose { + logger::raw(&format!( + " {} {}/{} {} ({elapsed:?})", + text, + migration.version, + migration.migration_type.label(), + migration.description, + )); + } + } + } + } + + // Close the connection before exiting: + // * For MySQL and Postgres this should ensure timely cleanup on the server side, + // including decrementing the open connection count. + // * For SQLite this should checkpoint and delete the WAL file to ensure the migrations + // were actually applied to the database file and aren't just sitting in the WAL file. + let _ = conn.close().await; + + Ok(()) +} diff --git a/zk_toolbox/crates/common/src/docker.rs b/zk_toolbox/crates/common/src/docker.rs new file mode 100644 index 000000000000..97bba57b8aae --- /dev/null +++ b/zk_toolbox/crates/common/src/docker.rs @@ -0,0 +1,9 @@ +use crate::cmd::Cmd; +use xshell::{cmd, Shell}; + +pub fn up(shell: &Shell, docker_compose_file: &str) -> anyhow::Result<()> { + Cmd::new(cmd!(shell, "docker-compose -f {docker_compose_file} up -d")).run() +} +pub fn down(shell: &Shell, docker_compose_file: &str) -> anyhow::Result<()> { + Cmd::new(cmd!(shell, "docker-compose -f {docker_compose_file} down")).run() +} diff --git a/zk_toolbox/crates/common/src/ethereum.rs b/zk_toolbox/crates/common/src/ethereum.rs new file mode 100644 index 000000000000..6e9c24488c5e --- /dev/null +++ b/zk_toolbox/crates/common/src/ethereum.rs @@ -0,0 +1,57 @@ +use std::{ops::Add, time::Duration}; + +use ethers::prelude::Signer; +use ethers::{ + core::k256::ecdsa::SigningKey, + middleware::MiddlewareBuilder, + prelude::{Http, LocalWallet, Provider}, + prelude::{SignerMiddleware, H256}, + providers::Middleware, + types::{Address, TransactionRequest}, +}; + +use crate::wallets::Wallet; + +pub fn create_ethers_client( + private_key: H256, + l1_rpc: String, + chain_id: Option, +) -> anyhow::Result, ethers::prelude::Wallet>> { + let mut wallet = LocalWallet::from_bytes(private_key.as_bytes())?; + if let Some(chain_id) = chain_id { + wallet = wallet.with_chain_id(chain_id); + } + let client = Provider::::try_from(l1_rpc)?.with_signer(wallet); + Ok(client) +} + +pub async fn distribute_eth( + main_wallet: Wallet, + addresses: Vec
, + l1_rpc: String, + chain_id: u32, + amount: u128, +) -> anyhow::Result<()> { + let client = create_ethers_client(main_wallet.private_key.unwrap(), l1_rpc, Some(chain_id))?; + let mut pending_txs = vec![]; + let mut nonce = client.get_transaction_count(client.address(), None).await?; + for address in addresses { + let tx = TransactionRequest::new() + .to(address) + .value(amount) + .nonce(nonce) + .chain_id(chain_id); + nonce = nonce.add(1); + pending_txs.push( + client + .send_transaction(tx, None) + .await? + // It's safe to set such low number of confirmations and low interval for localhost + .confirmations(1) + .interval(Duration::from_millis(30)), + ); + } + + futures::future::join_all(pending_txs).await; + Ok(()) +} diff --git a/zk_toolbox/crates/common/src/files.rs b/zk_toolbox/crates/common/src/files.rs new file mode 100644 index 000000000000..c29f79aaa202 --- /dev/null +++ b/zk_toolbox/crates/common/src/files.rs @@ -0,0 +1,40 @@ +use std::path::Path; + +use serde::Serialize; +use xshell::Shell; + +pub fn save_yaml_file( + shell: &Shell, + file_path: impl AsRef, + content: impl Serialize, + comment: impl ToString, +) -> anyhow::Result<()> { + let data = format!( + "{}{}", + comment.to_string(), + serde_yaml::to_string(&content)? + ); + shell.write_file(file_path, data)?; + Ok(()) +} + +pub fn save_toml_file( + shell: &Shell, + file_path: impl AsRef, + content: impl Serialize, + comment: impl ToString, +) -> anyhow::Result<()> { + let data = format!("{}{}", comment.to_string(), toml::to_string(&content)?); + shell.write_file(file_path, data)?; + Ok(()) +} + +pub fn save_json_file( + shell: &Shell, + file_path: impl AsRef, + content: impl Serialize, +) -> anyhow::Result<()> { + let data = serde_json::to_string_pretty(&content)?; + shell.write_file(file_path, data)?; + Ok(()) +} diff --git a/zk_toolbox/crates/common/src/forge.rs b/zk_toolbox/crates/common/src/forge.rs new file mode 100644 index 000000000000..4335765e330c --- /dev/null +++ b/zk_toolbox/crates/common/src/forge.rs @@ -0,0 +1,348 @@ +use std::path::{Path, PathBuf}; +use std::str::FromStr; + +use clap::{Parser, ValueEnum}; +use ethers::abi::Address; +use ethers::middleware::Middleware; +use ethers::prelude::{LocalWallet, Signer, U256}; +use ethers::{abi::AbiEncode, types::H256}; +use serde::{Deserialize, Serialize}; +use strum_macros::Display; +use xshell::{cmd, Shell}; + +use crate::cmd::Cmd; +use crate::ethereum::create_ethers_client; + +/// Forge is a wrapper around the forge binary. +pub struct Forge { + path: PathBuf, +} + +impl Forge { + /// Create a new Forge instance. + pub fn new(path: &Path) -> Self { + Forge { + path: path.to_path_buf(), + } + } + + /// Create a new ForgeScript instance. + /// + /// The script path can be passed as a relative path to the base path + /// or as an absolute path. + pub fn script(&self, path: &Path, args: ForgeScriptArgs) -> ForgeScript { + ForgeScript { + base_path: self.path.clone(), + script_path: path.to_path_buf(), + args, + } + } +} + +/// ForgeScript is a wrapper around the forge script command. +pub struct ForgeScript { + base_path: PathBuf, + script_path: PathBuf, + args: ForgeScriptArgs, +} + +impl ForgeScript { + /// Run the forge script command. + pub fn run(mut self, shell: &Shell) -> anyhow::Result<()> { + let _dir_guard = shell.push_dir(&self.base_path); + let script_path = self.script_path.as_os_str(); + let args = self.args.build(); + Cmd::new(cmd!(shell, "forge script {script_path} --legacy {args...}")).run()?; + Ok(()) + } + + pub fn wallet_args_passed(&self) -> bool { + self.args.wallet_args_passed() + } + + /// Add the ffi flag to the forge script command. + pub fn with_ffi(mut self) -> Self { + self.args.add_arg(ForgeScriptArg::Ffi); + self + } + + /// Add the rpc-url flag to the forge script command. + pub fn with_rpc_url(mut self, rpc_url: String) -> Self { + self.args.add_arg(ForgeScriptArg::RpcUrl { url: rpc_url }); + self + } + + /// Add the broadcast flag to the forge script command. + pub fn with_broadcast(mut self) -> Self { + self.args.add_arg(ForgeScriptArg::Broadcast); + self + } + + pub fn with_signature(mut self, signature: &str) -> Self { + self.args.add_arg(ForgeScriptArg::Sig { + sig: signature.to_string(), + }); + self + } + + /// Makes sure a transaction is sent, only after its previous one has been confirmed and succeeded. + pub fn with_slow(mut self) -> Self { + self.args.add_arg(ForgeScriptArg::Slow); + self + } + + /// Adds the private key of the deployer account. + pub fn with_private_key(mut self, private_key: H256) -> Self { + self.args.add_arg(ForgeScriptArg::PrivateKey { + private_key: private_key.encode_hex(), + }); + self + } + // Do not start the script if balance is not enough + pub fn private_key(&self) -> Option { + self.args.args.iter().find_map(|a| { + if let ForgeScriptArg::PrivateKey { private_key } = a { + Some(H256::from_str(private_key).unwrap()) + } else { + None + } + }) + } + + pub fn rpc_url(&self) -> Option { + self.args.args.iter().find_map(|a| { + if let ForgeScriptArg::RpcUrl { url } = a { + Some(url.clone()) + } else { + None + } + }) + } + + pub fn address(&self) -> Option
{ + self.private_key().and_then(|a| { + LocalWallet::from_bytes(a.as_bytes()) + .ok() + .map(|a| a.address()) + }) + } + + pub async fn check_the_balance(&self, minimum_value: U256) -> anyhow::Result { + let Some(rpc_url) = self.rpc_url() else { + return Ok(true); + }; + let Some(private_key) = self.private_key() else { + return Ok(true); + }; + let client = create_ethers_client(private_key, rpc_url, None)?; + let balance = client.get_balance(client.address(), None).await?; + Ok(balance > minimum_value) + } +} + +const PROHIBITED_ARGS: [&str; 10] = [ + "--contracts", + "--root", + "--lib-paths", + "--out", + "--sig", + "--target-contract", + "--chain-id", + "-C", + "-O", + "-s", +]; + +const WALLET_ARGS: [&str; 18] = [ + "-a", + "--froms", + "-i", + "--private-keys", + "--private-key", + "--mnemonics", + "--mnenomic-passphrases", + "--mnemonic-derivation-paths", + "--mnemonic-indexes", + "--keystore", + "--account", + "--password", + "--password-file", + "-l", + "--ledger", + "-t", + "--trezor", + "--aws", +]; + +/// Set of known forge script arguments necessary for execution. +#[derive(Display, Debug, Serialize, Deserialize, Clone, PartialEq)] +#[strum(serialize_all = "kebab-case", prefix = "--")] +pub enum ForgeScriptArg { + Broadcast, + #[strum(to_string = "etherscan-api-key={api_key}")] + EtherscanApiKey { + api_key: String, + }, + Ffi, + #[strum(to_string = "private-key={private_key}")] + PrivateKey { + private_key: String, + }, + #[strum(to_string = "rpc-url={url}")] + RpcUrl { + url: String, + }, + #[strum(to_string = "sig={sig}")] + Sig { + sig: String, + }, + Slow, + #[strum(to_string = "verifier={verifier}")] + Verifier { + verifier: String, + }, + #[strum(to_string = "verifier-url={url}")] + VerifierUrl { + url: String, + }, + Verify, +} + +/// ForgeScriptArgs is a set of arguments that can be passed to the forge script command. +#[derive(Default, Debug, Serialize, Deserialize, Parser, Clone)] +#[clap(next_help_heading = "Forge options")] +pub struct ForgeScriptArgs { + /// List of known forge script arguments. + #[clap(skip)] + args: Vec, + /// Verify deployed contracts + #[clap(long, default_missing_value = "true", num_args = 0..=1)] + pub verify: Option, + /// Verifier to use + #[clap(long, default_value_t = ForgeVerifier::Etherscan)] + pub verifier: ForgeVerifier, + /// Verifier URL, if using a custom provider + #[clap(long)] + pub verifier_url: Option, + /// Verifier API key + #[clap(long)] + pub verifier_api_key: Option, + /// List of additional arguments that can be passed through the CLI. + /// + /// e.g.: `zk_inception init -a --private-key=` + #[clap(long, short)] + #[arg(trailing_var_arg = true, allow_hyphen_values = true, hide = false)] + additional_args: Vec, +} + +impl ForgeScriptArgs { + /// Build the forge script command arguments. + pub fn build(&mut self) -> Vec { + self.add_verify_args(); + self.cleanup_contract_args(); + self.args + .iter() + .map(|arg| arg.to_string()) + .chain(self.additional_args.clone()) + .collect() + } + + /// Adds verify arguments to the forge script command. + fn add_verify_args(&mut self) { + if !self.verify.is_some_and(|v| v) { + return; + } + + self.add_arg(ForgeScriptArg::Verify); + if let Some(url) = &self.verifier_url { + self.add_arg(ForgeScriptArg::VerifierUrl { url: url.clone() }); + } + if let Some(api_key) = &self.verifier_api_key { + self.add_arg(ForgeScriptArg::EtherscanApiKey { + api_key: api_key.clone(), + }); + } + self.add_arg(ForgeScriptArg::Verifier { + verifier: self.verifier.to_string(), + }); + } + + /// Cleanup the contract arguments which are not allowed to be passed through the CLI. + fn cleanup_contract_args(&mut self) { + let mut skip_next = false; + let mut cleaned_args = vec![]; + let mut forbidden_args = vec![]; + + let prohibited_with_spacing: Vec = PROHIBITED_ARGS + .iter() + .flat_map(|arg| vec![format!("{arg} "), format!("{arg}\t")]) + .collect(); + + let prohibited_with_equals: Vec = PROHIBITED_ARGS + .iter() + .map(|arg| format!("{arg}=")) + .collect(); + + for arg in self.additional_args.iter() { + if skip_next { + skip_next = false; + continue; + } + + if prohibited_with_spacing + .iter() + .any(|prohibited_arg| arg.starts_with(prohibited_arg)) + { + skip_next = true; + forbidden_args.push(arg.clone()); + continue; + } + + if prohibited_with_equals + .iter() + .any(|prohibited_arg| arg.starts_with(prohibited_arg)) + { + skip_next = false; + forbidden_args.push(arg.clone()); + continue; + } + + cleaned_args.push(arg.clone()); + } + + if !forbidden_args.is_empty() { + println!( + "Warning: the following arguments are not allowed to be passed through the CLI and were skipped: {:?}", + forbidden_args + ); + } + + self.additional_args = cleaned_args; + } + + /// Add additional arguments to the forge script command. + /// If the argument already exists, a warning will be printed. + pub fn add_arg(&mut self, arg: ForgeScriptArg) { + if self.args.contains(&arg) { + println!("Warning: argument {arg:?} already exists"); + return; + } + self.args.push(arg); + } + + pub fn wallet_args_passed(&self) -> bool { + self.additional_args + .iter() + .any(|arg| WALLET_ARGS.contains(&arg.as_ref())) + } +} + +#[derive(Debug, Clone, ValueEnum, Display, Serialize, Deserialize, Default)] +#[strum(serialize_all = "snake_case")] +pub enum ForgeVerifier { + #[default] + Etherscan, + Sourcify, + Blockscout, + Oklink, +} diff --git a/zk_toolbox/crates/common/src/lib.rs b/zk_toolbox/crates/common/src/lib.rs new file mode 100644 index 000000000000..349cd751c5f6 --- /dev/null +++ b/zk_toolbox/crates/common/src/lib.rs @@ -0,0 +1,17 @@ +pub mod cmd; +pub mod config; +pub mod db; +pub mod docker; +pub mod ethereum; +pub mod files; +pub mod forge; +mod prerequisites; +mod prompt; +mod slugify; +mod term; +pub mod wallets; + +pub use prerequisites::check_prerequisites; +pub use prompt::{init_prompt_theme, Prompt, PromptConfirm, PromptSelect}; +pub use slugify::slugify; +pub use term::{logger, spinner}; diff --git a/zk_toolbox/crates/common/src/prerequisites.rs b/zk_toolbox/crates/common/src/prerequisites.rs new file mode 100644 index 000000000000..7551b247c681 --- /dev/null +++ b/zk_toolbox/crates/common/src/prerequisites.rs @@ -0,0 +1,64 @@ +use crate::{cmd::Cmd, logger}; +use xshell::{cmd, Shell}; + +const PREREQUISITES: [Prerequisite; 6] = [ + Prerequisite { + name: "git", + download_link: "https://git-scm.com/book/en/v2/Getting-Started-Installing-Git", + }, + Prerequisite { + name: "docker", + download_link: "https://docs.docker.com/get-docker/", + }, + Prerequisite { + name: "docker-compose", + download_link: "https://docs.docker.com/compose/install/", + }, + Prerequisite { + name: "forge", + download_link: "https://book.getfoundry.sh/getting-started/installation", + }, + Prerequisite { + name: "cargo", + download_link: "https://doc.rust-lang.org/cargo/getting-started/installation.html", + }, + Prerequisite { + name: "yarn", + download_link: "https://yarnpkg.com/getting-started/install", + }, +]; + +struct Prerequisite { + name: &'static str, + download_link: &'static str, +} + +pub fn check_prerequisites(shell: &Shell) { + let mut missing_prerequisites = vec![]; + + for prerequisite in &PREREQUISITES { + if !check_prerequisite(shell, prerequisite.name) { + missing_prerequisites.push(prerequisite); + } + } + + if !missing_prerequisites.is_empty() { + logger::error("Prerequisite check has failed"); + logger::error_note( + "The following prerequisites are missing", + &missing_prerequisites + .iter() + .map(|prerequisite| { + format!("- {} ({})", prerequisite.name, prerequisite.download_link) + }) + .collect::>() + .join("\n"), + ); + logger::outro("Failed"); + std::process::exit(1); + } +} + +fn check_prerequisite(shell: &Shell, name: &str) -> bool { + Cmd::new(cmd!(shell, "which {name}")).run().is_ok() +} diff --git a/zk_toolbox/crates/common/src/prompt/confirm.rs b/zk_toolbox/crates/common/src/prompt/confirm.rs new file mode 100644 index 000000000000..195654e7d65a --- /dev/null +++ b/zk_toolbox/crates/common/src/prompt/confirm.rs @@ -0,0 +1,24 @@ +use cliclack::Confirm; +use std::fmt::Display; + +pub struct PromptConfirm { + inner: Confirm, +} + +impl PromptConfirm { + pub fn new(question: impl Display) -> Self { + Self { + inner: Confirm::new(question), + } + } + + pub fn default(self, default: bool) -> Self { + Self { + inner: self.inner.initial_value(default), + } + } + + pub fn ask(mut self) -> bool { + self.inner.interact().unwrap() + } +} diff --git a/zk_toolbox/crates/common/src/prompt/input.rs b/zk_toolbox/crates/common/src/prompt/input.rs new file mode 100644 index 000000000000..c2cd275ecb50 --- /dev/null +++ b/zk_toolbox/crates/common/src/prompt/input.rs @@ -0,0 +1,40 @@ +use cliclack::{Input, Validate}; +use std::str::FromStr; + +pub struct Prompt { + inner: Input, +} + +impl Prompt { + pub fn new(question: &str) -> Self { + Self { + inner: Input::new(question), + } + } + + pub fn allow_empty(mut self) -> Self { + self.inner = self.inner.required(false); + self + } + + pub fn default(mut self, default: &str) -> Self { + self.inner = self.inner.default_input(default); + self + } + + pub fn validate_with(mut self, f: F) -> Self + where + F: Validate + 'static, + F::Err: ToString, + { + self.inner = self.inner.validate(f); + self + } + + pub fn ask(mut self) -> T + where + T: FromStr, + { + self.inner.interact().unwrap() + } +} diff --git a/zk_toolbox/crates/common/src/prompt/mod.rs b/zk_toolbox/crates/common/src/prompt/mod.rs new file mode 100644 index 000000000000..cd642302c06c --- /dev/null +++ b/zk_toolbox/crates/common/src/prompt/mod.rs @@ -0,0 +1,25 @@ +mod confirm; +mod input; +mod select; + +use cliclack::{Theme, ThemeState}; +pub use confirm::PromptConfirm; +use console::Style; +pub use input::Prompt; +pub use select::PromptSelect; + +pub struct CliclackTheme; + +impl Theme for CliclackTheme { + fn bar_color(&self, state: &ThemeState) -> Style { + match state { + ThemeState::Active => Style::new().cyan(), + ThemeState::Error(_) => Style::new().yellow(), + _ => Style::new().cyan().dim(), + } + } +} + +pub fn init_prompt_theme() { + cliclack::set_theme(CliclackTheme); +} diff --git a/zk_toolbox/crates/common/src/prompt/select.rs b/zk_toolbox/crates/common/src/prompt/select.rs new file mode 100644 index 000000000000..5908cf0a8fee --- /dev/null +++ b/zk_toolbox/crates/common/src/prompt/select.rs @@ -0,0 +1,32 @@ +use cliclack::Select; + +pub struct PromptSelect { + inner: Select, +} + +impl PromptSelect +where + T: Clone + Eq, +{ + pub fn new(question: &str, items: I) -> Self + where + T: ToString + Clone, + I: IntoIterator, + { + let items = items + .into_iter() + .map(|item| { + let label = item.to_string(); + let hint = ""; + (item, label, hint) + }) + .collect::>(); + Self { + inner: Select::new(question).items(&items), + } + } + + pub fn ask(mut self) -> T { + self.inner.interact().unwrap() + } +} diff --git a/zk_toolbox/crates/common/src/slugify.rs b/zk_toolbox/crates/common/src/slugify.rs new file mode 100644 index 000000000000..a934a56b5527 --- /dev/null +++ b/zk_toolbox/crates/common/src/slugify.rs @@ -0,0 +1,3 @@ +pub fn slugify(data: &str) -> String { + data.trim().replace(" ", "-") +} diff --git a/zk_toolbox/crates/common/src/term/logger.rs b/zk_toolbox/crates/common/src/term/logger.rs new file mode 100644 index 000000000000..9e13c2958078 --- /dev/null +++ b/zk_toolbox/crates/common/src/term/logger.rs @@ -0,0 +1,114 @@ +use std::fmt::Display; + +use cliclack::{intro as cliclak_intro, log, outro as cliclak_outro, Theme, ThemeState}; +use console::{style, Emoji, Term}; +use serde::Serialize; + +use crate::prompt::CliclackTheme; + +const S_BAR: Emoji = Emoji("│", "|"); + +fn term_write(msg: impl Display) { + let msg = &format!("{}", msg); + Term::stderr().write_str(msg).unwrap(); +} + +pub fn intro() { + cliclak_intro(style(" zkSync toolbox ").on_cyan().black()).unwrap(); +} + +pub fn outro(msg: impl Display) { + cliclak_outro(msg).unwrap(); +} + +pub fn info(msg: impl Display) { + log::info(msg).unwrap(); +} + +pub fn debug(msg: impl Display) { + let msg = &format!("{}", msg); + let log = CliclackTheme.format_log(msg, style("âš™").dim().to_string().as_str()); + Term::stderr().write_str(&log).unwrap(); +} + +pub fn warn(msg: impl Display) { + log::warning(msg).unwrap(); +} + +pub fn error(msg: impl Display) { + log::error(style(msg).red()).unwrap(); +} + +pub fn success(msg: impl Display) { + log::success(msg).unwrap(); +} + +pub fn raw(msg: impl Display) { + log::step(msg).unwrap(); +} + +pub fn note(msg: impl Display, content: impl Display) { + cliclack::note(msg, content).unwrap(); +} + +pub fn error_note(msg: &str, content: &str) { + let symbol = CliclackTheme.state_symbol(&ThemeState::Submit); + let note = CliclackTheme + .format_note(msg, content) + .replace(&symbol, &CliclackTheme.error_symbol()); + term_write(note); +} + +pub fn object_to_string(obj: impl Serialize) -> String { + let json = serde_json::to_value(obj).unwrap(); + + fn print_object(key: &str, value: &str, indentation: usize) -> String { + format!( + "{:indent$}∙ {} {}\n", + "", + style(format!("{key}:")).bold(), + style(value), + indent = indentation + ) + } + + fn print_header(header: &str, indentation: usize) -> String { + format!( + "{:indent$}∙ {}\n", + "", + style(format!("{header}:")).bold(), + indent = indentation + ) + } + + fn traverse_json(json: &serde_json::Value, indent: usize) -> String { + let mut values = String::new(); + + if let serde_json::Value::Object(obj) = json { + for (key, value) in obj { + match value { + serde_json::Value::Object(_) => { + values.push_str(&print_header(key, indent)); + values.push_str(&traverse_json(value, indent + 2)); + } + _ => values.push_str(&print_object(key, &value.to_string(), indent)), + } + } + } + + values + } + + traverse_json(&json, 2) +} + +pub fn new_empty_line() { + term_write("\n"); +} + +pub fn new_line() { + term_write(format!( + "{}\n", + CliclackTheme.bar_color(&ThemeState::Submit).apply_to(S_BAR) + )) +} diff --git a/zk_toolbox/crates/common/src/term/mod.rs b/zk_toolbox/crates/common/src/term/mod.rs new file mode 100644 index 000000000000..a82083530671 --- /dev/null +++ b/zk_toolbox/crates/common/src/term/mod.rs @@ -0,0 +1,2 @@ +pub mod logger; +pub mod spinner; diff --git a/zk_toolbox/crates/common/src/term/spinner.rs b/zk_toolbox/crates/common/src/term/spinner.rs new file mode 100644 index 000000000000..3e9322ba636c --- /dev/null +++ b/zk_toolbox/crates/common/src/term/spinner.rs @@ -0,0 +1,37 @@ +use std::time::Instant; + +use cliclack::{spinner, ProgressBar}; + +use crate::config::global_config; + +/// Spinner is a helper struct to show a spinner while some operation is running. +pub struct Spinner { + msg: String, + pb: ProgressBar, + time: Instant, +} + +impl Spinner { + /// Create a new spinner with a message. + pub fn new(msg: &str) -> Self { + let pb = spinner(); + pb.start(msg); + if global_config().verbose { + pb.stop(msg); + } + Spinner { + msg: msg.to_owned(), + pb, + time: Instant::now(), + } + } + + /// Manually finish the spinner. + pub fn finish(self) { + self.pb.stop(format!( + "{} done in {} secs", + self.msg, + self.time.elapsed().as_secs_f64() + )); + } +} diff --git a/zk_toolbox/crates/common/src/wallets.rs b/zk_toolbox/crates/common/src/wallets.rs new file mode 100644 index 000000000000..1349f31ebebe --- /dev/null +++ b/zk_toolbox/crates/common/src/wallets.rs @@ -0,0 +1,64 @@ +use ethers::{ + core::rand::Rng, + signers::{coins_bip39::English, LocalWallet, MnemonicBuilder, Signer}, + types::{H160, H256}, +}; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Wallet { + pub address: H160, + pub private_key: Option, +} + +impl Wallet { + pub fn random(rng: &mut impl Rng) -> Self { + let private_key = H256(rng.gen()); + let local_wallet = LocalWallet::from_bytes(private_key.as_bytes()).unwrap(); + + Self { + address: local_wallet.address(), + private_key: Some(private_key), + } + } + + pub fn new_with_key(private_key: H256) -> Self { + let local_wallet = LocalWallet::from_bytes(private_key.as_bytes()).unwrap(); + Self { + address: local_wallet.address(), + private_key: Some(private_key), + } + } + + pub fn from_mnemonic(mnemonic: &str, base_path: &str, index: u32) -> anyhow::Result { + let wallet = MnemonicBuilder::::default() + .phrase(mnemonic) + .derivation_path(&format!("{}/{}", base_path, index))? + .build()?; + let private_key = H256::from_slice(&wallet.signer().to_bytes()); + Ok(Self::new_with_key(private_key)) + } + + pub fn empty() -> Self { + Self { + address: H160::zero(), + private_key: Some(H256::zero()), + } + } +} + +#[test] +fn test_load_localhost_wallets() { + let wallet = Wallet::from_mnemonic( + "stuff slice staff easily soup parent arm payment cotton trade scatter struggle", + "m/44'/60'/0'/0", + 1, + ) + .unwrap(); + assert_eq!( + wallet.address, + H160::from_slice( + ðers::utils::hex::decode("0xa61464658AfeAf65CccaaFD3a512b69A83B77618").unwrap() + ) + ); +} diff --git a/zk_toolbox/crates/zk_inception/Cargo.toml b/zk_toolbox/crates/zk_inception/Cargo.toml new file mode 100644 index 000000000000..5ae3dd20e64b --- /dev/null +++ b/zk_toolbox/crates/zk_inception/Cargo.toml @@ -0,0 +1,30 @@ +[package] +name = "zk_inception" +version.workspace = true +edition.workspace = true +homepage.workspace = true +license.workspace = true +authors.workspace = true +exclude.workspace = true +repository.workspace = true +description.workspace = true +keywords.workspace = true + +[dependencies] +anyhow.workspace = true +clap.workspace = true +cliclack.workspace = true +console.workspace = true +human-panic.workspace = true +serde_yaml.workspace = true +serde.workspace = true +serde_json.workspace = true +xshell.workspace = true +ethers.workspace = true +common.workspace = true +tokio.workspace = true +strum_macros.workspace = true +strum.workspace = true +toml.workspace = true +url.workspace = true +thiserror.workspace = true diff --git a/zk_toolbox/crates/zk_inception/src/accept_ownership.rs b/zk_toolbox/crates/zk_inception/src/accept_ownership.rs new file mode 100644 index 000000000000..8c331dd63e05 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/accept_ownership.rs @@ -0,0 +1,95 @@ +use common::{ + forge::{Forge, ForgeScript, ForgeScriptArgs}, + spinner::Spinner, +}; +use ethers::{abi::Address, types::H256}; +use xshell::Shell; + +use crate::forge_utils::check_the_balance; +use crate::{ + configs::{ + forge_interface::accept_ownership::AcceptOwnershipInput, EcosystemConfig, SaveConfig, + }, + consts::ACCEPT_GOVERNANCE, + forge_utils::fill_forge_private_key, +}; + +pub async fn accept_admin( + shell: &Shell, + ecosystem_config: &EcosystemConfig, + governor_contract: Address, + governor: Option, + target_address: Address, + forge_args: &ForgeScriptArgs, + l1_rpc_url: String, +) -> anyhow::Result<()> { + let foundry_contracts_path = ecosystem_config.path_to_foundry(); + let forge = Forge::new(&foundry_contracts_path) + .script(&ACCEPT_GOVERNANCE.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(l1_rpc_url) + .with_broadcast() + .with_signature("acceptAdmin()"); + accept_ownership( + shell, + ecosystem_config, + governor_contract, + governor, + target_address, + forge, + ) + .await +} + +pub async fn accept_owner( + shell: &Shell, + ecosystem_config: &EcosystemConfig, + governor_contract: Address, + governor: Option, + target_address: Address, + forge_args: &ForgeScriptArgs, + l1_rpc_url: String, +) -> anyhow::Result<()> { + let foundry_contracts_path = ecosystem_config.path_to_foundry(); + let forge = Forge::new(&foundry_contracts_path) + .script(&ACCEPT_GOVERNANCE.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(l1_rpc_url) + .with_broadcast() + .with_signature("acceptOwner()"); + accept_ownership( + shell, + ecosystem_config, + governor_contract, + governor, + target_address, + forge, + ) + .await +} + +async fn accept_ownership( + shell: &Shell, + ecosystem_config: &EcosystemConfig, + governor_contract: Address, + governor: Option, + target_address: Address, + mut forge: ForgeScript, +) -> anyhow::Result<()> { + let input = AcceptOwnershipInput { + target_addr: target_address, + governor: governor_contract, + }; + input.save( + shell, + ACCEPT_GOVERNANCE.input(&ecosystem_config.link_to_code), + )?; + + forge = fill_forge_private_key(forge, governor)?; + + check_the_balance(&forge).await?; + let spinner = Spinner::new("Accepting governance"); + forge.run(shell)?; + spinner.finish(); + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/args/mod.rs b/zk_toolbox/crates/zk_inception/src/commands/args/mod.rs new file mode 100644 index 000000000000..bf1457ba92c6 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/args/mod.rs @@ -0,0 +1,3 @@ +mod run_server; + +pub use run_server::*; diff --git a/zk_toolbox/crates/zk_inception/src/commands/args/run_server.rs b/zk_toolbox/crates/zk_inception/src/commands/args/run_server.rs new file mode 100644 index 000000000000..7ae370d83879 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/args/run_server.rs @@ -0,0 +1,10 @@ +use clap::Parser; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Serialize, Deserialize, Parser)] +pub struct RunServerArgs { + #[clap(long, help = "Components of server to run")] + pub components: Option>, + #[clap(long, help = "Run server in genesis mode")] + pub genesis: bool, +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/args/create.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/args/create.rs new file mode 100644 index 000000000000..6afb46cbfb60 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/args/create.rs @@ -0,0 +1,146 @@ +use std::{path::PathBuf, str::FromStr}; + +use clap::Parser; +use common::{slugify, Prompt, PromptConfirm, PromptSelect}; +use ethers::types::H160; +use serde::{Deserialize, Serialize}; +use strum::IntoEnumIterator; +use strum_macros::{Display, EnumIter}; + +use crate::{ + defaults::L2_CHAIN_ID, + types::{BaseToken, L1BatchCommitDataGeneratorMode, ProverMode}, + wallets::WalletCreation, +}; + +#[derive(Debug, Serialize, Deserialize, Parser)] +pub struct ChainCreateArgs { + #[arg(long)] + pub chain_name: Option, + #[arg(value_parser = clap::value_parser!(u32).range(1..))] + pub chain_id: Option, + #[clap(long, help = "Prover options", value_enum)] + pub prover_mode: Option, + #[clap(long, help = "Wallet option", value_enum)] + pub wallet_creation: Option, + #[clap(long, help = "Wallet path")] + pub wallet_path: Option, + #[clap(long, help = "Commit data generation mode")] + pub l1_batch_commit_data_generator_mode: Option, + #[clap(long, help = "Base token address")] + pub base_token_address: Option, + #[clap(long, help = "Base token nominator")] + pub base_token_price_nominator: Option, + #[clap(long, help = "Base token denominator")] + pub base_token_price_denominator: Option, + #[clap(long, help = "Set as default chain", default_missing_value = "true", num_args = 0..=1)] + pub set_as_default: Option, +} + +impl ChainCreateArgs { + pub fn fill_values_with_prompt(self, number_of_chains: u32) -> ChainCreateArgsFinal { + let mut chain_name = self + .chain_name + .unwrap_or_else(|| Prompt::new("How do you want to name the chain?").ask()); + chain_name = slugify(&chain_name); + + let chain_id = self.chain_id.unwrap_or_else(|| { + Prompt::new("What's the chain id?") + .default(&(L2_CHAIN_ID + number_of_chains).to_string()) + .ask() + }); + + let wallet_creation = PromptSelect::new( + "Select how do you want to create the wallet", + WalletCreation::iter(), + ) + .ask(); + + let prover_version = + PromptSelect::new("Select the prover version", ProverMode::iter()).ask(); + + let l1_batch_commit_data_generator_mode = PromptSelect::new( + "Select the commit data generator mode", + L1BatchCommitDataGeneratorMode::iter(), + ) + .ask(); + + let wallet_path: Option = if self.wallet_creation == Some(WalletCreation::InFile) { + Some(self.wallet_path.unwrap_or_else(|| { + Prompt::new("What is the wallet path?") + .validate_with(|val: &String| { + PathBuf::from_str(val) + .map(|_| ()) + .map_err(|_| "Invalid path".to_string()) + }) + .ask() + })) + } else { + None + }; + + let base_token_selection = + PromptSelect::new("Select the base token to use", BaseTokenSelection::iter()).ask(); + let base_token = match base_token_selection { + BaseTokenSelection::Eth => BaseToken::eth(), + BaseTokenSelection::Custom => { + let number_validator = |val: &String| -> Result<(), String> { + let Ok(val) = val.parse::() else { + return Err("Numer is not zero".to_string()); + }; + if val == 0 { + return Err("Number should be greater than 0".to_string()); + } + Ok(()) + }; + let address: H160 = Prompt::new("What is the base token address?").ask(); + let nominator = Prompt::new("What is the base token price nominator?") + .validate_with(number_validator) + .ask(); + let denominator = Prompt::new("What is the base token price denominator?") + .validate_with(number_validator) + .ask(); + BaseToken { + address, + nominator, + denominator, + } + } + }; + + let set_as_default = self.set_as_default.unwrap_or_else(|| { + PromptConfirm::new("Set this chain as default?") + .default(true) + .ask() + }); + + ChainCreateArgsFinal { + chain_name, + chain_id, + prover_version, + wallet_creation, + l1_batch_commit_data_generator_mode, + wallet_path, + base_token, + set_as_default, + } + } +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +pub struct ChainCreateArgsFinal { + pub chain_name: String, + pub chain_id: u32, + pub prover_version: ProverMode, + pub wallet_creation: WalletCreation, + pub l1_batch_commit_data_generator_mode: L1BatchCommitDataGeneratorMode, + pub wallet_path: Option, + pub base_token: BaseToken, + pub set_as_default: bool, +} + +#[derive(Debug, Clone, EnumIter, Display, PartialEq, Eq)] +enum BaseTokenSelection { + Eth, + Custom, +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/args/genesis.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/args/genesis.rs new file mode 100644 index 000000000000..b24956c70c12 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/args/genesis.rs @@ -0,0 +1,101 @@ +use clap::Parser; +use common::{slugify, Prompt}; +use serde::{Deserialize, Serialize}; +use url::Url; + +use crate::{ + configs::{ChainConfig, DatabaseConfig, DatabasesConfig}, + defaults::{generate_db_names, DBNames, DATABASE_PROVER_URL, DATABASE_SERVER_URL}, +}; + +#[derive(Debug, Clone, Serialize, Deserialize, Parser, Default)] +pub struct GenesisArgs { + #[clap(long, help = "Server database url without database name")] + pub server_db_url: Option, + #[clap(long, help = "Server database name")] + pub server_db_name: Option, + #[clap(long, help = "Prover database url without database name")] + pub prover_db_url: Option, + #[clap(long, help = "Prover database name")] + pub prover_db_name: Option, + #[clap(long, short, help = "Use default database urls and names")] + pub use_default: bool, + #[clap(long, short, action)] + pub dont_drop: bool, +} + +impl GenesisArgs { + pub fn fill_values_with_prompt(self, config: &ChainConfig) -> GenesisArgsFinal { + let DBNames { + server_name, + prover_name, + } = generate_db_names(config); + let chain_name = config.name.clone(); + if self.use_default { + GenesisArgsFinal { + server_db_url: DATABASE_SERVER_URL.to_string(), + server_db_name: server_name, + prover_db_url: DATABASE_PROVER_URL.to_string(), + prover_db_name: prover_name, + dont_drop: self.dont_drop, + } + } else { + let server_db_url = self.server_db_url.unwrap_or_else(|| { + Prompt::new(&format!( + "Please provide server database url for chain {chain_name}" + )) + .default(DATABASE_SERVER_URL) + .ask() + }); + let server_db_name = slugify(&self.server_db_name.unwrap_or_else(|| { + Prompt::new(&format!( + "Please provide server database name for chain {chain_name}" + )) + .default(&server_name) + .ask() + })); + let prover_db_url = self.prover_db_url.unwrap_or_else(|| { + Prompt::new(&format!( + "Please provide prover database url for chain {chain_name}" + )) + .default(DATABASE_PROVER_URL) + .ask() + }); + let prover_db_name = slugify(&self.prover_db_name.unwrap_or_else(|| { + Prompt::new(&format!( + "Please provide prover database name for chain {chain_name}" + )) + .default(&prover_name) + .ask() + })); + GenesisArgsFinal { + server_db_url, + server_db_name, + prover_db_url, + prover_db_name, + dont_drop: self.dont_drop, + } + } + } +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct GenesisArgsFinal { + pub server_db_url: String, + pub server_db_name: String, + pub prover_db_url: String, + pub prover_db_name: String, + pub dont_drop: bool, +} + +impl GenesisArgsFinal { + pub fn databases_config(&self) -> anyhow::Result { + let server_url = Url::parse(&self.server_db_url)?; + let prover_url = Url::parse(&self.prover_db_url)?; + + Ok(DatabasesConfig { + server: DatabaseConfig::new(server_url, self.server_db_name.clone()), + prover: DatabaseConfig::new(prover_url, self.prover_db_name.clone()), + }) + } +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/args/init.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/args/init.rs new file mode 100644 index 000000000000..aaa6fb2f0ffa --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/args/init.rs @@ -0,0 +1,64 @@ +use clap::Parser; +use common::forge::ForgeScriptArgs; +use common::Prompt; +use serde::{Deserialize, Serialize}; +use url::Url; + +use super::genesis::GenesisArgsFinal; +use crate::defaults::LOCAL_RPC_URL; +use crate::types::L1Network; +use crate::{commands::chain::args::genesis::GenesisArgs, configs::ChainConfig}; + +#[derive(Debug, Clone, Serialize, Deserialize, Parser)] +pub struct InitArgs { + /// All ethereum environment related arguments + #[clap(flatten)] + #[serde(flatten)] + pub forge_args: ForgeScriptArgs, + #[clap(flatten, next_help_heading = "Genesis options")] + #[serde(flatten)] + pub genesis_args: GenesisArgs, + #[clap(long, default_missing_value = "true", num_args = 0..=1)] + pub deploy_paymaster: Option, + #[clap(long, help = "L1 RPC URL")] + pub l1_rpc_url: Option, +} + +impl InitArgs { + pub fn fill_values_with_prompt(self, config: &ChainConfig) -> InitArgsFinal { + let deploy_paymaster = self.deploy_paymaster.unwrap_or_else(|| { + common::PromptConfirm::new("Do you want to deploy a test paymaster?") + .default(true) + .ask() + }); + + let l1_rpc_url = self.l1_rpc_url.unwrap_or_else(|| { + let mut prompt = Prompt::new("What is the RPC URL of the L1 network?"); + if config.l1_network == L1Network::Localhost { + prompt = prompt.default(LOCAL_RPC_URL); + } + prompt + .validate_with(|val: &String| -> Result<(), String> { + Url::parse(val) + .map(|_| ()) + .map_err(|_| "Invalid RPC url".to_string()) + }) + .ask() + }); + + InitArgsFinal { + forge_args: self.forge_args, + genesis_args: self.genesis_args.fill_values_with_prompt(config), + deploy_paymaster, + l1_rpc_url, + } + } +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +pub struct InitArgsFinal { + pub forge_args: ForgeScriptArgs, + pub genesis_args: GenesisArgsFinal, + pub deploy_paymaster: bool, + pub l1_rpc_url: String, +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/args/mod.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/args/mod.rs new file mode 100644 index 000000000000..08f39a90a843 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/args/mod.rs @@ -0,0 +1,3 @@ +pub mod create; +pub mod genesis; +pub mod init; diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/create.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/create.rs new file mode 100644 index 000000000000..2be7044d64b8 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/create.rs @@ -0,0 +1,81 @@ +use std::cell::OnceCell; + +use common::{logger, spinner::Spinner}; +use xshell::Shell; + +use crate::{ + commands::chain::args::create::{ChainCreateArgs, ChainCreateArgsFinal}, + configs::{ChainConfig, EcosystemConfig, SaveConfig}, + consts::{CONFIG_NAME, LOCAL_CONFIGS_PATH, LOCAL_DB_PATH, WALLETS_FILE}, + types::ChainId, + wallets::create_wallets, +}; + +pub fn run(args: ChainCreateArgs, shell: &Shell) -> anyhow::Result<()> { + let mut ecosystem_config = EcosystemConfig::from_file(shell)?; + create(args, &mut ecosystem_config, shell) +} + +fn create( + args: ChainCreateArgs, + ecosystem_config: &mut EcosystemConfig, + shell: &Shell, +) -> anyhow::Result<()> { + let args = args.fill_values_with_prompt(ecosystem_config.list_of_chains().len() as u32); + + logger::note("Selected config:", logger::object_to_string(&args)); + logger::info("Creating chain"); + + let spinner = Spinner::new("Creating chain configurations..."); + let name = args.chain_name.clone(); + let set_as_default = args.set_as_default; + create_chain_inner(args, ecosystem_config, shell)?; + if set_as_default { + ecosystem_config.default_chain = name; + ecosystem_config.save(shell, CONFIG_NAME)?; + } + spinner.finish(); + + logger::success("Chain created successfully"); + + Ok(()) +} + +pub(crate) fn create_chain_inner( + args: ChainCreateArgsFinal, + ecosystem_config: &EcosystemConfig, + shell: &Shell, +) -> anyhow::Result<()> { + let default_chain_name = args.chain_name.clone(); + let chain_path = ecosystem_config.chains.join(&default_chain_name); + let chain_configs_path = shell.create_dir(chain_path.join(LOCAL_CONFIGS_PATH))?; + let chain_db_path = chain_path.join(LOCAL_DB_PATH); + let chain_id = ecosystem_config.list_of_chains().len() as u32; + + let chain_config = ChainConfig { + id: chain_id, + name: default_chain_name.clone(), + chain_id: ChainId::from(args.chain_id), + prover_version: args.prover_version, + l1_network: ecosystem_config.l1_network, + link_to_code: ecosystem_config.link_to_code.clone(), + rocks_db_path: chain_db_path, + configs: chain_configs_path.clone(), + l1_batch_commit_data_generator_mode: args.l1_batch_commit_data_generator_mode, + base_token: args.base_token, + wallet_creation: args.wallet_creation, + shell: OnceCell::from(shell.clone()), + }; + + create_wallets( + shell, + &chain_config.configs.join(WALLETS_FILE), + &ecosystem_config.link_to_code, + chain_id, + args.wallet_creation, + args.wallet_path, + )?; + + chain_config.save(shell, chain_path.join(CONFIG_NAME))?; + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/deploy_paymaster.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/deploy_paymaster.rs new file mode 100644 index 000000000000..177b27cb2ff5 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/deploy_paymaster.rs @@ -0,0 +1,59 @@ +use anyhow::Context; +use common::{ + config::global_config, + forge::{Forge, ForgeScriptArgs}, + spinner::Spinner, +}; +use xshell::Shell; + +use crate::forge_utils::check_the_balance; +use crate::{ + configs::{ + forge_interface::paymaster::{DeployPaymasterInput, DeployPaymasterOutput}, + update_paymaster, ChainConfig, EcosystemConfig, ReadConfig, SaveConfig, + }, + consts::DEPLOY_PAYMASTER, + forge_utils::fill_forge_private_key, +}; + +pub async fn run(args: ForgeScriptArgs, shell: &Shell) -> anyhow::Result<()> { + let chain_name = global_config().chain_name.clone(); + let ecosystem_config = EcosystemConfig::from_file(shell)?; + let chain_config = ecosystem_config + .load_chain(chain_name) + .context("Chain not initialized. Please create a chain first")?; + deploy_paymaster(shell, &chain_config, args).await +} + +pub async fn deploy_paymaster( + shell: &Shell, + chain_config: &ChainConfig, + forge_args: ForgeScriptArgs, +) -> anyhow::Result<()> { + let input = DeployPaymasterInput::new(chain_config)?; + let foundry_contracts_path = chain_config.path_to_foundry(); + input.save(shell, DEPLOY_PAYMASTER.input(&chain_config.link_to_code))?; + let secrets = chain_config.get_secrets_config()?; + + let mut forge = Forge::new(&foundry_contracts_path) + .script(&DEPLOY_PAYMASTER.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(secrets.l1.l1_rpc_url.clone()) + .with_broadcast(); + + forge = fill_forge_private_key( + forge, + chain_config.get_wallets_config()?.governor_private_key(), + )?; + + let spinner = Spinner::new("Deploying paymaster"); + check_the_balance(&forge).await?; + forge.run(shell)?; + spinner.finish(); + + let output = + DeployPaymasterOutput::read(shell, DEPLOY_PAYMASTER.output(&chain_config.link_to_code))?; + + update_paymaster(shell, chain_config, &output)?; + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/genesis.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/genesis.rs new file mode 100644 index 000000000000..4fe2f0bbb118 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/genesis.rs @@ -0,0 +1,129 @@ +use std::path::PathBuf; + +use anyhow::Context; +use common::{ + config::global_config, + db::{drop_db_if_exists, init_db, migrate_db}, + logger, + spinner::Spinner, +}; +use xshell::Shell; + +use super::args::genesis::GenesisArgsFinal; +use crate::{ + commands::chain::args::genesis::GenesisArgs, + configs::{ + update_database_secrets, update_general_config, ChainConfig, DatabasesConfig, + EcosystemConfig, + }, + server::{RunServer, ServerMode}, +}; + +const SERVER_MIGRATIONS: &str = "core/lib/dal/migrations"; +const PROVER_MIGRATIONS: &str = "prover/prover_dal/migrations"; + +pub async fn run(args: GenesisArgs, shell: &Shell) -> anyhow::Result<()> { + let chain_name = global_config().chain_name.clone(); + let ecosystem_config = EcosystemConfig::from_file(shell)?; + let chain_config = ecosystem_config + .load_chain(chain_name) + .context("Chain not initialized. Please create a chain first")?; + let args = args.fill_values_with_prompt(&chain_config); + + genesis(args, shell, &chain_config).await?; + logger::outro("Genesis completed successfully"); + + Ok(()) +} + +pub async fn genesis( + args: GenesisArgsFinal, + shell: &Shell, + config: &ChainConfig, +) -> anyhow::Result<()> { + // Clean the rocksdb + shell.remove_path(&config.rocks_db_path)?; + shell.create_dir(&config.rocks_db_path)?; + + let db_config = args + .databases_config() + .context("Database config was not fully generated")?; + update_general_config(shell, config)?; + update_database_secrets(shell, config, &db_config)?; + + logger::note( + "Selected config:", + logger::object_to_string(serde_json::json!({ + "chain_config": config, + "db_config": db_config, + })), + ); + logger::info("Starting genesis process"); + + let spinner = Spinner::new("Initializing databases..."); + initialize_databases( + shell, + db_config, + config.link_to_code.clone(), + args.dont_drop, + ) + .await?; + spinner.finish(); + + let spinner = Spinner::new( + "Starting the genesis of the server. Building the entire server may take a lot of time...", + ); + run_server_genesis(config, shell)?; + spinner.finish(); + + Ok(()) +} + +async fn initialize_databases( + shell: &Shell, + db_config: DatabasesConfig, + link_to_code: PathBuf, + dont_drop: bool, +) -> anyhow::Result<()> { + let path_to_server_migration = link_to_code.join(SERVER_MIGRATIONS); + + if global_config().verbose { + logger::debug("Initializing server database") + } + if !dont_drop { + drop_db_if_exists(&db_config.server.base_url, &db_config.server.database_name) + .await + .context("Failed to drop server database")?; + init_db(&db_config.server.base_url, &db_config.server.database_name).await?; + } + migrate_db( + shell, + path_to_server_migration, + &db_config.server.full_url(), + ) + .await?; + + if global_config().verbose { + logger::debug("Initializing prover database") + } + if !dont_drop { + drop_db_if_exists(&db_config.prover.base_url, &db_config.prover.database_name) + .await + .context("Failed to drop prover database")?; + init_db(&db_config.prover.base_url, &db_config.prover.database_name).await?; + } + let path_to_prover_migration = link_to_code.join(PROVER_MIGRATIONS); + migrate_db( + shell, + path_to_prover_migration, + &db_config.prover.full_url(), + ) + .await?; + + Ok(()) +} + +fn run_server_genesis(chain_config: &ChainConfig, shell: &Shell) -> anyhow::Result<()> { + let server = RunServer::new(None, chain_config); + server.run(shell, ServerMode::Genesis) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/init.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/init.rs new file mode 100644 index 000000000000..80776ab277df --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/init.rs @@ -0,0 +1,132 @@ +use anyhow::Context; +use common::{ + config::global_config, + forge::{Forge, ForgeScriptArgs}, + logger, + spinner::Spinner, +}; +use xshell::Shell; + +use super::args::init::InitArgsFinal; +use crate::configs::update_l1_rpc_url_secret; +use crate::forge_utils::check_the_balance; +use crate::{ + accept_ownership::accept_admin, + commands::chain::{ + args::init::InitArgs, deploy_paymaster, genesis::genesis, initialize_bridges, + }, + configs::{ + copy_configs, + forge_interface::register_chain::{ + input::RegisterChainL1Config, output::RegisterChainOutput, + }, + update_genesis, update_l1_contracts, ChainConfig, ContractsConfig, EcosystemConfig, + ReadConfig, SaveConfig, + }, + consts::{CONTRACTS_FILE, REGISTER_CHAIN}, + forge_utils::fill_forge_private_key, +}; + +pub(crate) async fn run(args: InitArgs, shell: &Shell) -> anyhow::Result<()> { + let chain_name = global_config().chain_name.clone(); + let config = EcosystemConfig::from_file(shell)?; + let chain_config = config.load_chain(chain_name).context("Chain not found")?; + let mut args = args.fill_values_with_prompt(&chain_config); + + logger::note("Selected config:", logger::object_to_string(&chain_config)); + logger::info("Initializing chain"); + + init(&mut args, shell, &config, &chain_config).await?; + + logger::success("Chain initialized successfully"); + Ok(()) +} + +pub async fn init( + init_args: &mut InitArgsFinal, + shell: &Shell, + ecosystem_config: &EcosystemConfig, + chain_config: &ChainConfig, +) -> anyhow::Result<()> { + copy_configs(shell, &ecosystem_config.link_to_code, &chain_config.configs)?; + + update_genesis(shell, chain_config)?; + update_l1_rpc_url_secret(shell, chain_config, init_args.l1_rpc_url.clone())?; + let mut contracts_config = + ContractsConfig::read(shell, ecosystem_config.config.join(CONTRACTS_FILE))?; + contracts_config.l1.base_token_addr = chain_config.base_token.address; + // Copy ecosystem contracts + contracts_config.save(shell, chain_config.configs.join(CONTRACTS_FILE))?; + + let spinner = Spinner::new("Registering chain..."); + contracts_config = register_chain( + shell, + init_args.forge_args.clone(), + ecosystem_config, + chain_config, + init_args.l1_rpc_url.clone(), + ) + .await?; + spinner.finish(); + let spinner = Spinner::new("Accepting admin..."); + accept_admin( + shell, + ecosystem_config, + contracts_config.l1.governance_addr, + chain_config.get_wallets_config()?.governor_private_key(), + contracts_config.l1.diamond_proxy_addr, + &init_args.forge_args.clone(), + init_args.l1_rpc_url.clone(), + ) + .await?; + spinner.finish(); + + initialize_bridges::initialize_bridges( + shell, + chain_config, + ecosystem_config, + init_args.forge_args.clone(), + ) + .await?; + + if init_args.deploy_paymaster { + deploy_paymaster::deploy_paymaster(shell, chain_config, init_args.forge_args.clone()) + .await?; + } + + genesis(init_args.genesis_args.clone(), shell, chain_config) + .await + .context("Unable to perform genesis on the database")?; + + Ok(()) +} + +async fn register_chain( + shell: &Shell, + forge_args: ForgeScriptArgs, + config: &EcosystemConfig, + chain_config: &ChainConfig, + l1_rpc_url: String, +) -> anyhow::Result { + let deploy_config_path = REGISTER_CHAIN.input(&config.link_to_code); + + let contracts = config + .get_contracts_config() + .context("Ecosystem contracts config not found")?; + let deploy_config = RegisterChainL1Config::new(chain_config, &contracts)?; + deploy_config.save(shell, deploy_config_path)?; + + let mut forge = Forge::new(&config.path_to_foundry()) + .script(®ISTER_CHAIN.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(l1_rpc_url) + .with_broadcast(); + + forge = fill_forge_private_key(forge, config.get_wallets()?.governor_private_key())?; + check_the_balance(&forge).await?; + forge.run(shell)?; + + let register_chain_output = + RegisterChainOutput::read(shell, REGISTER_CHAIN.output(&chain_config.link_to_code))?; + update_l1_contracts(shell, chain_config, ®ister_chain_output) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/initialize_bridges.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/initialize_bridges.rs new file mode 100644 index 000000000000..ebeacc1c15af --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/initialize_bridges.rs @@ -0,0 +1,74 @@ +use std::path::Path; + +use anyhow::Context; +use common::{ + cmd::Cmd, + config::global_config, + forge::{Forge, ForgeScriptArgs}, + spinner::Spinner, +}; +use xshell::{cmd, Shell}; + +use crate::forge_utils::check_the_balance; +use crate::{ + configs::{ + forge_interface::initialize_bridges::{ + input::InitializeBridgeInput, output::InitializeBridgeOutput, + }, + update_l2_shared_bridge, ChainConfig, EcosystemConfig, ReadConfig, SaveConfig, + }, + consts::INITIALIZE_BRIDGES, + forge_utils::fill_forge_private_key, +}; + +pub async fn run(args: ForgeScriptArgs, shell: &Shell) -> anyhow::Result<()> { + let chain_name = global_config().chain_name.clone(); + let ecosystem_config = EcosystemConfig::from_file(shell)?; + let chain_config = ecosystem_config + .load_chain(chain_name) + .context("Chain not initialized. Please create a chain first")?; + + let spinner = Spinner::new("Initializing bridges"); + initialize_bridges(shell, &chain_config, &ecosystem_config, args).await?; + spinner.finish(); + + Ok(()) +} + +pub async fn initialize_bridges( + shell: &Shell, + chain_config: &ChainConfig, + ecosystem_config: &EcosystemConfig, + forge_args: ForgeScriptArgs, +) -> anyhow::Result<()> { + build_l2_contracts(shell, &ecosystem_config.link_to_code)?; + let input = InitializeBridgeInput::new(chain_config, ecosystem_config.era_chain_id)?; + let foundry_contracts_path = chain_config.path_to_foundry(); + let secrets = chain_config.get_secrets_config()?; + input.save(shell, INITIALIZE_BRIDGES.input(&chain_config.link_to_code))?; + + let mut forge = Forge::new(&foundry_contracts_path) + .script(&INITIALIZE_BRIDGES.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(secrets.l1.l1_rpc_url.clone()) + .with_broadcast(); + + forge = fill_forge_private_key( + forge, + ecosystem_config.get_wallets()?.governor_private_key(), + )?; + + check_the_balance(&forge).await?; + forge.run(shell)?; + + let output = + InitializeBridgeOutput::read(shell, INITIALIZE_BRIDGES.output(&chain_config.link_to_code))?; + + update_l2_shared_bridge(shell, chain_config, &output)?; + Ok(()) +} + +fn build_l2_contracts(shell: &Shell, link_to_code: &Path) -> anyhow::Result<()> { + let _dir_guard = shell.push_dir(link_to_code.join("contracts")); + Cmd::new(cmd!(shell, "yarn l2 build")).run() +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/chain/mod.rs b/zk_toolbox/crates/zk_inception/src/commands/chain/mod.rs new file mode 100644 index 000000000000..759b4aaea557 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/chain/mod.rs @@ -0,0 +1,38 @@ +pub(crate) mod args; +mod create; +pub mod deploy_paymaster; +pub mod genesis; +pub(crate) mod init; +mod initialize_bridges; + +pub(crate) use args::create::ChainCreateArgsFinal; +use clap::Subcommand; +use common::forge::ForgeScriptArgs; +pub(crate) use create::create_chain_inner; +use xshell::Shell; + +use crate::commands::chain::args::{create::ChainCreateArgs, genesis::GenesisArgs, init::InitArgs}; + +#[derive(Subcommand, Debug)] +pub enum ChainCommands { + /// Create a new chain, setting the necessary configurations for later initialization + Create(ChainCreateArgs), + /// Initialize chain, deploying necessary contracts and performing on-chain operations + Init(InitArgs), + /// Run server genesis + Genesis(GenesisArgs), + /// Initialize bridges on l2 + InitializeBridges(ForgeScriptArgs), + /// Initialize bridges on l2 + DeployPaymaster(ForgeScriptArgs), +} + +pub(crate) async fn run(shell: &Shell, args: ChainCommands) -> anyhow::Result<()> { + match args { + ChainCommands::Create(args) => create::run(args, shell), + ChainCommands::Init(args) => init::run(args, shell).await, + ChainCommands::Genesis(args) => genesis::run(args, shell).await, + ChainCommands::InitializeBridges(args) => initialize_bridges::run(args, shell).await, + ChainCommands::DeployPaymaster(args) => deploy_paymaster::run(args, shell).await, + } +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/containers.rs b/zk_toolbox/crates/zk_inception/src/commands/containers.rs new file mode 100644 index 000000000000..82bb2b48520e --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/containers.rs @@ -0,0 +1,75 @@ +use anyhow::{anyhow, Context}; +use common::{docker, logger, spinner::Spinner}; +use std::path::PathBuf; +use xshell::Shell; + +use crate::{configs::EcosystemConfig, consts::DOCKER_COMPOSE_FILE}; + +pub fn run(shell: &Shell) -> anyhow::Result<()> { + let ecosystem = + EcosystemConfig::from_file(shell).context("Failed to find ecosystem folder.")?; + + initialize_docker(shell, &ecosystem)?; + + logger::info("Starting containers"); + + let spinner = Spinner::new("Starting containers using docker..."); + start_containers(shell)?; + spinner.finish(); + + logger::outro("Containers started successfully"); + Ok(()) +} + +pub fn initialize_docker(shell: &Shell, ecosystem: &EcosystemConfig) -> anyhow::Result<()> { + if !shell.path_exists("volumes") { + create_docker_folders(shell)?; + }; + + if !shell.path_exists(DOCKER_COMPOSE_FILE) { + copy_dockerfile(shell, ecosystem.link_to_code.clone())?; + }; + + Ok(()) +} + +pub fn start_containers(shell: &Shell) -> anyhow::Result<()> { + while let Err(err) = docker::up(shell, DOCKER_COMPOSE_FILE) { + logger::error(err.to_string()); + if !common::PromptConfirm::new( + "Failed to start containers. Make sure there is nothing running on default ports for Ethereum node l1 and postgres. Want to try again?", + ).default(true) + .ask() + { + return Err(err); + } + } + Ok(()) +} + +fn create_docker_folders(shell: &Shell) -> anyhow::Result<()> { + shell.create_dir("volumes")?; + shell.create_dir("volumes/postgres")?; + shell.create_dir("volumes/reth")?; + shell.create_dir("volumes/reth/data")?; + Ok(()) +} + +fn copy_dockerfile(shell: &Shell, link_to_code: PathBuf) -> anyhow::Result<()> { + let docker_compose_file = link_to_code.join(DOCKER_COMPOSE_FILE); + + let docker_compose_text = shell.read_file(&docker_compose_file).map_err(|err| { + anyhow!( + "Failed to read docker compose file from {:?}: {}", + &docker_compose_file, + err + ) + })?; + let original_source = "./etc/reth/chaindata"; + let new_source = link_to_code.join(original_source); + let new_source = new_source.to_str().unwrap(); + + let data = docker_compose_text.replace(original_source, new_source); + shell.write_file(DOCKER_COMPOSE_FILE, data)?; + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/change_default.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/change_default.rs new file mode 100644 index 000000000000..041e6a2eb40a --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/change_default.rs @@ -0,0 +1,7 @@ +use clap::Parser; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Clone, Serialize, Deserialize, Parser)] +pub struct ChangeDefaultChain { + pub name: Option, +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/create.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/create.rs new file mode 100644 index 000000000000..259050bce049 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/create.rs @@ -0,0 +1,99 @@ +use std::path::PathBuf; + +use clap::Parser; +use common::{slugify, Prompt, PromptConfirm, PromptSelect}; +use serde::{Deserialize, Serialize}; +use strum::IntoEnumIterator; +use strum_macros::{Display, EnumIter}; + +use crate::{ + commands::chain::{args::create::ChainCreateArgs, ChainCreateArgsFinal}, + types::L1Network, + wallets::WalletCreation, +}; + +#[derive(Debug, Serialize, Deserialize, Parser)] +pub struct EcosystemCreateArgs { + #[arg(long)] + pub ecosystem_name: Option, + #[clap(long, help = "L1 Network", value_enum)] + pub l1_network: Option, + #[clap(long, help = "Code link")] + pub link_to_code: Option, + #[clap(flatten)] + #[serde(flatten)] + pub chain: ChainCreateArgs, + #[clap(long, help = "Start reth and postgres containers after creation", default_missing_value = "true", num_args = 0..=1)] + pub start_containers: Option, +} + +impl EcosystemCreateArgs { + pub fn fill_values_with_prompt(mut self) -> EcosystemCreateArgsFinal { + let mut ecosystem_name = self + .ecosystem_name + .unwrap_or_else(|| Prompt::new("How do you want to name the ecosystem?").ask()); + ecosystem_name = slugify(&ecosystem_name); + + let link_to_code = self.link_to_code.unwrap_or_else(|| { + let link_to_code_selection = PromptSelect::new( + "Select the origin of zksync-era repository", + LinkToCodeSelection::iter(), + ) + .ask(); + match link_to_code_selection { + LinkToCodeSelection::Clone => "".to_string(), + LinkToCodeSelection::Path => Prompt::new("Where's the code located?").ask(), + } + }); + + let l1_network = PromptSelect::new("Select the L1 network", L1Network::iter()).ask(); + + // Make the only chain as a default one + self.chain.set_as_default = Some(true); + + let chain = self.chain.fill_values_with_prompt(0); + + let start_containers = self.start_containers.unwrap_or_else(|| { + PromptConfirm::new( + "Do you want to start database and L1 containers after creating the ecosystem?", + ) + .default(true) + .ask() + }); + + EcosystemCreateArgsFinal { + ecosystem_name, + l1_network, + link_to_code, + wallet_creation: chain.wallet_creation, + wallet_path: chain.wallet_path.clone(), + chain_args: chain, + start_containers, + } + } +} + +#[derive(Debug, Serialize, Deserialize)] +pub struct EcosystemCreateArgsFinal { + pub ecosystem_name: String, + pub l1_network: L1Network, + pub link_to_code: String, + pub wallet_creation: WalletCreation, + pub wallet_path: Option, + pub chain_args: ChainCreateArgsFinal, + pub start_containers: bool, +} + +impl EcosystemCreateArgsFinal { + pub fn chain_config(&self) -> ChainCreateArgsFinal { + self.chain_args.clone() + } +} + +#[derive(Debug, Clone, EnumIter, Display, PartialEq, Eq)] +enum LinkToCodeSelection { + #[strum(serialize = "Clone for me (recommended)")] + Clone, + #[strum(serialize = "I have the code already")] + Path, +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/init.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/init.rs new file mode 100644 index 000000000000..e1bda4736ac8 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/init.rs @@ -0,0 +1,108 @@ +use std::path::PathBuf; + +use clap::Parser; +use common::{forge::ForgeScriptArgs, Prompt, PromptConfirm}; +use serde::{Deserialize, Serialize}; +use url::Url; + +use crate::commands::chain::args::genesis::GenesisArgs; +use crate::defaults::LOCAL_RPC_URL; +use crate::types::L1Network; + +#[derive(Debug, Clone, Serialize, Deserialize, Parser)] +pub struct EcosystemArgs { + /// Deploy ecosystem contracts + #[clap(long, default_missing_value = "true", num_args = 0..=1)] + pub deploy_ecosystem: Option, + /// Path to ecosystem contracts + #[clap(long)] + pub ecosystem_contracts_path: Option, + #[clap(long, help = "L1 RPC URL")] + pub l1_rpc_url: Option, +} + +impl EcosystemArgs { + pub fn fill_values_with_prompt(self, l1_network: L1Network) -> EcosystemArgsFinal { + let deploy_ecosystem = self.deploy_ecosystem.unwrap_or_else(|| { + PromptConfirm::new("Do you want to deploy ecosystem contracts? (Not needed if you already have an existing one)") + .default(true) + .ask() + }); + + let l1_rpc_url = self.l1_rpc_url.unwrap_or_else(|| { + let mut prompt = Prompt::new("What is the RPC URL of the L1 network?"); + if l1_network == L1Network::Localhost { + prompt = prompt.default(LOCAL_RPC_URL); + } + prompt + .validate_with(|val: &String| -> Result<(), String> { + Url::parse(val) + .map(|_| ()) + .map_err(|_| "Invalid RPC url".to_string()) + }) + .ask() + }); + EcosystemArgsFinal { + deploy_ecosystem, + ecosystem_contracts_path: self.ecosystem_contracts_path, + l1_rpc_url, + } + } +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct EcosystemArgsFinal { + pub deploy_ecosystem: bool, + pub ecosystem_contracts_path: Option, + pub l1_rpc_url: String, +} + +#[derive(Debug, Clone, Serialize, Deserialize, Parser)] +pub struct EcosystemInitArgs { + /// Deploy Paymaster contract + #[clap(long, default_missing_value = "true", num_args = 0..=1)] + pub deploy_paymaster: Option, + /// Deploy ERC20 contracts + #[clap(long, default_missing_value = "true", num_args = 0..=1)] + pub deploy_erc20: Option, + #[clap(flatten)] + #[serde(flatten)] + pub ecosystem: EcosystemArgs, + #[clap(flatten)] + #[serde(flatten)] + pub forge_args: ForgeScriptArgs, + #[clap(flatten, next_help_heading = "Genesis options")] + #[serde(flatten)] + pub genesis_args: GenesisArgs, +} + +impl EcosystemInitArgs { + pub fn fill_values_with_prompt(self, l1_network: L1Network) -> EcosystemInitArgsFinal { + let deploy_paymaster = self.deploy_paymaster.unwrap_or_else(|| { + PromptConfirm::new("Do you want to deploy paymaster?") + .default(true) + .ask() + }); + let deploy_erc20 = self.deploy_erc20.unwrap_or_else(|| { + PromptConfirm::new("Do you want to deploy some test ERC20s?") + .default(true) + .ask() + }); + let ecosystem = self.ecosystem.fill_values_with_prompt(l1_network); + + EcosystemInitArgsFinal { + deploy_paymaster, + deploy_erc20, + ecosystem, + forge_args: self.forge_args.clone(), + } + } +} + +#[derive(Debug, Serialize, Deserialize)] +pub struct EcosystemInitArgsFinal { + pub deploy_paymaster: bool, + pub deploy_erc20: bool, + pub ecosystem: EcosystemArgsFinal, + pub forge_args: ForgeScriptArgs, +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/mod.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/mod.rs new file mode 100644 index 000000000000..8a6048a8643b --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/args/mod.rs @@ -0,0 +1,3 @@ +pub mod change_default; +pub mod create; +pub mod init; diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/change_default.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/change_default.rs new file mode 100644 index 000000000000..2541e8af88ea --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/change_default.rs @@ -0,0 +1,29 @@ +use common::PromptSelect; +use xshell::Shell; + +use crate::{ + commands::ecosystem::args::change_default::ChangeDefaultChain, + configs::{EcosystemConfig, SaveConfig}, + consts::CONFIG_NAME, +}; + +pub fn run(args: ChangeDefaultChain, shell: &Shell) -> anyhow::Result<()> { + let mut ecosystem_config = EcosystemConfig::from_file(shell)?; + + let chains = ecosystem_config.list_of_chains(); + let chain_name = args.name.unwrap_or_else(|| { + PromptSelect::new("What chain you want to set as default?", &chains) + .ask() + .to_string() + }); + + if !chains.contains(&chain_name) { + anyhow::bail!( + "Chain with name {} doesnt exist, please choose one of {:?}", + chain_name, + &chains + ); + } + ecosystem_config.default_chain = chain_name; + ecosystem_config.save(shell, CONFIG_NAME) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/create.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/create.rs new file mode 100644 index 000000000000..d3548a154607 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/create.rs @@ -0,0 +1,123 @@ +use std::path::Path; +use std::{path::PathBuf, str::FromStr}; + +use anyhow::bail; +use common::{cmd::Cmd, logger, spinner::Spinner}; +use xshell::{cmd, Shell}; + +use crate::{ + commands::{ + chain::create_chain_inner, + containers::{initialize_docker, start_containers}, + ecosystem::{ + args::create::EcosystemCreateArgs, + create_configs::{create_erc20_deployment_config, create_initial_deployments_config}, + }, + }, + configs::{EcosystemConfig, EcosystemConfigFromFileError, SaveConfig}, + consts::{CONFIG_NAME, ERA_CHAIN_ID, LOCAL_CONFIGS_PATH, WALLETS_FILE, ZKSYNC_ERA_GIT_REPO}, + wallets::create_wallets, +}; + +pub fn run(args: EcosystemCreateArgs, shell: &Shell) -> anyhow::Result<()> { + match EcosystemConfig::from_file(shell) { + Ok(_) => bail!("Ecosystem already exists"), + Err(EcosystemConfigFromFileError::InvalidConfig { .. }) => { + bail!("Invalid ecosystem configuration") + } + Err(EcosystemConfigFromFileError::NotExists) => create(args, shell)?, + }; + + Ok(()) +} + +fn create(args: EcosystemCreateArgs, shell: &Shell) -> anyhow::Result<()> { + let args = args.fill_values_with_prompt(); + + logger::note("Selected config:", logger::object_to_string(&args)); + logger::info("Creating ecosystem"); + + let ecosystem_name = &args.ecosystem_name; + shell.create_dir(ecosystem_name)?; + shell.change_dir(ecosystem_name); + + let configs_path = shell.create_dir(LOCAL_CONFIGS_PATH)?; + + let link_to_code = if args.link_to_code.is_empty() { + let spinner = Spinner::new("Cloning zksync-era repository..."); + let link_to_code = clone_era_repo(shell)?; + spinner.finish(); + link_to_code + } else { + let path = PathBuf::from_str(&args.link_to_code)?; + update_submodules_recursive(shell, &path)?; + path + }; + + let spinner = Spinner::new("Creating initial configurations..."); + let chain_config = args.chain_config(); + let chains_path = shell.create_dir("chains")?; + let default_chain_name = args.chain_args.chain_name.clone(); + + create_initial_deployments_config(shell, &configs_path)?; + create_erc20_deployment_config(shell, &configs_path)?; + + let ecosystem_config = EcosystemConfig { + name: ecosystem_name.clone(), + l1_network: args.l1_network, + link_to_code: link_to_code.clone(), + chains: chains_path.clone(), + config: configs_path, + default_chain: default_chain_name.clone(), + era_chain_id: ERA_CHAIN_ID, + prover_version: chain_config.prover_version, + wallet_creation: args.wallet_creation, + shell: shell.clone().into(), + }; + + // Use 0 id for ecosystem wallets + create_wallets( + shell, + &ecosystem_config.config.join(WALLETS_FILE), + &ecosystem_config.link_to_code, + 0, + args.wallet_creation, + args.wallet_path, + )?; + ecosystem_config.save(shell, CONFIG_NAME)?; + spinner.finish(); + + let spinner = Spinner::new("Creating default chain..."); + create_chain_inner(chain_config, &ecosystem_config, shell)?; + spinner.finish(); + + if args.start_containers { + let spinner = Spinner::new("Starting containers..."); + initialize_docker(shell, &ecosystem_config)?; + start_containers(shell)?; + spinner.finish(); + } + + logger::outro("Ecosystem created successfully"); + Ok(()) +} + +fn clone_era_repo(shell: &Shell) -> anyhow::Result { + Cmd::new(cmd!( + shell, + "git clone --recurse-submodules {ZKSYNC_ERA_GIT_REPO}" + )) + .run()?; + Ok(shell.current_dir().join("zksync-era")) +} + +fn update_submodules_recursive(shell: &Shell, link_to_code: &Path) -> anyhow::Result<()> { + let _dir_guard = shell.push_dir(link_to_code); + Cmd::new(cmd!( + shell, + "git submodule update --init --recursive +" + )) + .run()?; + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/create_configs.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/create_configs.rs new file mode 100644 index 000000000000..e99da136b916 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/create_configs.rs @@ -0,0 +1,35 @@ +use std::path::Path; + +use xshell::Shell; + +use crate::{ + configs::{ + forge_interface::deploy_ecosystem::input::{ + Erc20DeploymentConfig, InitialDeploymentConfig, + }, + SaveConfigWithComment, + }, + consts::{ERC20_DEPLOYMENT_FILE, INITIAL_DEPLOYMENT_FILE}, +}; + +pub fn create_initial_deployments_config( + shell: &Shell, + ecosystem_configs_path: &Path, +) -> anyhow::Result { + let config = InitialDeploymentConfig::default(); + config.save_with_comment(shell, ecosystem_configs_path.join(INITIAL_DEPLOYMENT_FILE), "ATTENTION: This file contains sensible placeholders. Please check them and update with the desired values.")?; + Ok(config) +} + +pub fn create_erc20_deployment_config( + shell: &Shell, + ecosystem_configs_path: &Path, +) -> anyhow::Result { + let config = Erc20DeploymentConfig::default(); + config.save_with_comment( + shell, + ecosystem_configs_path.join(ERC20_DEPLOYMENT_FILE), + "ATTENTION: This file should be filled with the desired ERC20 tokens to deploy.", + )?; + Ok(config) +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/init.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/init.rs new file mode 100644 index 000000000000..451acfbf0968 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/init.rs @@ -0,0 +1,350 @@ +use std::{ + path::{Path, PathBuf}, + str::FromStr, +}; + +use anyhow::Context; +use common::{ + cmd::Cmd, + config::global_config, + forge::{Forge, ForgeScriptArgs}, + logger, + spinner::Spinner, + Prompt, +}; +use xshell::{cmd, Shell}; + +use super::args::init::{EcosystemArgsFinal, EcosystemInitArgs, EcosystemInitArgsFinal}; +use crate::forge_utils::check_the_balance; +use crate::{ + accept_ownership::accept_owner, + commands::{ + chain, + ecosystem::create_configs::{ + create_erc20_deployment_config, create_initial_deployments_config, + }, + }, + configs::{ + forge_interface::deploy_ecosystem::{ + input::{ + DeployErc20Config, DeployL1Config, Erc20DeploymentConfig, InitialDeploymentConfig, + }, + output::{DeployErc20Output, DeployL1Output}, + }, + ChainConfig, ContractsConfig, EcosystemConfig, GenesisConfig, ReadConfig, SaveConfig, + }, + consts::{ + AMOUNT_FOR_DISTRIBUTION_TO_WALLETS, CONFIGS_PATH, CONTRACTS_FILE, DEPLOY_ECOSYSTEM, + DEPLOY_ERC20, ECOSYSTEM_PATH, ERC20_CONFIGS_FILE, GENESIS_FILE, + }, + forge_utils::fill_forge_private_key, + types::{L1Network, ProverMode}, + wallets::WalletCreation, +}; + +pub async fn run(args: EcosystemInitArgs, shell: &Shell) -> anyhow::Result<()> { + let ecosystem_config = EcosystemConfig::from_file(shell)?; + + let initial_deployment_config = match ecosystem_config.get_initial_deployment_config() { + Ok(config) => config, + Err(_) => create_initial_deployments_config(shell, &ecosystem_config.config)?, + }; + + let genesis_args = args.genesis_args.clone(); + let mut final_ecosystem_args = args.fill_values_with_prompt(ecosystem_config.l1_network); + + logger::info("Initializing ecosystem"); + + let contracts_config = init( + &mut final_ecosystem_args, + shell, + &ecosystem_config, + &initial_deployment_config, + ) + .await?; + + if final_ecosystem_args.deploy_erc20 { + logger::info("Deploying ERC20 contracts"); + let erc20_deployment_config = match ecosystem_config.get_erc20_deployment_config() { + Ok(config) => config, + Err(_) => create_erc20_deployment_config(shell, &ecosystem_config.config)?, + }; + deploy_erc20( + shell, + &erc20_deployment_config, + &ecosystem_config, + &contracts_config, + final_ecosystem_args.forge_args.clone(), + final_ecosystem_args.ecosystem.l1_rpc_url.clone(), + ) + .await?; + } + + // If the name of chain passed then we deploy exactly this chain otherwise deploy all chains + let list_of_chains = if let Some(name) = global_config().chain_name.clone() { + vec![name] + } else { + ecosystem_config.list_of_chains() + }; + + for chain_name in &list_of_chains { + logger::info(format!("Initializing chain {chain_name}")); + let chain_config = ecosystem_config + .load_chain(Some(chain_name.clone())) + .context("Chain not initialized. Please create a chain first")?; + + let mut chain_init_args = chain::args::init::InitArgsFinal { + forge_args: final_ecosystem_args.forge_args.clone(), + genesis_args: genesis_args.clone().fill_values_with_prompt(&chain_config), + deploy_paymaster: final_ecosystem_args.deploy_paymaster, + l1_rpc_url: final_ecosystem_args.ecosystem.l1_rpc_url.clone(), + }; + + distribute_eth( + &ecosystem_config, + &chain_config, + final_ecosystem_args.ecosystem.l1_rpc_url.clone(), + ) + .await?; + + chain::init::init( + &mut chain_init_args, + shell, + &ecosystem_config, + &chain_config, + ) + .await?; + } + + logger::outro(format!( + "Ecosystem initialized successfully with chains {}", + list_of_chains.join(",") + )); + + Ok(()) +} + +// Distribute eth to the chain wallets for localhost environment +pub async fn distribute_eth( + ecosystem_config: &EcosystemConfig, + chain_config: &ChainConfig, + l1_rpc_url: String, +) -> anyhow::Result<()> { + if chain_config.wallet_creation == WalletCreation::Localhost + && ecosystem_config.l1_network == L1Network::Localhost + { + let spinner = Spinner::new("Distributing eth..."); + let wallets = ecosystem_config.get_wallets()?; + let chain_wallets = chain_config.get_wallets_config()?; + let mut addresses = vec![ + chain_wallets.operator.address, + chain_wallets.blob_operator.address, + chain_wallets.governor.address, + ]; + if let Some(deployer) = chain_wallets.deployer { + addresses.push(deployer.address) + } + common::ethereum::distribute_eth( + wallets.operator, + addresses, + l1_rpc_url, + ecosystem_config.l1_network.chain_id(), + AMOUNT_FOR_DISTRIBUTION_TO_WALLETS, + ) + .await?; + spinner.finish(); + } + Ok(()) +} + +async fn init( + init_args: &mut EcosystemInitArgsFinal, + shell: &Shell, + ecosystem_config: &EcosystemConfig, + initial_deployment_config: &InitialDeploymentConfig, +) -> anyhow::Result { + let spinner = Spinner::new("Installing and building dependencies..."); + install_yarn_dependencies(shell, &ecosystem_config.link_to_code)?; + build_system_contracts(shell, &ecosystem_config.link_to_code)?; + spinner.finish(); + + let contracts = deploy_ecosystem( + shell, + &mut init_args.ecosystem, + init_args.forge_args.clone(), + ecosystem_config, + initial_deployment_config, + ) + .await?; + contracts.save(shell, ecosystem_config.config.clone().join(CONTRACTS_FILE))?; + Ok(contracts) +} + +async fn deploy_erc20( + shell: &Shell, + erc20_deployment_config: &Erc20DeploymentConfig, + ecosystem_config: &EcosystemConfig, + contracts_config: &ContractsConfig, + forge_args: ForgeScriptArgs, + l1_rpc_url: String, +) -> anyhow::Result { + let deploy_config_path = DEPLOY_ERC20.input(&ecosystem_config.link_to_code); + DeployErc20Config::new(erc20_deployment_config, contracts_config) + .save(shell, deploy_config_path)?; + + let mut forge = Forge::new(&ecosystem_config.path_to_foundry()) + .script(&DEPLOY_ERC20.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(l1_rpc_url) + .with_broadcast(); + + forge = fill_forge_private_key( + forge, + ecosystem_config.get_wallets()?.deployer_private_key(), + )?; + + let spinner = Spinner::new("Deploying ERC20 contracts..."); + check_the_balance(&forge).await?; + forge.run(shell)?; + spinner.finish(); + + let result = + DeployErc20Output::read(shell, DEPLOY_ERC20.output(&ecosystem_config.link_to_code))?; + result.save(shell, ecosystem_config.config.join(ERC20_CONFIGS_FILE))?; + Ok(result) +} + +async fn deploy_ecosystem( + shell: &Shell, + ecosystem: &mut EcosystemArgsFinal, + forge_args: ForgeScriptArgs, + ecosystem_config: &EcosystemConfig, + initial_deployment_config: &InitialDeploymentConfig, +) -> anyhow::Result { + if ecosystem.deploy_ecosystem { + return deploy_ecosystem_inner( + shell, + forge_args, + ecosystem_config, + initial_deployment_config, + ecosystem.l1_rpc_url.clone(), + ) + .await; + } + + let ecosystem_contracts_path = match &ecosystem.ecosystem_contracts_path { + Some(path) => Some(path.clone()), + None => { + let input_path: String = Prompt::new("Provide the path to the ecosystem contracts or keep it empty and you will be added to ZkSync ecosystem") + .allow_empty() + .validate_with(|val: &String| { + if val.is_empty() { + return Ok(()); + } + PathBuf::from_str(val).map(|_| ()).map_err(|_| "Invalid path".to_string()) + }) + .ask(); + if input_path.is_empty() { + None + } else { + Some(input_path.into()) + } + } + }; + + let ecosystem_contracts_path = + ecosystem_contracts_path.unwrap_or_else(|| match ecosystem_config.l1_network { + L1Network::Localhost => ecosystem_config.config.join(CONTRACTS_FILE), + L1Network::Sepolia => ecosystem_config + .link_to_code + .join(ECOSYSTEM_PATH) + .join(ecosystem_config.l1_network.to_string().to_lowercase()), + L1Network::Mainnet => ecosystem_config + .link_to_code + .join(ECOSYSTEM_PATH) + .join(ecosystem_config.l1_network.to_string().to_lowercase()), + }); + + ContractsConfig::read(shell, ecosystem_contracts_path) +} + +async fn deploy_ecosystem_inner( + shell: &Shell, + forge_args: ForgeScriptArgs, + config: &EcosystemConfig, + initial_deployment_config: &InitialDeploymentConfig, + l1_rpc_url: String, +) -> anyhow::Result { + let deploy_config_path = DEPLOY_ECOSYSTEM.input(&config.link_to_code); + + let default_genesis_config = GenesisConfig::read( + shell, + config.link_to_code.join(CONFIGS_PATH).join(GENESIS_FILE), + ) + .context("Context")?; + + let wallets_config = config.get_wallets()?; + // For deploying ecosystem we only need genesis batch params + let deploy_config = DeployL1Config::new( + &default_genesis_config, + &wallets_config, + initial_deployment_config, + config.era_chain_id, + config.prover_version == ProverMode::NoProofs, + ); + deploy_config.save(shell, deploy_config_path)?; + + let mut forge = Forge::new(&config.path_to_foundry()) + .script(&DEPLOY_ECOSYSTEM.script(), forge_args.clone()) + .with_ffi() + .with_rpc_url(l1_rpc_url.clone()) + .with_broadcast(); + + if config.l1_network == L1Network::Localhost { + // It's a kludge for reth, just because it doesn't behave properly with large amount of txs + forge = forge.with_slow(); + } + + forge = fill_forge_private_key(forge, wallets_config.deployer_private_key())?; + + let spinner = Spinner::new("Deploying ecosystem contracts..."); + check_the_balance(&forge).await?; + forge.run(shell)?; + spinner.finish(); + + let script_output = DeployL1Output::read(shell, DEPLOY_ECOSYSTEM.output(&config.link_to_code))?; + let mut contracts_config = ContractsConfig::default(); + contracts_config.update_from_l1_output(&script_output); + accept_owner( + shell, + config, + contracts_config.l1.governance_addr, + config.get_wallets()?.governor_private_key(), + contracts_config.ecosystem_contracts.bridgehub_proxy_addr, + &forge_args, + l1_rpc_url.clone(), + ) + .await?; + + accept_owner( + shell, + config, + contracts_config.l1.governance_addr, + config.get_wallets()?.governor_private_key(), + contracts_config.bridges.shared.l1_address, + &forge_args, + l1_rpc_url.clone(), + ) + .await?; + Ok(contracts_config) +} + +fn install_yarn_dependencies(shell: &Shell, link_to_code: &Path) -> anyhow::Result<()> { + let _dir_guard = shell.push_dir(link_to_code); + Cmd::new(cmd!(shell, "yarn install")).run() +} + +fn build_system_contracts(shell: &Shell, link_to_code: &Path) -> anyhow::Result<()> { + let _dir_guard = shell.push_dir(link_to_code.join("contracts")); + Cmd::new(cmd!(shell, "yarn sc build")).run() +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/ecosystem/mod.rs b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/mod.rs new file mode 100644 index 000000000000..1e232b5cf6c6 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/ecosystem/mod.rs @@ -0,0 +1,32 @@ +use clap::Subcommand; +use xshell::Shell; + +use crate::commands::ecosystem::args::{ + change_default::ChangeDefaultChain, create::EcosystemCreateArgs, init::EcosystemInitArgs, +}; + +mod args; +mod change_default; +mod create; +pub mod create_configs; +mod init; + +#[derive(Subcommand, Debug)] +pub enum EcosystemCommands { + /// Create a new ecosystem and chain, + /// setting necessary configurations for later initialization + Create(EcosystemCreateArgs), + /// Initialize ecosystem and chain, + /// deploying necessary contracts and performing on-chain operations + Init(EcosystemInitArgs), + /// Change the default chain + ChangeDefaultChain(ChangeDefaultChain), +} + +pub(crate) async fn run(shell: &Shell, args: EcosystemCommands) -> anyhow::Result<()> { + match args { + EcosystemCommands::Create(args) => create::run(args, shell), + EcosystemCommands::Init(args) => init::run(args, shell).await, + EcosystemCommands::ChangeDefaultChain(args) => change_default::run(args, shell), + } +} diff --git a/zk_toolbox/crates/zk_inception/src/commands/mod.rs b/zk_toolbox/crates/zk_inception/src/commands/mod.rs new file mode 100644 index 000000000000..8ed7a82b8334 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/mod.rs @@ -0,0 +1,5 @@ +pub mod args; +pub mod chain; +pub mod containers; +pub mod ecosystem; +pub mod server; diff --git a/zk_toolbox/crates/zk_inception/src/commands/server.rs b/zk_toolbox/crates/zk_inception/src/commands/server.rs new file mode 100644 index 000000000000..a46b42c17050 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/commands/server.rs @@ -0,0 +1,37 @@ +use anyhow::Context; +use common::{config::global_config, logger}; +use xshell::Shell; + +use crate::{ + commands::args::RunServerArgs, + configs::{ChainConfig, EcosystemConfig}, + server::{RunServer, ServerMode}, +}; + +pub fn run(shell: &Shell, args: RunServerArgs) -> anyhow::Result<()> { + let ecosystem_config = EcosystemConfig::from_file(shell)?; + + let chain = global_config().chain_name.clone(); + let chain_config = ecosystem_config + .load_chain(chain) + .context("Chain not initialized. Please create a chain first")?; + + logger::info("Starting server"); + run_server(args, &chain_config, shell)?; + + Ok(()) +} + +fn run_server( + args: RunServerArgs, + chain_config: &ChainConfig, + shell: &Shell, +) -> anyhow::Result<()> { + let server = RunServer::new(args.components.clone(), chain_config); + let mode = if args.genesis { + ServerMode::Genesis + } else { + ServerMode::Normal + }; + server.run(shell, mode) +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/chain.rs b/zk_toolbox/crates/zk_inception/src/configs/chain.rs new file mode 100644 index 000000000000..08ecc583801f --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/chain.rs @@ -0,0 +1,114 @@ +use std::{ + cell::OnceCell, + path::{Path, PathBuf}, +}; + +use serde::{Deserialize, Serialize, Serializer}; +use xshell::Shell; + +use crate::{ + configs::{ContractsConfig, GenesisConfig, ReadConfig, SaveConfig, Secrets, WalletsConfig}, + consts::{CONTRACTS_FILE, GENESIS_FILE, L1_CONTRACTS_FOUNDRY, SECRETS_FILE, WALLETS_FILE}, + types::{BaseToken, ChainId, L1BatchCommitDataGeneratorMode, L1Network, ProverMode}, + wallets::{create_localhost_wallets, WalletCreation}, +}; + +/// Chain configuration file. This file is created in the chain +/// directory before network initialization. +#[derive(Debug, Serialize, Deserialize, Clone)] +pub struct ChainConfigInternal { + // The id of chain on this machine allows to easily setup multiple chains, + // needs for local setups only + pub id: u32, + pub name: String, + pub chain_id: ChainId, + pub prover_version: ProverMode, + pub configs: PathBuf, + pub rocks_db_path: PathBuf, + pub l1_batch_commit_data_generator_mode: L1BatchCommitDataGeneratorMode, + pub base_token: BaseToken, + pub wallet_creation: WalletCreation, +} + +/// Chain configuration file. This file is created in the chain +/// directory before network initialization. +#[derive(Debug)] +pub struct ChainConfig { + pub id: u32, + pub name: String, + pub chain_id: ChainId, + pub prover_version: ProverMode, + pub l1_network: L1Network, + pub link_to_code: PathBuf, + pub rocks_db_path: PathBuf, + pub configs: PathBuf, + pub l1_batch_commit_data_generator_mode: L1BatchCommitDataGeneratorMode, + pub base_token: BaseToken, + pub wallet_creation: WalletCreation, + pub shell: OnceCell, +} + +impl Serialize for ChainConfig { + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + self.get_internal().serialize(serializer) + } +} + +impl ChainConfig { + pub(crate) fn get_shell(&self) -> &Shell { + self.shell.get().expect("Not initialized") + } + + pub fn get_genesis_config(&self) -> anyhow::Result { + GenesisConfig::read(self.get_shell(), self.configs.join(GENESIS_FILE)) + } + + pub fn get_wallets_config(&self) -> anyhow::Result { + let path = self.configs.join(WALLETS_FILE); + if let Ok(wallets) = WalletsConfig::read(self.get_shell(), &path) { + return Ok(wallets); + } + if self.wallet_creation == WalletCreation::Localhost { + let wallets = create_localhost_wallets(self.get_shell(), &self.link_to_code, self.id)?; + wallets.save(self.get_shell(), &path)?; + return Ok(wallets); + } + anyhow::bail!("Wallets configs has not been found"); + } + pub fn get_contracts_config(&self) -> anyhow::Result { + ContractsConfig::read(self.get_shell(), self.configs.join(CONTRACTS_FILE)) + } + + pub fn get_secrets_config(&self) -> anyhow::Result { + Secrets::read(self.get_shell(), self.configs.join(SECRETS_FILE)) + } + + pub fn path_to_foundry(&self) -> PathBuf { + self.link_to_code.join(L1_CONTRACTS_FOUNDRY) + } + + pub fn save(&self, shell: &Shell, path: impl AsRef) -> anyhow::Result<()> { + let config = self.get_internal(); + config.save(shell, path) + } + + fn get_internal(&self) -> ChainConfigInternal { + ChainConfigInternal { + id: self.id, + name: self.name.clone(), + chain_id: self.chain_id, + prover_version: self.prover_version, + configs: self.configs.clone(), + rocks_db_path: self.rocks_db_path.clone(), + l1_batch_commit_data_generator_mode: self.l1_batch_commit_data_generator_mode, + base_token: self.base_token.clone(), + wallet_creation: self.wallet_creation, + } + } +} + +impl ReadConfig for ChainConfigInternal {} +impl SaveConfig for ChainConfigInternal {} diff --git a/zk_toolbox/crates/zk_inception/src/configs/contracts.rs b/zk_toolbox/crates/zk_inception/src/configs/contracts.rs new file mode 100644 index 000000000000..c5302ae21298 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/contracts.rs @@ -0,0 +1,109 @@ +use ethers::{addressbook::Address, types::H256}; +use serde::{Deserialize, Serialize}; + +use crate::configs::{ + forge_interface::deploy_ecosystem::output::DeployL1Output, ReadConfig, SaveConfig, +}; + +#[derive(Debug, Deserialize, Serialize, Clone, Default)] +pub struct ContractsConfig { + pub create2_factory_addr: Address, + pub create2_factory_salt: H256, + pub ecosystem_contracts: EcosystemContracts, + pub bridges: BridgesContracts, + pub l1: L1Contracts, + pub l2: L2Contracts, + #[serde(flatten)] + pub other: serde_json::Value, +} + +impl ContractsConfig { + pub fn update_from_l1_output(&mut self, deploy_l1_output: &DeployL1Output) { + self.create2_factory_addr = deploy_l1_output.create2_factory_addr; + self.create2_factory_salt = deploy_l1_output.create2_factory_salt; + self.bridges.erc20.l1_address = deploy_l1_output + .deployed_addresses + .bridges + .erc20_bridge_proxy_addr; + self.bridges.shared.l1_address = deploy_l1_output + .deployed_addresses + .bridges + .shared_bridge_proxy_addr; + self.ecosystem_contracts.bridgehub_proxy_addr = deploy_l1_output + .deployed_addresses + .bridgehub + .bridgehub_proxy_addr; + self.ecosystem_contracts.state_transition_proxy_addr = deploy_l1_output + .deployed_addresses + .state_transition + .state_transition_proxy_addr; + self.ecosystem_contracts.transparent_proxy_admin_addr = deploy_l1_output + .deployed_addresses + .transparent_proxy_admin_addr; + self.l1.default_upgrade_addr = deploy_l1_output + .deployed_addresses + .state_transition + .default_upgrade_addr; + self.l1.diamond_proxy_addr = deploy_l1_output + .deployed_addresses + .state_transition + .diamond_proxy_addr; + self.l1.governance_addr = deploy_l1_output.deployed_addresses.governance_addr; + self.l1.multicall3_addr = deploy_l1_output.multicall3_addr; + self.ecosystem_contracts.validator_timelock_addr = + deploy_l1_output.deployed_addresses.validator_timelock_addr; + self.l1.verifier_addr = deploy_l1_output + .deployed_addresses + .state_transition + .verifier_addr; + self.l1.validator_timelock_addr = + deploy_l1_output.deployed_addresses.validator_timelock_addr; + self.ecosystem_contracts + .diamond_cut_data + .clone_from(&deploy_l1_output.contracts_config.diamond_cut_data); + } +} + +impl ReadConfig for ContractsConfig {} +impl SaveConfig for ContractsConfig {} + +#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Default)] +pub struct EcosystemContracts { + pub bridgehub_proxy_addr: Address, + pub state_transition_proxy_addr: Address, + pub transparent_proxy_admin_addr: Address, + pub validator_timelock_addr: Address, + pub diamond_cut_data: String, +} + +impl ReadConfig for EcosystemContracts {} +impl SaveConfig for EcosystemContracts {} + +#[derive(Debug, Serialize, Deserialize, Clone, Default)] +pub struct BridgesContracts { + pub erc20: BridgeContractsDefinition, + pub shared: BridgeContractsDefinition, +} + +#[derive(Debug, Serialize, Deserialize, Clone, Default)] +pub struct BridgeContractsDefinition { + pub l1_address: Address, + #[serde(skip_serializing_if = "Option::is_none")] + pub l2_address: Option
, +} + +#[derive(Debug, Serialize, Deserialize, Clone, Default)] +pub struct L1Contracts { + pub default_upgrade_addr: Address, + pub diamond_proxy_addr: Address, + pub governance_addr: Address, + pub multicall3_addr: Address, + pub verifier_addr: Address, + pub validator_timelock_addr: Address, + pub base_token_addr: Address, +} + +#[derive(Debug, Serialize, Deserialize, Clone, Default)] +pub struct L2Contracts { + pub testnet_paymaster_addr: Address, +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/ecosystem.rs b/zk_toolbox/crates/zk_inception/src/configs/ecosystem.rs new file mode 100644 index 000000000000..66e90f22f99b --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/ecosystem.rs @@ -0,0 +1,196 @@ +use std::{cell::OnceCell, path::PathBuf}; + +use serde::{Deserialize, Deserializer, Serialize, Serializer}; +use thiserror::Error; +use xshell::Shell; + +use crate::{ + configs::{ + forge_interface::deploy_ecosystem::input::{ + Erc20DeploymentConfig, InitialDeploymentConfig, + }, + ChainConfig, ChainConfigInternal, ContractsConfig, ReadConfig, SaveConfig, WalletsConfig, + }, + consts::{ + CONFIG_NAME, CONTRACTS_FILE, ERC20_DEPLOYMENT_FILE, INITIAL_DEPLOYMENT_FILE, + L1_CONTRACTS_FOUNDRY, WALLETS_FILE, + }, + types::{ChainId, L1Network, ProverMode}, + wallets::{create_localhost_wallets, WalletCreation}, +}; + +/// Ecosystem configuration file. This file is created in the chain +/// directory before network initialization. +#[derive(Debug, Clone, Serialize, Deserialize)] +struct EcosystemConfigInternal { + pub name: String, + pub l1_network: L1Network, + pub link_to_code: PathBuf, + pub chains: PathBuf, + pub config: PathBuf, + pub default_chain: String, + pub era_chain_id: ChainId, + pub prover_version: ProverMode, + pub wallet_creation: WalletCreation, +} + +/// Ecosystem configuration file. This file is created in the chain +/// directory before network initialization. +#[derive(Debug, Clone)] +pub struct EcosystemConfig { + pub name: String, + pub l1_network: L1Network, + pub link_to_code: PathBuf, + pub chains: PathBuf, + pub config: PathBuf, + pub default_chain: String, + pub era_chain_id: ChainId, + pub prover_version: ProverMode, + pub wallet_creation: WalletCreation, + pub shell: OnceCell, +} + +impl Serialize for EcosystemConfig { + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + self.get_internal().serialize(serializer) + } +} + +impl<'de> Deserialize<'de> for EcosystemConfig { + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + let config: EcosystemConfigInternal = Deserialize::deserialize(deserializer)?; + Ok(EcosystemConfig { + name: config.name.clone(), + l1_network: config.l1_network, + link_to_code: config.link_to_code.clone(), + chains: config.chains.clone(), + config: config.config.clone(), + default_chain: config.default_chain.clone(), + era_chain_id: config.era_chain_id, + prover_version: config.prover_version, + wallet_creation: config.wallet_creation, + shell: Default::default(), + }) + } +} + +impl ReadConfig for EcosystemConfig {} +impl SaveConfig for EcosystemConfig {} + +impl EcosystemConfig { + fn get_shell(&self) -> &Shell { + self.shell.get().expect("Must be initialized") + } + + pub fn from_file(shell: &Shell) -> Result { + let path = PathBuf::from(CONFIG_NAME); + if !shell.path_exists(path) { + return Err(EcosystemConfigFromFileError::NotExists); + } + + let mut config = EcosystemConfig::read(shell, CONFIG_NAME) + .map_err(|e| EcosystemConfigFromFileError::InvalidConfig { source: e })?; + config.shell = shell.clone().into(); + + Ok(config) + } + + pub fn load_chain(&self, name: Option) -> Option { + let name = name.unwrap_or(self.default_chain.clone()); + self.load_chain_inner(&name) + } + + fn load_chain_inner(&self, name: &str) -> Option { + let path = self.chains.join(name).join(CONFIG_NAME); + let config = ChainConfigInternal::read(self.get_shell(), path).ok()?; + + Some(ChainConfig { + id: config.id, + name: config.name, + chain_id: config.chain_id, + prover_version: config.prover_version, + configs: config.configs, + l1_batch_commit_data_generator_mode: config.l1_batch_commit_data_generator_mode, + l1_network: self.l1_network, + link_to_code: self.link_to_code.clone(), + base_token: config.base_token, + rocks_db_path: config.rocks_db_path, + wallet_creation: config.wallet_creation, + shell: self.get_shell().clone().into(), + }) + } + + pub fn get_initial_deployment_config(&self) -> anyhow::Result { + InitialDeploymentConfig::read(self.get_shell(), self.config.join(INITIAL_DEPLOYMENT_FILE)) + } + + pub fn get_erc20_deployment_config(&self) -> anyhow::Result { + Erc20DeploymentConfig::read(self.get_shell(), self.config.join(ERC20_DEPLOYMENT_FILE)) + } + + pub fn get_wallets(&self) -> anyhow::Result { + let path = self.config.join(WALLETS_FILE); + if let Ok(wallets) = WalletsConfig::read(self.get_shell(), &path) { + return Ok(wallets); + } + if self.wallet_creation == WalletCreation::Localhost { + // Use 0 id for ecosystem wallets + let wallets = create_localhost_wallets(self.get_shell(), &self.link_to_code, 0)?; + wallets.save(self.get_shell(), &path)?; + return Ok(wallets); + } + anyhow::bail!("Wallets configs has not been found"); + } + + pub fn get_contracts_config(&self) -> anyhow::Result { + ContractsConfig::read(self.get_shell(), self.config.join(CONTRACTS_FILE)) + } + + pub fn path_to_foundry(&self) -> PathBuf { + self.link_to_code.join(L1_CONTRACTS_FOUNDRY) + } + + pub fn list_of_chains(&self) -> Vec { + self.get_shell() + .read_dir(&self.chains) + .unwrap() + .iter() + .filter_map(|file| { + if file.is_dir() { + file.file_name().map(|a| a.to_str().unwrap().to_string()) + } else { + None + } + }) + .collect() + } + + fn get_internal(&self) -> EcosystemConfigInternal { + EcosystemConfigInternal { + name: self.name.clone(), + l1_network: self.l1_network, + link_to_code: self.link_to_code.clone(), + chains: self.chains.clone(), + config: self.config.clone(), + default_chain: self.default_chain.clone(), + era_chain_id: self.era_chain_id, + prover_version: self.prover_version, + wallet_creation: self.wallet_creation, + } + } +} + +/// Result of checking if the ecosystem exists. +#[derive(Error, Debug)] +pub enum EcosystemConfigFromFileError { + #[error("Ecosystem configuration not found")] + NotExists, + #[error("Invalid ecosystem configuration")] + InvalidConfig { source: anyhow::Error }, +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/accept_ownership/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/accept_ownership/mod.rs new file mode 100644 index 000000000000..cd56d6ae0fb8 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/accept_ownership/mod.rs @@ -0,0 +1,13 @@ +use ethers::prelude::Address; +use serde::{Deserialize, Serialize}; + +use crate::configs::{ReadConfig, SaveConfig}; + +impl ReadConfig for AcceptOwnershipInput {} +impl SaveConfig for AcceptOwnershipInput {} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct AcceptOwnershipInput { + pub target_addr: Address, + pub governor: Address, +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/input.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/input.rs new file mode 100644 index 000000000000..12b7b1633f14 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/input.rs @@ -0,0 +1,245 @@ +use std::{collections::HashMap, str::FromStr}; + +use ethers::{ + addressbook::Address, + core::{rand, rand::Rng}, + prelude::H256, +}; +use serde::{Deserialize, Serialize}; + +use crate::{ + configs::{ + ContractsConfig, GenesisConfig, ReadConfig, SaveConfig, SaveConfigWithComment, + WalletsConfig, + }, + types::ChainId, +}; + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct InitialDeploymentConfig { + #[serde(skip_serializing_if = "Option::is_none")] + pub create2_factory_addr: Option
, + pub create2_factory_salt: H256, + pub governance_min_delay: u64, + pub max_number_of_chains: u64, + pub diamond_init_batch_overhead_l1_gas: u64, + pub diamond_init_max_l2_gas_per_batch: u64, + pub diamond_init_max_pubdata_per_batch: u64, + pub diamond_init_minimal_l2_gas_price: u64, + pub diamond_init_priority_tx_max_pubdata: u64, + pub diamond_init_pubdata_pricing_mode: u64, + pub priority_tx_max_gas_limit: u64, + pub validator_timelock_execution_delay: u64, + pub token_weth_address: Address, + pub bridgehub_create_new_chain_salt: u64, +} + +impl Default for InitialDeploymentConfig { + fn default() -> Self { + Self { + create2_factory_addr: None, + create2_factory_salt: H256::random(), + governance_min_delay: 0, + max_number_of_chains: 100, + diamond_init_batch_overhead_l1_gas: 1000000, + diamond_init_max_l2_gas_per_batch: 80000000, + diamond_init_max_pubdata_per_batch: 120000, + diamond_init_minimal_l2_gas_price: 250000000, + diamond_init_priority_tx_max_pubdata: 99000, + diamond_init_pubdata_pricing_mode: 0, + priority_tx_max_gas_limit: 72000000, + validator_timelock_execution_delay: 0, + token_weth_address: Address::from_str("0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2") + .unwrap(), + // toml crate u64 support is backed by i64 implementation + // https://github.com/toml-rs/toml/issues/705 + bridgehub_create_new_chain_salt: rand::thread_rng().gen_range(0..=i64::MAX) as u64, + } + } +} + +impl ReadConfig for InitialDeploymentConfig {} +impl SaveConfig for InitialDeploymentConfig {} +impl SaveConfigWithComment for InitialDeploymentConfig {} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct Erc20DeploymentConfig { + pub tokens: Vec, +} + +impl ReadConfig for Erc20DeploymentConfig {} +impl SaveConfig for Erc20DeploymentConfig {} +impl SaveConfigWithComment for Erc20DeploymentConfig {} + +impl Default for Erc20DeploymentConfig { + fn default() -> Self { + Self { + tokens: vec![ + Erc20DeploymentTokensConfig { + name: String::from("DAI"), + symbol: String::from("DAI"), + decimals: 18, + implementation: String::from("TestnetERC20Token.sol"), + mint: 10000000000, + }, + Erc20DeploymentTokensConfig { + name: String::from("Wrapped Ether"), + symbol: String::from("WETH"), + decimals: 18, + implementation: String::from("WETH9.sol"), + mint: 0, + }, + ], + } + } +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct Erc20DeploymentTokensConfig { + pub name: String, + pub symbol: String, + pub decimals: u64, + pub implementation: String, + pub mint: u64, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct DeployL1Config { + pub era_chain_id: ChainId, + pub owner_address: Address, + pub testnet_verifier: bool, + pub contracts: ContractsDeployL1Config, + pub tokens: TokensDeployL1Config, +} + +impl ReadConfig for DeployL1Config {} +impl SaveConfig for DeployL1Config {} + +impl DeployL1Config { + pub fn new( + genesis_config: &GenesisConfig, + wallets_config: &WalletsConfig, + initial_deployment_config: &InitialDeploymentConfig, + era_chain_id: ChainId, + testnet_verifier: bool, + ) -> Self { + Self { + era_chain_id, + testnet_verifier, + owner_address: wallets_config.governor.address, + contracts: ContractsDeployL1Config { + create2_factory_addr: initial_deployment_config.create2_factory_addr, + create2_factory_salt: initial_deployment_config.create2_factory_salt, + // TODO verify correctnesss + governance_security_council_address: wallets_config.governor.address, + governance_min_delay: initial_deployment_config.governance_min_delay, + max_number_of_chains: initial_deployment_config.max_number_of_chains, + diamond_init_batch_overhead_l1_gas: initial_deployment_config + .diamond_init_batch_overhead_l1_gas, + diamond_init_max_l2_gas_per_batch: initial_deployment_config + .diamond_init_max_l2_gas_per_batch, + diamond_init_max_pubdata_per_batch: initial_deployment_config + .diamond_init_max_pubdata_per_batch, + diamond_init_minimal_l2_gas_price: initial_deployment_config + .diamond_init_minimal_l2_gas_price, + bootloader_hash: genesis_config.bootloader_hash, + default_aa_hash: genesis_config.default_aa_hash, + diamond_init_priority_tx_max_pubdata: initial_deployment_config + .diamond_init_priority_tx_max_pubdata, + diamond_init_pubdata_pricing_mode: initial_deployment_config + .diamond_init_pubdata_pricing_mode, + genesis_batch_commitment: genesis_config.genesis_batch_commitment, + genesis_rollup_leaf_index: genesis_config.genesis_rollup_leaf_index, + genesis_root: genesis_config.genesis_root, + latest_protocol_version: genesis_config.genesis_protocol_version, + recursion_circuits_set_vks_hash: H256::zero(), + recursion_leaf_level_vk_hash: H256::zero(), + recursion_node_level_vk_hash: H256::zero(), + priority_tx_max_gas_limit: initial_deployment_config.priority_tx_max_gas_limit, + validator_timelock_execution_delay: initial_deployment_config + .validator_timelock_execution_delay, + }, + tokens: TokensDeployL1Config { + token_weth_address: initial_deployment_config.token_weth_address, + }, + } + } +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct ContractsDeployL1Config { + pub governance_security_council_address: Address, + pub governance_min_delay: u64, + pub max_number_of_chains: u64, + pub create2_factory_salt: H256, + #[serde(skip_serializing_if = "Option::is_none")] + pub create2_factory_addr: Option
, + pub validator_timelock_execution_delay: u64, + pub genesis_root: H256, + pub genesis_rollup_leaf_index: u32, + pub genesis_batch_commitment: H256, + pub latest_protocol_version: u64, + pub recursion_node_level_vk_hash: H256, + pub recursion_leaf_level_vk_hash: H256, + pub recursion_circuits_set_vks_hash: H256, + pub priority_tx_max_gas_limit: u64, + pub diamond_init_pubdata_pricing_mode: u64, + pub diamond_init_batch_overhead_l1_gas: u64, + pub diamond_init_max_pubdata_per_batch: u64, + pub diamond_init_max_l2_gas_per_batch: u64, + pub diamond_init_priority_tx_max_pubdata: u64, + pub diamond_init_minimal_l2_gas_price: u64, + pub bootloader_hash: H256, + pub default_aa_hash: H256, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct TokensDeployL1Config { + pub token_weth_address: Address, +} + +// TODO check for ability to resuse Erc20DeploymentConfig +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct DeployErc20Config { + pub create2_factory_salt: H256, + pub create2_factory_addr: Address, + pub tokens: HashMap, +} + +impl ReadConfig for DeployErc20Config {} +impl SaveConfig for DeployErc20Config {} + +impl DeployErc20Config { + pub fn new( + erc20_deployment_config: &Erc20DeploymentConfig, + contracts_config: &ContractsConfig, + ) -> Self { + let mut tokens = HashMap::new(); + for token in &erc20_deployment_config.tokens { + tokens.insert( + token.symbol.clone(), + TokenDeployErc20Config { + name: token.name.clone(), + symbol: token.symbol.clone(), + decimals: token.decimals, + implementation: token.implementation.clone(), + mint: token.mint, + }, + ); + } + Self { + create2_factory_addr: contracts_config.create2_factory_addr, + create2_factory_salt: contracts_config.create2_factory_salt, + tokens, + } + } +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct TokenDeployErc20Config { + pub name: String, + pub symbol: String, + pub decimals: u64, + pub implementation: String, + pub mint: u64, +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/mod.rs new file mode 100644 index 000000000000..7d1a54844d0c --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/mod.rs @@ -0,0 +1,2 @@ +pub mod input; +pub mod output; diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/output.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/output.rs new file mode 100644 index 000000000000..6b4a117488ef --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/deploy_ecosystem/output.rs @@ -0,0 +1,95 @@ +use std::collections::HashMap; + +use ethers::{addressbook::Address, prelude::H256}; +use serde::{Deserialize, Serialize}; + +use crate::configs::{ReadConfig, SaveConfig}; + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct DeployL1Output { + pub create2_factory_addr: Address, + pub create2_factory_salt: H256, + pub deployer_addr: Address, + pub era_chain_id: u32, + pub l1_chain_id: u32, + pub multicall3_addr: Address, + pub owner_addr: Address, + pub contracts_config: DeployL1ContractsConfigOutput, + pub deployed_addresses: DeployL1DeployedAddressesOutput, +} + +impl ReadConfig for DeployL1Output {} +impl SaveConfig for DeployL1Output {} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct DeployL1ContractsConfigOutput { + pub diamond_init_max_l2_gas_per_batch: u64, + pub diamond_init_batch_overhead_l1_gas: u64, + pub diamond_init_max_pubdata_per_batch: u64, + pub diamond_init_minimal_l2_gas_price: u64, + pub diamond_init_priority_tx_max_pubdata: u64, + pub diamond_init_pubdata_pricing_mode: u64, + pub priority_tx_max_gas_limit: u64, + pub recursion_circuits_set_vks_hash: H256, + pub recursion_leaf_level_vk_hash: H256, + pub recursion_node_level_vk_hash: H256, + pub diamond_cut_data: String, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct DeployL1DeployedAddressesOutput { + pub blob_versioned_hash_retriever_addr: Address, + pub governance_addr: Address, + pub transparent_proxy_admin_addr: Address, + pub validator_timelock_addr: Address, + pub bridgehub: L1BridgehubOutput, + pub bridges: L1BridgesOutput, + pub state_transition: L1StateTransitionOutput, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct L1BridgehubOutput { + pub bridgehub_implementation_addr: Address, + pub bridgehub_proxy_addr: Address, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct L1BridgesOutput { + pub erc20_bridge_implementation_addr: Address, + pub erc20_bridge_proxy_addr: Address, + pub shared_bridge_implementation_addr: Address, + pub shared_bridge_proxy_addr: Address, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct L1StateTransitionOutput { + pub admin_facet_addr: Address, + pub default_upgrade_addr: Address, + pub diamond_init_addr: Address, + pub diamond_proxy_addr: Address, + pub executor_facet_addr: Address, + pub genesis_upgrade_addr: Address, + pub getters_facet_addr: Address, + pub mailbox_facet_addr: Address, + pub state_transition_implementation_addr: Address, + pub state_transition_proxy_addr: Address, + pub verifier_addr: Address, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct TokenDeployErc20Output { + pub address: Address, + pub name: String, + pub symbol: String, + pub decimals: u64, + pub implementation: String, + pub mint: u64, +} + +impl ReadConfig for DeployErc20Output {} +impl SaveConfig for DeployErc20Output {} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct DeployErc20Output { + pub tokens: HashMap, +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/input.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/input.rs new file mode 100644 index 000000000000..2bbe46fd2c92 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/input.rs @@ -0,0 +1,35 @@ +use ethers::addressbook::Address; +use serde::{Deserialize, Serialize}; + +use crate::{ + configs::{ChainConfig, ReadConfig, SaveConfig}, + types::ChainId, +}; + +impl ReadConfig for InitializeBridgeInput {} +impl SaveConfig for InitializeBridgeInput {} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct InitializeBridgeInput { + pub era_chain_id: ChainId, + pub chain_id: ChainId, + pub l1_shared_bridge: Address, + pub bridgehub: Address, + pub governance: Address, + pub erc20_bridge: Address, +} + +impl InitializeBridgeInput { + pub fn new(chain_config: &ChainConfig, era_chain_id: ChainId) -> anyhow::Result { + let contracts = chain_config.get_contracts_config()?; + let wallets = chain_config.get_wallets_config()?; + Ok(Self { + era_chain_id, + chain_id: chain_config.chain_id, + l1_shared_bridge: contracts.bridges.shared.l1_address, + bridgehub: contracts.ecosystem_contracts.bridgehub_proxy_addr, + governance: wallets.governor.address, + erc20_bridge: contracts.bridges.erc20.l1_address, + }) + } +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/mod.rs new file mode 100644 index 000000000000..7d1a54844d0c --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/mod.rs @@ -0,0 +1,2 @@ +pub mod input; +pub mod output; diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/output.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/output.rs new file mode 100644 index 000000000000..bf6cf41dfa7a --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/initialize_bridges/output.rs @@ -0,0 +1,12 @@ +use ethers::addressbook::Address; +use serde::{Deserialize, Serialize}; + +use crate::configs::ReadConfig; + +impl ReadConfig for InitializeBridgeOutput {} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct InitializeBridgeOutput { + pub l2_shared_bridge_implementation: Address, + pub l2_shared_bridge_proxy: Address, +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/mod.rs new file mode 100644 index 000000000000..3e7619560d1d --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/mod.rs @@ -0,0 +1,5 @@ +pub mod accept_ownership; +pub mod deploy_ecosystem; +pub mod initialize_bridges; +pub mod paymaster; +pub mod register_chain; diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/paymaster/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/paymaster/mod.rs new file mode 100644 index 000000000000..a15a007522aa --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/paymaster/mod.rs @@ -0,0 +1,35 @@ +use ethers::addressbook::Address; +use serde::{Deserialize, Serialize}; + +use crate::{ + configs::{ChainConfig, ReadConfig, SaveConfig}, + types::ChainId, +}; + +#[derive(Debug, Serialize, Deserialize, Clone)] +pub struct DeployPaymasterInput { + pub chain_id: ChainId, + pub l1_shared_bridge: Address, + pub bridgehub: Address, +} + +impl DeployPaymasterInput { + pub fn new(chain_config: &ChainConfig) -> anyhow::Result { + let contracts = chain_config.get_contracts_config()?; + Ok(Self { + chain_id: chain_config.chain_id, + l1_shared_bridge: contracts.bridges.shared.l1_address, + bridgehub: contracts.ecosystem_contracts.bridgehub_proxy_addr, + }) + } +} +impl SaveConfig for DeployPaymasterInput {} +impl ReadConfig for DeployPaymasterInput {} + +#[derive(Debug, Serialize, Deserialize, Clone)] +pub struct DeployPaymasterOutput { + pub paymaster: Address, +} + +impl SaveConfig for DeployPaymasterOutput {} +impl ReadConfig for DeployPaymasterOutput {} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/input.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/input.rs new file mode 100644 index 000000000000..bf7e52771680 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/input.rs @@ -0,0 +1,96 @@ +use ethers::{ + addressbook::Address, + core::{rand, rand::Rng}, +}; +use serde::{Deserialize, Serialize}; + +use crate::{ + configs::{ChainConfig, ContractsConfig, ReadConfig, SaveConfig}, + types::{ChainId, L1BatchCommitDataGeneratorMode}, +}; + +#[derive(Debug, Deserialize, Serialize, Clone)] +struct Bridgehub { + bridgehub_proxy_addr: Address, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +struct StateTransition { + state_transition_proxy_addr: Address, +} +#[derive(Debug, Deserialize, Serialize, Clone)] +struct DeployedAddresses { + state_transition: StateTransition, + bridgehub: Bridgehub, + validator_timelock_addr: Address, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +struct Contracts { + diamond_cut_data: String, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct RegisterChainL1Config { + contracts_config: Contracts, + deployed_addresses: DeployedAddresses, + chain: ChainL1Config, + owner_address: Address, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct ChainL1Config { + pub chain_chain_id: ChainId, + pub base_token_addr: Address, + pub bridgehub_create_new_chain_salt: u64, + pub validium_mode: bool, + pub validator_sender_operator_commit_eth: Address, + pub validator_sender_operator_blobs_eth: Address, + pub base_token_gas_price_multiplier_nominator: u64, + pub base_token_gas_price_multiplier_denominator: u64, + pub governance_security_council_address: Address, + pub governance_min_delay: u64, +} + +impl ReadConfig for RegisterChainL1Config {} + +impl SaveConfig for RegisterChainL1Config {} + +impl RegisterChainL1Config { + pub fn new(chain_config: &ChainConfig, contracts: &ContractsConfig) -> anyhow::Result { + let genesis_config = chain_config.get_genesis_config()?; + let wallets_config = chain_config.get_wallets_config()?; + Ok(Self { + contracts_config: Contracts { + diamond_cut_data: contracts.ecosystem_contracts.diamond_cut_data.clone(), + }, + deployed_addresses: DeployedAddresses { + state_transition: StateTransition { + state_transition_proxy_addr: contracts + .ecosystem_contracts + .state_transition_proxy_addr, + }, + bridgehub: Bridgehub { + bridgehub_proxy_addr: contracts.ecosystem_contracts.bridgehub_proxy_addr, + }, + validator_timelock_addr: contracts.ecosystem_contracts.validator_timelock_addr, + }, + chain: ChainL1Config { + chain_chain_id: genesis_config.l2_chain_id, + base_token_gas_price_multiplier_nominator: chain_config.base_token.nominator, + base_token_gas_price_multiplier_denominator: chain_config.base_token.denominator, + base_token_addr: chain_config.base_token.address, + // TODO specify + governance_security_council_address: Default::default(), + governance_min_delay: 0, + // TODO verify + bridgehub_create_new_chain_salt: rand::thread_rng().gen_range(0..=i64::MAX) as u64, + validium_mode: chain_config.l1_batch_commit_data_generator_mode + == L1BatchCommitDataGeneratorMode::Validium, + validator_sender_operator_commit_eth: wallets_config.operator.address, + validator_sender_operator_blobs_eth: wallets_config.blob_operator.address, + }, + owner_address: wallets_config.governor.address, + }) + } +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/mod.rs new file mode 100644 index 000000000000..7d1a54844d0c --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/mod.rs @@ -0,0 +1,2 @@ +pub mod input; +pub mod output; diff --git a/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/output.rs b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/output.rs new file mode 100644 index 000000000000..4e97af0254b0 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/forge_interface/register_chain/output.rs @@ -0,0 +1,13 @@ +use ethers::addressbook::Address; +use serde::{Deserialize, Serialize}; + +use crate::configs::{ReadConfig, SaveConfig}; + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct RegisterChainOutput { + pub diamond_proxy_addr: Address, + pub governance_addr: Address, +} + +impl ReadConfig for RegisterChainOutput {} +impl SaveConfig for RegisterChainOutput {} diff --git a/zk_toolbox/crates/zk_inception/src/configs/general.rs b/zk_toolbox/crates/zk_inception/src/configs/general.rs new file mode 100644 index 000000000000..5acb6762e9c6 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/general.rs @@ -0,0 +1,69 @@ +use std::path::PathBuf; + +use ethers::types::{Address, H256}; +use serde::{Deserialize, Serialize}; + +use crate::{ + configs::{ReadConfig, SaveConfig}, + types::{ChainId, L1BatchCommitDataGeneratorMode}, +}; + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct GenesisConfig { + pub l2_chain_id: ChainId, + pub l1_chain_id: u32, + pub l1_batch_commit_data_generator_mode: Option, + pub bootloader_hash: H256, + pub default_aa_hash: H256, + pub fee_account: Address, + pub genesis_batch_commitment: H256, + pub genesis_rollup_leaf_index: u32, + pub genesis_root: H256, + pub genesis_protocol_version: u64, + #[serde(flatten)] + pub other: serde_json::Value, +} + +impl ReadConfig for GenesisConfig {} +impl SaveConfig for GenesisConfig {} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct EthConfig { + pub sender: EthSender, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct EthSender { + pub proof_sending_mode: String, + pub pubdata_sending_mode: String, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct GeneralConfig { + pub db: RocksDBConfig, + pub eth: EthConfig, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct RocksDBConfig { + pub state_keeper_db_path: PathBuf, + pub merkle_tree: MerkleTreeDB, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Deserialize, Serialize, Clone)] +pub struct MerkleTreeDB { + pub path: PathBuf, + #[serde(flatten)] + pub other: serde_json::Value, +} + +impl ReadConfig for GeneralConfig {} +impl SaveConfig for GeneralConfig {} diff --git a/zk_toolbox/crates/zk_inception/src/configs/manipulations.rs b/zk_toolbox/crates/zk_inception/src/configs/manipulations.rs new file mode 100644 index 000000000000..e8522a0446d9 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/manipulations.rs @@ -0,0 +1,124 @@ +use std::path::Path; + +use xshell::Shell; + +use crate::{ + configs::{ + chain::ChainConfig, + contracts::ContractsConfig, + forge_interface::{ + initialize_bridges::output::InitializeBridgeOutput, paymaster::DeployPaymasterOutput, + register_chain::output::RegisterChainOutput, + }, + DatabasesConfig, GeneralConfig, GenesisConfig, ReadConfig, SaveConfig, Secrets, + }, + consts::{ + CONFIGS_PATH, CONTRACTS_FILE, GENERAL_FILE, GENESIS_FILE, SECRETS_FILE, WALLETS_FILE, + }, + defaults::{ROCKS_DB_STATE_KEEPER, ROCKS_DB_TREE}, + types::ProverMode, +}; + +pub(crate) fn copy_configs( + shell: &Shell, + link_to_code: &Path, + chain_config_path: &Path, +) -> anyhow::Result<()> { + let original_configs = link_to_code.join(CONFIGS_PATH); + for file in shell.read_dir(original_configs)? { + if let Some(name) = file.file_name() { + // Do not copy wallets file + if name != WALLETS_FILE { + shell.copy_file(file, chain_config_path)?; + } + } + } + Ok(()) +} + +pub(crate) fn update_genesis(shell: &Shell, config: &ChainConfig) -> anyhow::Result<()> { + let path = config.configs.join(GENESIS_FILE); + let mut genesis = GenesisConfig::read(shell, &path)?; + + genesis.l2_chain_id = config.chain_id; + genesis.l1_chain_id = config.l1_network.chain_id(); + genesis.l1_batch_commit_data_generator_mode = Some(config.l1_batch_commit_data_generator_mode); + + genesis.save(shell, &path)?; + Ok(()) +} + +pub(crate) fn update_database_secrets( + shell: &Shell, + config: &ChainConfig, + db_config: &DatabasesConfig, +) -> anyhow::Result<()> { + let path = config.configs.join(SECRETS_FILE); + let mut secrets = Secrets::read(shell, &path)?; + secrets.database.server_url = db_config.server.full_url(); + secrets.database.prover_url = db_config.prover.full_url(); + secrets.save(shell, path)?; + Ok(()) +} + +pub(crate) fn update_l1_rpc_url_secret( + shell: &Shell, + config: &ChainConfig, + l1_rpc_url: String, +) -> anyhow::Result<()> { + let path = config.configs.join(SECRETS_FILE); + let mut secrets = Secrets::read(shell, &path)?; + secrets.l1.l1_rpc_url = l1_rpc_url; + secrets.save(shell, path)?; + Ok(()) +} +pub(crate) fn update_general_config(shell: &Shell, config: &ChainConfig) -> anyhow::Result<()> { + let path = config.configs.join(GENERAL_FILE); + let mut general = GeneralConfig::read(shell, &path)?; + general.db.state_keeper_db_path = + shell.create_dir(config.rocks_db_path.join(ROCKS_DB_STATE_KEEPER))?; + general.db.merkle_tree.path = shell.create_dir(config.rocks_db_path.join(ROCKS_DB_TREE))?; + if config.prover_version != ProverMode::NoProofs { + general.eth.sender.proof_sending_mode = "ONLY_REAL_PROOFS".to_string(); + } + general.save(shell, path)?; + Ok(()) +} + +pub fn update_l1_contracts( + shell: &Shell, + config: &ChainConfig, + register_chain_output: &RegisterChainOutput, +) -> anyhow::Result { + let contracts_config_path = config.configs.join(CONTRACTS_FILE); + let mut contracts_config = ContractsConfig::read(shell, &contracts_config_path)?; + contracts_config.l1.diamond_proxy_addr = register_chain_output.diamond_proxy_addr; + contracts_config.l1.governance_addr = register_chain_output.governance_addr; + contracts_config.save(shell, &contracts_config_path)?; + Ok(contracts_config) +} + +pub fn update_l2_shared_bridge( + shell: &Shell, + config: &ChainConfig, + initialize_bridges_output: &InitializeBridgeOutput, +) -> anyhow::Result<()> { + let contracts_config_path = config.configs.join(CONTRACTS_FILE); + let mut contracts_config = ContractsConfig::read(shell, &contracts_config_path)?; + contracts_config.bridges.shared.l2_address = + Some(initialize_bridges_output.l2_shared_bridge_proxy); + contracts_config.save(shell, &contracts_config_path)?; + Ok(()) +} + +pub fn update_paymaster( + shell: &Shell, + config: &ChainConfig, + paymaster_output: &DeployPaymasterOutput, +) -> anyhow::Result<()> { + let contracts_config_path = config.configs.join(CONTRACTS_FILE); + let mut contracts_config = ContractsConfig::read(shell, &contracts_config_path)?; + contracts_config.l2.testnet_paymaster_addr = paymaster_output.paymaster; + contracts_config.save(shell, &contracts_config_path)?; + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/mod.rs b/zk_toolbox/crates/zk_inception/src/configs/mod.rs new file mode 100644 index 000000000000..329eeb5c1f46 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/mod.rs @@ -0,0 +1,18 @@ +mod chain; +pub mod contracts; +mod ecosystem; +pub mod forge_interface; +mod general; +mod manipulations; +mod secrets; +mod traits; +mod wallets; + +pub use chain::*; +pub use contracts::*; +pub use ecosystem::*; +pub use general::*; +pub use manipulations::*; +pub use secrets::*; +pub use traits::*; +pub use wallets::*; diff --git a/zk_toolbox/crates/zk_inception/src/configs/secrets.rs b/zk_toolbox/crates/zk_inception/src/configs/secrets.rs new file mode 100644 index 000000000000..e95dd05df6a2 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/secrets.rs @@ -0,0 +1,55 @@ +use serde::{Deserialize, Serialize}; +use url::Url; + +use crate::configs::{ReadConfig, SaveConfig}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct DatabaseSecrets { + pub server_url: String, + pub prover_url: String, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct L1Secret { + pub(crate) l1_rpc_url: String, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Secrets { + pub database: DatabaseSecrets, + pub(crate) l1: L1Secret, + #[serde(flatten)] + pub other: serde_json::Value, +} + +#[derive(Debug, Serialize)] +pub struct DatabaseConfig { + pub base_url: Url, + pub database_name: String, +} + +impl DatabaseConfig { + pub fn new(base_url: Url, database_name: String) -> Self { + Self { + base_url, + database_name, + } + } + + pub fn full_url(&self) -> String { + format!("{}/{}", self.base_url, self.database_name) + } +} + +#[derive(Debug, Serialize)] +pub struct DatabasesConfig { + pub server: DatabaseConfig, + pub prover: DatabaseConfig, +} + +impl ReadConfig for Secrets {} +impl SaveConfig for Secrets {} diff --git a/zk_toolbox/crates/zk_inception/src/configs/traits.rs b/zk_toolbox/crates/zk_inception/src/configs/traits.rs new file mode 100644 index 000000000000..29e9fe6c22af --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/traits.rs @@ -0,0 +1,77 @@ +use std::path::Path; + +use anyhow::{bail, Context}; +use common::files::{save_json_file, save_toml_file, save_yaml_file}; +use serde::{de::DeserializeOwned, Serialize}; +use xshell::Shell; + +/// Reads a config file from a given path, correctly parsing file extension. +/// Supported file extensions are: `yaml`, `yml`, `toml`, `json`. +pub trait ReadConfig: DeserializeOwned + Clone { + fn read(shell: &Shell, path: impl AsRef) -> anyhow::Result { + let file = shell.read_file(&path).with_context(|| { + format!( + "Failed to open config file. Please check if the file exists: {:?}", + path.as_ref() + ) + })?; + let error_context = || format!("Failed to parse config file {:?}.", path.as_ref()); + + match path.as_ref().extension().and_then(|ext| ext.to_str()) { + Some("yaml") | Some("yml") => serde_yaml::from_str(&file).with_context(error_context), + Some("toml") => toml::from_str(&file).with_context(error_context), + Some("json") => serde_json::from_str(&file).with_context(error_context), + _ => bail!(format!( + "Unsupported file extension for config file {:?}.", + path.as_ref() + )), + } + } +} + +/// Saves a config file to a given path, correctly parsing file extension. +/// Supported file extensions are: `yaml`, `yml`, `toml`, `json`. +pub trait SaveConfig: Serialize + Sized { + fn save(&self, shell: &Shell, path: impl AsRef) -> anyhow::Result<()> { + save_with_comment(shell, path, self, "") + } +} + +/// Saves a config file to a given path, correctly parsing file extension. +/// Supported file extensions are: `yaml`, `yml`, `toml`. +pub trait SaveConfigWithComment: Serialize + Sized { + fn save_with_comment( + &self, + shell: &Shell, + path: impl AsRef, + comment: &str, + ) -> anyhow::Result<()> { + let comment_char = match path.as_ref().extension().and_then(|ext| ext.to_str()) { + Some("yaml") | Some("yml") | Some("toml") => "#", + _ => bail!("Unsupported file extension for config file."), + }; + let comment_lines = comment + .lines() + .map(|line| format!("{comment_char} {line}")) + .chain(std::iter::once("".to_string())) // Add a newline after the comment + .collect::>() + .join("\n"); + + save_with_comment(shell, path, self, comment_lines) + } +} + +fn save_with_comment( + shell: &Shell, + path: impl AsRef, + data: impl Serialize, + comment: impl ToString, +) -> anyhow::Result<()> { + match path.as_ref().extension().and_then(|ext| ext.to_str()) { + Some("yaml") | Some("yml") => save_yaml_file(shell, path, data, comment)?, + Some("toml") => save_toml_file(shell, path, data, comment)?, + Some("json") => save_json_file(shell, path, data)?, + _ => bail!("Unsupported file extension for config file."), + } + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/configs/wallets.rs b/zk_toolbox/crates/zk_inception/src/configs/wallets.rs new file mode 100644 index 000000000000..fc0b43fcbc01 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/configs/wallets.rs @@ -0,0 +1,60 @@ +use ethers::{core::rand::Rng, types::H256}; +use serde::{Deserialize, Serialize}; + +use crate::{ + configs::{ReadConfig, SaveConfig}, + wallets::Wallet, +}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct WalletsConfig { + pub deployer: Option, + pub operator: Wallet, + pub blob_operator: Wallet, + pub fee_account: Wallet, + pub governor: Wallet, +} + +impl WalletsConfig { + /// Generate random wallets + pub fn random(rng: &mut impl Rng) -> Self { + Self { + deployer: Some(Wallet::random(rng)), + operator: Wallet::random(rng), + blob_operator: Wallet::random(rng), + fee_account: Wallet::random(rng), + governor: Wallet::random(rng), + } + } + + /// Generate placeholder wallets + pub fn empty() -> Self { + Self { + deployer: Some(Wallet::empty()), + operator: Wallet::empty(), + blob_operator: Wallet::empty(), + fee_account: Wallet::empty(), + governor: Wallet::empty(), + } + } + pub fn deployer_private_key(&self) -> Option { + self.deployer.as_ref().and_then(|wallet| wallet.private_key) + } + + pub fn governor_private_key(&self) -> Option { + self.governor.private_key + } +} + +impl ReadConfig for WalletsConfig {} +impl SaveConfig for WalletsConfig {} + +/// ETH config from zkync repository +#[derive(Debug, Serialize, Deserialize, Clone)] +pub(crate) struct EthMnemonicConfig { + pub(crate) test_mnemonic: String, + pub(super) mnemonic: String, + pub(crate) base_path: String, +} + +impl ReadConfig for EthMnemonicConfig {} diff --git a/zk_toolbox/crates/zk_inception/src/consts.rs b/zk_toolbox/crates/zk_inception/src/consts.rs new file mode 100644 index 000000000000..8993981c4c98 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/consts.rs @@ -0,0 +1,105 @@ +use std::path::{Path, PathBuf}; + +use crate::types::ChainId; + +/// Name of the main configuration file +pub(super) const CONFIG_NAME: &str = "ZkStack.yaml"; +/// Name of the wallets file +pub(super) const WALLETS_FILE: &str = "wallets.yaml"; +/// Name of the secrets config file +pub(super) const SECRETS_FILE: &str = "secrets.yaml"; +/// Name of the general config file +pub(super) const GENERAL_FILE: &str = "general.yaml"; +/// Name of the genesis config file +pub(super) const GENESIS_FILE: &str = "genesis.yaml"; + +pub(super) const ERC20_CONFIGS_FILE: &str = "erc20.yaml"; +/// Name of the initial deployments config file +pub(super) const INITIAL_DEPLOYMENT_FILE: &str = "initial_deployments.yaml"; +/// Name of the erc20 deployments config file +pub(super) const ERC20_DEPLOYMENT_FILE: &str = "erc20_deployments.yaml"; +/// Name of the contracts file +pub(super) const CONTRACTS_FILE: &str = "contracts.yaml"; +/// Main repository for the zkSync project +pub(super) const ZKSYNC_ERA_GIT_REPO: &str = "https://github.com/matter-labs/zksync-era"; +/// Name of the docker-compose file inside zksync repository +pub(super) const DOCKER_COMPOSE_FILE: &str = "docker-compose.yml"; +/// Path to the config file with mnemonic for localhost wallets +pub(super) const CONFIGS_PATH: &str = "etc/env/file_based"; +pub(super) const LOCAL_CONFIGS_PATH: &str = "configs/"; +pub(super) const LOCAL_DB_PATH: &str = "db/"; + +/// Path to ecosystem contacts +pub(super) const ECOSYSTEM_PATH: &str = "etc/ecosystem"; + +/// Path to l1 contracts foundry folder inside zksync-era +pub(super) const L1_CONTRACTS_FOUNDRY: &str = "contracts/l1-contracts-foundry"; +/// Path to DeployL1.s.sol script inside zksync-era relative to `L1_CONTRACTS_FOUNDRY` + +pub(super) const ERA_CHAIN_ID: ChainId = ChainId(270); + +pub(super) const TEST_CONFIG_PATH: &str = "etc/test_config/constant/eth.json"; +pub(super) const BASE_PATH: &str = "m/44'/60'/0'"; +pub(super) const AMOUNT_FOR_DISTRIBUTION_TO_WALLETS: u128 = 1000000000000000000000; + +pub(super) const MINIMUM_BALANCE_FOR_WALLET: u128 = 5000000000000000000; + +#[derive(PartialEq, Debug, Clone)] +pub struct ForgeScriptParams { + input: &'static str, + output: &'static str, + script_path: &'static str, +} + +impl ForgeScriptParams { + // Path to the input file for forge script + pub fn input(&self, link_to_code: &Path) -> PathBuf { + link_to_code.join(L1_CONTRACTS_FOUNDRY).join(self.input) + } + + // Path to the output file for forge script + pub fn output(&self, link_to_code: &Path) -> PathBuf { + link_to_code.join(L1_CONTRACTS_FOUNDRY).join(self.output) + } + + // Path to the script + pub fn script(&self) -> PathBuf { + PathBuf::from(self.script_path) + } +} + +pub const DEPLOY_ECOSYSTEM: ForgeScriptParams = ForgeScriptParams { + input: "script-config/config-deploy-l1.toml", + output: "script-out/output-deploy-l1.toml", + script_path: "script/DeployL1.s.sol", +}; + +pub const INITIALIZE_BRIDGES: ForgeScriptParams = ForgeScriptParams { + input: "script-config/config-initialize-shared-bridges.toml", + output: "script-out/output-initialize-shared-bridges.toml", + script_path: "script/InitializeSharedBridgeOnL2.sol", +}; + +pub const REGISTER_CHAIN: ForgeScriptParams = ForgeScriptParams { + input: "script-config/register-hyperchain.toml", + output: "script-out/output-register-hyperchain.toml", + script_path: "script/RegisterHyperchain.s.sol", +}; + +pub const DEPLOY_ERC20: ForgeScriptParams = ForgeScriptParams { + input: "script-config/config-deploy-erc20.toml", + output: "script-out/output-deploy-erc20.toml", + script_path: "script/DeployErc20.s.sol", +}; + +pub const DEPLOY_PAYMASTER: ForgeScriptParams = ForgeScriptParams { + input: "script-config/config-deploy-paymaster.toml", + output: "script-out/output-deploy-paymaster.toml", + script_path: "script/DeployPaymaster.s.sol", +}; + +pub const ACCEPT_GOVERNANCE: ForgeScriptParams = ForgeScriptParams { + input: "script-config/config-accept-admin.toml", + output: "script-out/output-accept-admin.toml", + script_path: "script/AcceptAdmin.s.sol", +}; diff --git a/zk_toolbox/crates/zk_inception/src/defaults.rs b/zk_toolbox/crates/zk_inception/src/defaults.rs new file mode 100644 index 000000000000..4ac90a54fc37 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/defaults.rs @@ -0,0 +1,31 @@ +use crate::configs::ChainConfig; + +pub const DATABASE_SERVER_URL: &str = "postgres://postgres:notsecurepassword@localhost:5432"; +pub const DATABASE_PROVER_URL: &str = "postgres://postgres:notsecurepassword@localhost:5432"; + +pub const ROCKS_DB_STATE_KEEPER: &str = "main/state_keeper"; +pub const ROCKS_DB_TREE: &str = "main/tree"; + +pub const L2_CHAIN_ID: u32 = 271; +/// Path to base chain configuration inside zksync-era +/// Local RPC url +pub(super) const LOCAL_RPC_URL: &str = "http://localhost:8545"; + +pub struct DBNames { + pub server_name: String, + pub prover_name: String, +} +pub fn generate_db_names(config: &ChainConfig) -> DBNames { + DBNames { + server_name: format!( + "zksync_server_{}_{}", + config.l1_network.to_string().to_ascii_lowercase(), + config.name + ), + prover_name: format!( + "zksync_prover_{}_{}", + config.l1_network.to_string().to_ascii_lowercase(), + config.name + ), + } +} diff --git a/zk_toolbox/crates/zk_inception/src/forge_utils.rs b/zk_toolbox/crates/zk_inception/src/forge_utils.rs new file mode 100644 index 000000000000..5ee7564ddf74 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/forge_utils.rs @@ -0,0 +1,31 @@ +use crate::consts::MINIMUM_BALANCE_FOR_WALLET; +use anyhow::anyhow; +use common::forge::ForgeScript; +use ethers::types::H256; + +pub fn fill_forge_private_key( + mut forge: ForgeScript, + private_key: Option, +) -> anyhow::Result { + if !forge.wallet_args_passed() { + forge = + forge.with_private_key(private_key.ok_or(anyhow!("Deployer private key is not set"))?); + } + Ok(forge) +} + +pub async fn check_the_balance(forge: &ForgeScript) -> anyhow::Result<()> { + let Some(address) = forge.address() else { + return Ok(()); + }; + + while !forge + .check_the_balance(MINIMUM_BALANCE_FOR_WALLET.into()) + .await? + { + if common::PromptConfirm::new(format!("Address {address:?} doesn't have enough money to deploy contracts do you want to continue?")).ask() { + break; + } + } + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/main.rs b/zk_toolbox/crates/zk_inception/src/main.rs new file mode 100644 index 000000000000..e4996b4893cf --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/main.rs @@ -0,0 +1,135 @@ +use clap::{command, Parser, Subcommand}; +use common::{ + check_prerequisites, + config::{global_config, init_global_config, GlobalConfig}, + init_prompt_theme, logger, +}; +use xshell::Shell; + +use crate::{ + commands::{args::RunServerArgs, chain::ChainCommands, ecosystem::EcosystemCommands}, + configs::EcosystemConfig, +}; + +pub mod accept_ownership; +mod commands; +mod configs; +mod consts; +mod defaults; +pub mod forge_utils; +pub mod server; +mod types; +mod wallets; + +#[derive(Parser, Debug)] +#[command(version, about)] +struct Inception { + #[command(subcommand)] + command: InceptionSubcommands, + #[clap(flatten)] + global: InceptionGlobalArgs, +} + +#[derive(Subcommand, Debug)] +pub enum InceptionSubcommands { + /// Ecosystem related commands + #[command(subcommand)] + Ecosystem(EcosystemCommands), + /// Chain related commands + #[command(subcommand)] + Chain(ChainCommands), + /// Run server + Server(RunServerArgs), + /// Run containers for local development + Containers, +} + +#[derive(Parser, Debug)] +#[clap(next_help_heading = "Global options")] +struct InceptionGlobalArgs { + /// Verbose mode + #[clap(short, long, global = true)] + verbose: bool, + /// Chain to use + #[clap(long, global = true)] + chain: Option, + /// Ignores prerequisites checks + #[clap(long, global = true)] + ignore_prerequisites: bool, +} + +#[tokio::main] +async fn main() -> anyhow::Result<()> { + human_panic::setup_panic!(); + + init_prompt_theme(); + + logger::new_empty_line(); + logger::intro(); + + let shell = Shell::new().unwrap(); + let inception_args = Inception::parse(); + + init_global_config_inner(&shell, &inception_args.global)?; + + if !global_config().ignore_prerequisites { + check_prerequisites(&shell); + } + + match run_subcommand(inception_args, &shell).await { + Ok(_) => {} + Err(e) => { + logger::error(e.to_string()); + + if e.chain().count() > 1 { + logger::error_note( + "Caused by:", + &e.chain() + .skip(1) + .enumerate() + .map(|(i, cause)| format!(" {i}: {}", cause)) + .collect::>() + .join("\n"), + ); + } + + logger::outro("Failed"); + std::process::exit(1); + } + } + Ok(()) +} + +async fn run_subcommand(inception_args: Inception, shell: &Shell) -> anyhow::Result<()> { + match inception_args.command { + InceptionSubcommands::Ecosystem(args) => commands::ecosystem::run(shell, args).await?, + InceptionSubcommands::Chain(args) => commands::chain::run(shell, args).await?, + InceptionSubcommands::Server(args) => commands::server::run(shell, args)?, + InceptionSubcommands::Containers => commands::containers::run(shell)?, + } + Ok(()) +} + +fn init_global_config_inner( + shell: &Shell, + inception_args: &InceptionGlobalArgs, +) -> anyhow::Result<()> { + if let Some(name) = &inception_args.chain { + if let Ok(config) = EcosystemConfig::from_file(shell) { + let chains = config.list_of_chains(); + if !chains.contains(name) { + anyhow::bail!( + "Chain with name {} doesnt exist, please choose one of {:?}", + name, + &chains + ); + } + } + } + init_global_config(GlobalConfig { + verbose: inception_args.verbose, + chain_name: inception_args.chain.clone(), + ignore_prerequisites: inception_args.ignore_prerequisites, + }); + Ok(()) +} diff --git a/zk_toolbox/crates/zk_inception/src/server.rs b/zk_toolbox/crates/zk_inception/src/server.rs new file mode 100644 index 000000000000..a2cc48677af6 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/server.rs @@ -0,0 +1,94 @@ +use std::path::PathBuf; + +use anyhow::Context; +use common::cmd::Cmd; +use xshell::{cmd, Shell}; + +use crate::{ + configs::ChainConfig, + consts::{CONTRACTS_FILE, GENERAL_FILE, GENESIS_FILE, SECRETS_FILE, WALLETS_FILE}, +}; + +pub struct RunServer { + components: Option>, + code_path: PathBuf, + wallets: PathBuf, + contracts: PathBuf, + general_config: PathBuf, + genesis: PathBuf, + secrets: PathBuf, +} + +pub enum ServerMode { + Normal, + Genesis, +} + +impl RunServer { + pub fn new(components: Option>, chain_config: &ChainConfig) -> Self { + let wallets = chain_config.configs.join(WALLETS_FILE); + let general_config = chain_config.configs.join(GENERAL_FILE); + let genesis = chain_config.configs.join(GENESIS_FILE); + let contracts = chain_config.configs.join(CONTRACTS_FILE); + let secrets = chain_config.configs.join(SECRETS_FILE); + + Self { + components, + code_path: chain_config.link_to_code.clone(), + wallets, + contracts, + general_config, + genesis, + secrets, + } + } + + pub fn run(&self, shell: &Shell, server_mode: ServerMode) -> anyhow::Result<()> { + shell.change_dir(&self.code_path); + let config_genesis = &self.genesis.to_str().unwrap(); + let config_wallets = &self.wallets.to_str().unwrap(); + let config_general_config = &self.general_config.to_str().unwrap(); + let config_contracts = &self.contracts.to_str().unwrap(); + let secrets = &self.secrets.to_str().unwrap(); + let mut additional_args = vec![]; + if let Some(components) = self.components() { + additional_args.push(format!("--components={}", components)) + } + if let ServerMode::Genesis = server_mode { + additional_args.push("--genesis".to_string()); + } + + let mut cmd = Cmd::new( + cmd!( + shell, + "cargo run --release --bin zksync_server -- + --genesis-path {config_genesis} + --wallets-path {config_wallets} + --config-path {config_general_config} + --secrets-path {secrets} + --contracts-config-path {config_contracts} + " + ) + .args(additional_args) + .env_remove("RUSTUP_TOOLCHAIN"), + ); + + // If we are running server in normal mode + // we need to get the output to the console + if let ServerMode::Normal = server_mode { + cmd = cmd.with_force_run(); + } + + cmd.run().context("Failed to run server")?; + Ok(()) + } + + fn components(&self) -> Option { + self.components.as_ref().and_then(|components| { + if components.is_empty() { + return None; + } + Some(components.join(",")) + }) + } +} diff --git a/zk_toolbox/crates/zk_inception/src/types.rs b/zk_toolbox/crates/zk_inception/src/types.rs new file mode 100644 index 000000000000..75c10c804927 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/types.rs @@ -0,0 +1,108 @@ +use std::{fmt::Display, str::FromStr}; + +use clap::ValueEnum; +use ethers::types::Address; +use serde::{Deserialize, Serialize}; +use strum_macros::EnumIter; + +#[derive( + Debug, + Serialize, + Deserialize, + Clone, + Copy, + ValueEnum, + EnumIter, + strum_macros::Display, + Default, + PartialEq, + Eq, +)] +pub enum L1BatchCommitDataGeneratorMode { + #[default] + Rollup, + Validium, +} + +#[derive( + Debug, + Serialize, + Deserialize, + Clone, + Copy, + ValueEnum, + EnumIter, + strum_macros::Display, + PartialEq, + Eq, +)] +pub enum ProverMode { + NoProofs, + Gpu, + Cpu, +} + +#[derive(Copy, Clone, Debug, Default, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)] +pub struct ChainId(pub u32); + +impl Display for ChainId { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!(f, "{}", self.0) + } +} + +impl From for ChainId { + fn from(value: u32) -> Self { + Self(value) + } +} + +#[derive( + Copy, + Clone, + Debug, + Default, + PartialEq, + Eq, + PartialOrd, + Ord, + Serialize, + Deserialize, + ValueEnum, + EnumIter, + strum_macros::Display, +)] +pub enum L1Network { + #[default] + Localhost, + Sepolia, + Mainnet, +} + +impl L1Network { + pub fn chain_id(&self) -> u32 { + match self { + L1Network::Localhost => 9, + L1Network::Sepolia => 11155111, + L1Network::Mainnet => 1, + } + } +} + +#[derive(Debug, Serialize, Deserialize, Clone)] + +pub struct BaseToken { + pub address: Address, + pub nominator: u64, + pub denominator: u64, +} + +impl BaseToken { + pub fn eth() -> Self { + Self { + nominator: 1, + denominator: 1, + address: Address::from_str("0x0000000000000000000000000000000000000001").unwrap(), + } + } +} diff --git a/zk_toolbox/crates/zk_inception/src/wallets/config.rs b/zk_toolbox/crates/zk_inception/src/wallets/config.rs new file mode 100644 index 000000000000..43cb5e969b93 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/wallets/config.rs @@ -0,0 +1,30 @@ +use clap::ValueEnum; +use serde::{Deserialize, Serialize}; +use strum_macros::EnumIter; + +#[derive( + Clone, + Copy, + Debug, + Default, + PartialEq, + Eq, + PartialOrd, + Ord, + Serialize, + Deserialize, + ValueEnum, + EnumIter, + strum_macros::Display, +)] +pub enum WalletCreation { + /// Load wallets from localhost mnemonic, they are funded for localhost env + #[default] + Localhost, + /// Generate random wallets + Random, + /// Generate placeholder wallets + Empty, + /// Specify file with wallets + InFile, +} diff --git a/zk_toolbox/crates/zk_inception/src/wallets/create.rs b/zk_toolbox/crates/zk_inception/src/wallets/create.rs new file mode 100644 index 000000000000..d395206c1804 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/wallets/create.rs @@ -0,0 +1,61 @@ +use std::path::{Path, PathBuf}; + +use common::wallets::Wallet; +use ethers::core::rand::thread_rng; +use xshell::Shell; + +use crate::{ + configs::{EthMnemonicConfig, ReadConfig, SaveConfig, WalletsConfig}, + consts::{BASE_PATH, TEST_CONFIG_PATH}, + wallets::WalletCreation, +}; + +pub fn create_wallets( + shell: &Shell, + dst_wallet_path: &Path, + link_to_code: &Path, + id: u32, + wallet_creation: WalletCreation, + initial_wallet_path: Option, +) -> anyhow::Result<()> { + let wallets = match wallet_creation { + WalletCreation::Random => { + let rng = &mut thread_rng(); + WalletsConfig::random(rng) + } + WalletCreation::Empty => WalletsConfig::empty(), + // Use id of chain for creating + WalletCreation::Localhost => create_localhost_wallets(shell, link_to_code, id)?, + WalletCreation::InFile => { + let path = initial_wallet_path.ok_or(anyhow::anyhow!( + "Wallet path for in file option is required" + ))?; + WalletsConfig::read(shell, path)? + } + }; + + wallets.save(shell, dst_wallet_path)?; + Ok(()) +} + +// Create wallets based on id +pub fn create_localhost_wallets( + shell: &Shell, + link_to_code: &Path, + id: u32, +) -> anyhow::Result { + let path = link_to_code.join(TEST_CONFIG_PATH); + let eth_mnemonic = EthMnemonicConfig::read(shell, path)?; + let base_path = format!("{}/{}", BASE_PATH, id); + Ok(WalletsConfig { + deployer: Some(Wallet::from_mnemonic( + ð_mnemonic.test_mnemonic, + &base_path, + 0, + )?), + operator: Wallet::from_mnemonic(ð_mnemonic.test_mnemonic, &base_path, 1)?, + blob_operator: Wallet::from_mnemonic(ð_mnemonic.test_mnemonic, &base_path, 2)?, + fee_account: Wallet::from_mnemonic(ð_mnemonic.test_mnemonic, &base_path, 3)?, + governor: Wallet::from_mnemonic(ð_mnemonic.test_mnemonic, &base_path, 4)?, + }) +} diff --git a/zk_toolbox/crates/zk_inception/src/wallets/mod.rs b/zk_toolbox/crates/zk_inception/src/wallets/mod.rs new file mode 100644 index 000000000000..eec0d6b0a297 --- /dev/null +++ b/zk_toolbox/crates/zk_inception/src/wallets/mod.rs @@ -0,0 +1,6 @@ +mod config; +mod create; + +pub use common::wallets::Wallet; +pub use config::WalletCreation; +pub use create::{create_localhost_wallets, create_wallets}; diff --git a/zk_toolbox/crates/zk_supervisor/Cargo.toml b/zk_toolbox/crates/zk_supervisor/Cargo.toml new file mode 100644 index 000000000000..74e04fc68aac --- /dev/null +++ b/zk_toolbox/crates/zk_supervisor/Cargo.toml @@ -0,0 +1,14 @@ +[package] +name = "zk_supervisor" +version.workspace = true +edition.workspace = true +homepage.workspace = true +license.workspace = true +authors.workspace = true +exclude.workspace = true +repository.workspace = true +description.workspace = true +keywords.workspace = true + +[dependencies] +human-panic.workspace = true diff --git a/zk_toolbox/crates/zk_supervisor/src/main.rs b/zk_toolbox/crates/zk_supervisor/src/main.rs new file mode 100644 index 000000000000..9936141be106 --- /dev/null +++ b/zk_toolbox/crates/zk_supervisor/src/main.rs @@ -0,0 +1,4 @@ +fn main() { + human_panic::setup_panic!(); + println!("Hello, world!"); +} diff --git a/zk_toolbox/rust-toolchain b/zk_toolbox/rust-toolchain new file mode 100644 index 000000000000..54227249d1ff --- /dev/null +++ b/zk_toolbox/rust-toolchain @@ -0,0 +1 @@ +1.78.0