From afc36ca73146d7e22887fb11c6631a7129d68dd2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?D=C3=B3nal=20Murray?= Date: Tue, 1 Oct 2024 23:43:09 +0200 Subject: [PATCH] Add migration to fix coretime state (#458) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit The `LeaseOffset` was not taken into consideration in the [Polkadot migration](https://github.com/polkadot-fellows/runtimes/blob/e6b9132701df4311e7d6331e3a9e051ae8de0104/relay/polkadot/src/lib.rs#L1965) that ran when [ref 1143](https://polkadot.subsquare.io/referenda/1143) enacted and upgraded the relay chain, so the end timeslices of the leases are short by about 64 days. This PR adds a migration to fix the Coretime state, which currently includes end timeslices which are incorrect. Full notes and plan here: https://hackmd.io/kurv6rTXTF6yvaUJ9uSNoQ for work to continue while I'm OOO try-runtime tests are included in the PR and the HackMD linked above includes steps to manually test the upgrade with chopsticks. --------- Co-authored-by: Tsvetomir Dimitrov Co-authored-by: eskimor Co-authored-by: eskimor Co-authored-by: Bastian Köcher Co-authored-by: fellowship-merge-bot[bot] <151052383+fellowship-merge-bot[bot]@users.noreply.github.com> --- .github/workflows/runtimes-matrix.json | 1 + CHANGELOG.md | 3 +- Cargo.lock | 1 + .../coretime/coretime-polkadot/Cargo.toml | 2 + .../coretime/coretime-polkadot/src/lib.rs | 4 +- .../coretime-polkadot/src/migrations.rs | 556 ++++++++++++++++++ 6 files changed, 565 insertions(+), 2 deletions(-) create mode 100644 system-parachains/coretime/coretime-polkadot/src/migrations.rs diff --git a/.github/workflows/runtimes-matrix.json b/.github/workflows/runtimes-matrix.json index 2107e8d0b0..b38a129e60 100644 --- a/.github/workflows/runtimes-matrix.json +++ b/.github/workflows/runtimes-matrix.json @@ -65,6 +65,7 @@ "name": "coretime-polkadot", "package": "coretime-polkadot-runtime", "path": "system-parachains/coretime/coretime-polkadot", + "uri": "wss://polkadot-coretime-rpc.polkadot.io:443", "is_relay": false }, { diff --git a/CHANGELOG.md b/CHANGELOG.md index 38e51447b8..dfc3118520 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -16,6 +16,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). ### Fixed - Chain-spec generator: propagate the `on_chain_release_build` feature to the chain-spec generator. Without this the live/genesis chain-specs contain a wrongly-configured WASM blob ([polkadot-fellows/runtimes#450](https://github.com/polkadot-fellows/runtimes/pull/450)). +- Adds a migration to the Polkadot Coretime chain to fix an issue from the initial Coretime migration. ([polkadot-fellows/runtimes#458](https://github.com/polkadot-fellows/runtimes/pull/458)) ### Added @@ -25,7 +26,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). ### Fixed -- Kusama: Revert accidental changes to inflation formula ([polkadot-fellows/runtimes#445](tps://github.com/polkadot-fellows/runtimes/pull/445)). +- Kusama: Revert accidental changes to inflation formula ([polkadot-fellows/runtimes#445](https://github.com/polkadot-fellows/runtimes/pull/445)). ## [1.3.1] 23.08.2024 diff --git a/Cargo.lock b/Cargo.lock index 152c89e518..ea1847038f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2806,6 +2806,7 @@ dependencies = [ "serde", "serde_json", "sp-api", + "sp-arithmetic 26.0.0", "sp-block-builder", "sp-consensus-aura", "sp-core 34.0.0", diff --git a/system-parachains/coretime/coretime-polkadot/Cargo.toml b/system-parachains/coretime/coretime-polkadot/Cargo.toml index 164d621dfe..91ef384c04 100644 --- a/system-parachains/coretime/coretime-polkadot/Cargo.toml +++ b/system-parachains/coretime/coretime-polkadot/Cargo.toml @@ -42,6 +42,7 @@ pallet-transaction-payment = { workspace = true } pallet-transaction-payment-rpc-runtime-api = { workspace = true } pallet-utility = { workspace = true } sp-api = { workspace = true } +sp-arithmetic = { workspace = true } sp-block-builder = { workspace = true } sp-consensus-aura = { workspace = true } sp-core = { workspace = true } @@ -131,6 +132,7 @@ std = [ "serde", "serde_json/std", "sp-api/std", + "sp-arithmetic/std", "sp-block-builder/std", "sp-consensus-aura/std", "sp-core/std", diff --git a/system-parachains/coretime/coretime-polkadot/src/lib.rs b/system-parachains/coretime/coretime-polkadot/src/lib.rs index 7b1cdf1ce2..6ae37db720 100644 --- a/system-parachains/coretime/coretime-polkadot/src/lib.rs +++ b/system-parachains/coretime/coretime-polkadot/src/lib.rs @@ -23,6 +23,7 @@ include!(concat!(env!("OUT_DIR"), "/wasm_binary.rs")); mod coretime; +mod migrations; // Genesis preset configurations. pub mod genesis_config_presets; #[cfg(test)] @@ -117,6 +118,7 @@ pub type UncheckedExtrinsic = pub type Migrations = ( // permanent pallet_xcm::migration::MigrateToLatestXcmVersion, + migrations::FixMigration, ); /// Executive: handles dispatch to the various modules. @@ -140,7 +142,7 @@ pub const VERSION: RuntimeVersion = RuntimeVersion { spec_name: create_runtime_str!("coretime-polkadot"), impl_name: create_runtime_str!("coretime-polkadot"), authoring_version: 1, - spec_version: 1_003_000, + spec_version: 1_003_003, impl_version: 0, apis: RUNTIME_API_VERSIONS, transaction_version: 0, diff --git a/system-parachains/coretime/coretime-polkadot/src/migrations.rs b/system-parachains/coretime/coretime-polkadot/src/migrations.rs new file mode 100644 index 0000000000..5e06724cac --- /dev/null +++ b/system-parachains/coretime/coretime-polkadot/src/migrations.rs @@ -0,0 +1,556 @@ +// Copyright (C) Parity Technologies and the various Polkadot contributors, see Contributions.md +// for a list of specific contributors. +// SPDX-License-Identifier: Apache-2.0 + +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +//! Migration to fix the coretime migration lease offset-related issues +//! +//! This fixes a problem with the leases where the relay migration did not take the lease offset +//! into consideration, so the end of leases is 64 days short, in some cases leading to them being +//! dropped completely. + +extern crate alloc; + +use crate::{Runtime, RuntimeOrigin}; +use frame_support::{pallet_prelude::*, traits::OnRuntimeUpgrade}; +use pallet_broker::{ + CompletionStatus, CoreAssignment, CoreIndex, CoreMask, Leases, PotentialRenewalId, + PotentialRenewalRecord, PotentialRenewals, SaleInfo, ScheduleItem, TaskId, Timeslice, + WeightInfo, Workplan, +}; + +use alloc::collections::btree_map::BTreeMap; +use sp_arithmetic::traits::Saturating; +use sp_std::vec::Vec; + +#[cfg(feature = "try-runtime")] +use pallet_broker::{ + CoreAssignment::{Pool, Task}, + LeaseRecordItem, LeasesRecordOf, Reservations, SaleInfoRecordOf, Schedule, +}; +#[cfg(feature = "try-runtime")] +use sp_runtime::TryRuntimeError; +use system_parachains_constants::polkadot::currency::UNITS; + +/// The log target. +const TARGET: &str = "runtime::bootstrapping::fix-migration"; + +struct Weights; +impl Weights { + fn get_first_usable_core() -> Weight { + ::DbWeight::get().reads(1) + } + + fn add_pool_core() -> Weight { + ::DbWeight::get() + .reads_writes(1, 1) + .saturating_add(::WeightInfo::reserve()) + } + + fn extend_short_leases() -> Weight { + ::DbWeight::get().reads_writes(1, 1) + } + + fn remove_premature_renewals_add_back_leases() -> Weight { + let removed_renewals = INCORRECT_RENEWAL_IDS.len() as u64; + ::DbWeight::get() + .reads_writes(removed_renewals, removed_renewals) + .saturating_add( + ::WeightInfo::set_lease() + .saturating_mul(removed_renewals), + ) + } + + fn give_dropped_leases_renewal_rights_and_workplan_entry() -> Weight { + let added_renewals = POTENTIAL_RENEWALS.len() as u64; + ::DbWeight::get().reads_writes(1, 2 * added_renewals) + } +} + +pub struct FixMigration; + +impl FixMigration { + fn get_first_usable_core() -> Option { + let sale_info = SaleInfo::::get()?; + // Cores between first_core and cores_offered are for sale and are thus not usable. + Some(sale_info.first_core + sale_info.cores_offered) + } + + fn add_pool_core(first_core: &mut CoreIndex) { + let Some(sale_info) = SaleInfo::::get() else { + log::error!(target: TARGET, "Retrieving `SaleInfo` failed!"); + return + }; + + // Workplan add (speed up by one cycle): + let schedule = BoundedVec::truncate_from(Vec::from([ScheduleItem { + mask: CoreMask::complete(), + assignment: CoreAssignment::Pool, + }])); + Workplan::::insert((sale_info.region_begin, *first_core), schedule.clone()); + first_core.saturating_inc(); + + // And reserve: + if let Err(err) = pallet_broker::Pallet::::reserve(RuntimeOrigin::root(), schedule) + { + log::error!(target: TARGET, "Adding pool reservation failed: {:?}", err); + } + } + + fn extend_short_leases() { + let short_leases: BTreeMap = [ + (2035, 359280), + (3344, 344160), + (3370, 389520), + (3367, 374400), + (2086, 389520), + (2032, 359280), + (2051, 389520), + (3369, 313920), + (2008, 389520), + (2025, 359280), + (2000, 313920), + (2092, 404640), + (2002, 359280), + (3359, 389520), + (2030, 374400), + (3378, 404640), + (2104, 404640), + (2046, 374400), + (3345, 344160), + (3340, 344160), + (3338, 329040), + (2004, 329040), + (3377, 389520), + (3373, 389520), + (2031, 344160), + (3389, 404640), + (3366, 374400), + (2037, 374400), + (2034, 359280), + (2090, 404640), + (3346, 344160), + (2012, 359280), + (3397, 404640), + (2043, 374400), + (2091, 404640), + (2026, 344160), + (3388, 404640), + (3354, 359280), + (2006, 344160), + (2013, 374400), + ] + .into(); + + let leases = Leases::::get(); + let leases = BoundedVec::truncate_from( + leases + .into_iter() + .map(|mut l| { + let Some(new_until) = short_leases.get(&l.task) else { return l }; + l.until = *new_until; + l + }) + .collect(), + ); + Leases::::put(leases); + } + + // Undo rotate_sale effect on leases that wrongly expired. + // + // Needs to be called before `give_dropped_leases_renewal_rights_and_workplan_entry`. + fn remove_premature_renewals_add_back_leases() { + let premature_renewals = [ + (2094, 298800), + (2040, 298800), + (3333, 298800), + (2106, 298800), + (2093, 298800), + (2101, 298800), + ]; + + if PotentialRenewals::::iter().count() != premature_renewals.len() { + log::error!(target: TARGET, "Unexpected renewal state!"); + } + + if PotentialRenewals::::clear(premature_renewals.len() as u32, None) + .maybe_cursor + .is_some() + { + log::error!(target: TARGET, "Unexpected cursor from clear!"); + } + + for (task, until) in premature_renewals { + if let Err(err) = + pallet_broker::Pallet::::set_lease(RuntimeOrigin::root(), task, until) + { + log::error!(target: TARGET, "Setting lease failed: {:?}", err); + } + } + } + + fn give_dropped_leases_renewal_rights_and_workplan_entry(first_core: &mut CoreIndex) { + let Some(sale_info) = SaleInfo::::get() else { + log::error!(target: TARGET, "Retrieving `SaleInfo` failed!"); + return + }; + + let dropped_leases = + [(2048, 283680), (3375, 283680), (3358, 283680), (2053, 283680), (2056, 283680)]; + + // Leases should have been added, but removed again by rotate_sale - replaces with work + // plan items + renewal rights. + // https://github.com/paritytech/polkadot-sdk/blob/f170af615c0dc413482100892758b236d1fda93b/substrate/frame/broker/src/tick_impls.rs#L212 + // So we need to: + // - Don't change leases, already correct. + // - Add work plan entry. + // - add to potential renewals + + for (task, _) in dropped_leases { + // Workplan + let mask = CoreMask::complete(); + let assignment = CoreAssignment::Task(task); + // DONE: enough to start at next rotation? + // - Yes all good, assignments are all for next rotation already. Hence relay chain + // state is persisted for now. + let schedule = + BoundedVec::truncate_from(Vec::from([ScheduleItem { mask, assignment }])); + Workplan::::insert((sale_info.region_begin, *first_core), schedule.clone()); + + // Renewal: + let renewal_id = PotentialRenewalId { core: *first_core, when: sale_info.region_end }; + let record = PotentialRenewalRecord { + price: 100 * UNITS, + completion: CompletionStatus::Complete(schedule), + }; + PotentialRenewals::::insert(renewal_id, &record); + + first_core.saturating_inc() + } + } +} + +impl OnRuntimeUpgrade for FixMigration { + fn on_runtime_upgrade() -> Weight { + if PotentialRenewals::::get(PotentialRenewalId { core: 6, when: 292605 }).is_none() + { + // Idempotency check - this core will never be renewable at this timeslice ever again. + log::error!(target: TARGET, "This migration includes hardcoded values not relevant to this runtime. Bailing."); + return ::DbWeight::get().reads(1); + } + + let Some(mut first_core) = Self::get_first_usable_core() else { + log::error!(target: TARGET, "Retrieving `SaleInfo` (first_core) failed!"); + // Return dummy weight. This should really not happen and if it does we have bigger + // problems than wrong weights. + return ::DbWeight::get().writes(50) + }; + + Self::extend_short_leases(); + Self::remove_premature_renewals_add_back_leases(); + Self::give_dropped_leases_renewal_rights_and_workplan_entry(&mut first_core); + Self::add_pool_core(&mut first_core); + + match pallet_broker::Pallet::::request_core_count( + RuntimeOrigin::root(), + first_core, + ) { + Ok(_) => log::info!(target: TARGET, "Request for 62 cores sent."), + Err(_) => log::error!(target: TARGET, "Request for 62 cores failed to send."), + } + + Weights::get_first_usable_core() + .saturating_add(Weights::extend_short_leases()) + .saturating_add(Weights::remove_premature_renewals_add_back_leases()) + .saturating_add(Weights::give_dropped_leases_renewal_rights_and_workplan_entry()) + .saturating_add(Weights::add_pool_core()) + .saturating_add(::WeightInfo::request_core_count(62)) + } + + #[cfg(feature = "try-runtime")] + fn pre_upgrade() -> Result, sp_runtime::TryRuntimeError> { + // Idempotency check - this core will never be renewable at this timeslice ever again. + if PotentialRenewals::::get(PotentialRenewalId { core: 6, when: 292605 }).is_none() + { + return Ok(Vec::new()) + } + let sale_info = SaleInfo::::get().unwrap(); + let leases = Leases::::get(); + let pre_upgrade_state = (sale_info, leases); + + assert_eq!(PotentialRenewals::::iter().count(), INCORRECT_RENEWAL_IDS.len()); + + Ok(pre_upgrade_state.encode()) + } + + #[cfg(feature = "try-runtime")] + fn post_upgrade(state: Vec) -> Result<(), TryRuntimeError> { + if state.is_empty() { + return Ok(()) + } + let (prev_sale_info, prev_leases): (SaleInfoRecordOf, LeasesRecordOf) = + Decode::decode(&mut &state[..]).unwrap(); + + log::info!(target: TARGET, "Checking migration."); + + let sale_info = SaleInfo::::get().unwrap(); + + // Check the sale start has not changed. + assert_eq!(sale_info, prev_sale_info); + + // The workplan entries start from the region begin reported by the new SaleInfo. + let workplan_start = sale_info.region_begin; + + let system_chains = [Task(1001), Task(1002), Task(1000), Task(1004), Task(1005), Pool]; + + let reservations = Reservations::::get(); + assert_eq!(reservations.len(), system_chains.len()); + + // Check the reservations are still in the workplan out of an abundance of caution. + log::trace!(target: TARGET, "Checking system chains"); + for (i, (task, reservation)) in + system_chains.iter().zip(reservations.into_iter()).enumerate() + { + log::trace!(target: TARGET, "Checking system chain {:?}", i); + let core_id = if i < 5 { + i + } else { + // 51 ... first core for sale + // 5 ... five cores for sale + // 5 ... the dropped leases + 51 + 5 + 5 + }; + assert_eq!( + Workplan::::get((workplan_start, core_id as u16)), + Some(Schedule::truncate_from(Vec::from([ScheduleItem { + mask: CoreMask::complete(), + assignment: task.clone(), + }]))) + ); + + assert_eq!(reservation.len(), 1); + let reservation = reservation.into_iter().next().unwrap(); + assert_eq!(reservation.assignment, *task); + } + + // Make sure we've got all the leases. + log::trace!(target: TARGET, "Checking leases"); + let leases = Leases::::get(); + assert_eq!(leases.len(), LEASES.iter().filter(|(_, l)| sale_info.region_end <= *l).count()); + + // Make something out of the leases that is easier to check against. + let leases_vec: Vec<(u32, u32)> = + leases.iter().map(|LeaseRecordItem { until, task }| (*task, *until)).collect(); + + // Iterate through hardcoded leases and check they're all correctly in state and scheduled + // in the workplan. + log::trace!(target: TARGET, "Checking leases more thoroughly"); + for (i, (para_id, until)) in LEASES.iter().enumerate() { + // Add the system parachains as an offset - these should come before the leases. + let core_id = i as u16 + 5; + + log::trace!(target: TARGET, "Checking lease existence, core: {:?}, lease: {:?}", core_id, para_id); + assert!( + leases_vec.contains(&(*para_id, *until)), + "Lease entry not found: {:?}", + *para_id + ); + + // This is the entry found in Workplan + let workload = Schedule::truncate_from(Vec::from([ScheduleItem { + mask: CoreMask::complete(), + assignment: Task(*para_id), + }])); + + // They should all be in the workplan for next region. + log::trace!(target: TARGET, "Checking workplan existence"); + assert_eq!( + Workplan::::get((workplan_start, core_id)), + Some(workload), + "Workplan entry not found/invalid." + ); + } + + // For the leases we had before their lease should extend for an additional 11520 + // timeslices (64 days). + log::trace!(target: TARGET, "Checking lease durations"); + for LeaseRecordItem { task, until } in prev_leases.iter() { + log::error!("{task}, {until}"); + assert!(leases_vec.contains(&(*task, *until + 11520))) + } + + // Iterate through hardcoded potential renewals and check they're all correctly in state + // and scheduled in the workplan. + log::trace!(target: TARGET, "Checking potential renewals"); + for (i, (para_id, until)) in POTENTIAL_RENEWALS.iter().enumerate() { + // Add the system parachains and leases as an offset. + // system chains + leases + new cores = 5 + 46 + 5 = 56. + let core_id = i as u16 + 56; + + // This is the entry found in Workplan and PotentialRenewals. + let workload = Schedule::truncate_from(Vec::from([ScheduleItem { + mask: CoreMask::complete(), + assignment: Task(*para_id), + }])); + + // Make sure they're not in the leases. + log::trace!(target: TARGET, "Potential renewal not in leases?"); + assert!(!leases.contains(&LeaseRecordItem { until: *until, task: *para_id })); + + // Ensure they can renew in the next region. + log::trace!(target: TARGET, "Potential renewal exists as expected? core_id: {:?}, when: {:?}, para: {:?}", core_id, sale_info.region_end, para_id); + let renewal_entry = PotentialRenewals::::get(PotentialRenewalId { + core: core_id, + when: sale_info.region_end, + }) + .unwrap(); + log::trace!(target: TARGET, "Renewal entry found: {:?} {:?}", renewal_entry.price / UNITS, renewal_entry.completion); + assert_eq!( + renewal_entry, + PotentialRenewalRecord { + price: 100 * UNITS, + completion: pallet_broker::CompletionStatus::Complete(workload.clone()) + } + ); + assert_eq!(PotentialRenewals::::iter().count(), POTENTIAL_RENEWALS.len()); + + // They should all be in the workplan for next sale. + log::trace!(target: TARGET, "Workplan entry exists as expected?"); + assert_eq!(Workplan::::get((workplan_start, core_id)), Some(workload)); + } + + // Walk the workplan at timeslice 287565 and make sure there is an entry for all 62 cores. + log::trace!(target: TARGET, "Checking workplan"); + for i in 0..62 { + if i >= 51 && i < 56 { + // Cores offered for sale, we don't know anything about them. + continue; + } + log::trace!(target: TARGET, "Core: {:?}", i); + + let entry = Workplan::::get((workplan_start, i)).expect("Entry should exist"); + log::trace!(target: TARGET, "Found entry"); + assert_eq!(entry.len(), 1); + assert_eq!(entry.get(0).unwrap().mask, CoreMask::complete()); + log::trace!(target: TARGET, "Entry complete"); + if i < 5 { + // system chains + assert_eq!(entry.get(0).unwrap().assignment, system_chains[i as usize]); + } else if i < 51 { + // leases + assert_eq!( + entry.get(0).unwrap().assignment, + Task(LEASES.get(i as usize - 5).unwrap().0) + ); + } else if i <= 60 { + // 5 potential renewals + assert_eq!( + entry.get(0).unwrap().assignment, + Task(POTENTIAL_RENEWALS.get(i as usize - 56).unwrap().0) + ); + } else { + // Pool core: + assert_eq!(entry.get(0).unwrap().assignment, Pool,); + } + } + + log::info!(target: TARGET, "Workplan entries: {:?}", Workplan::::iter().count()); + log::debug!(target: TARGET, "Complete workplan:"); + let workplan_by_core: BTreeMap<_, _> = Workplan::::iter() + .map(|((timeslice, core_index), schedule)| { + (core_index, (schedule.into_iter().next().unwrap().assignment, timeslice)) + }) + .collect(); + for (core_index, (task, timeslice)) in workplan_by_core { + log::debug!(target: TARGET, "{:?}, {:?}, {:?}", core_index, task, timeslice); + } + + // Ensure we have requested the correct number of cores. + log::trace!(target: TARGET, "Checking requested core count"); + assert!(frame_system::Pallet::::read_events_no_consensus().any(|e| { + match e.event { + crate::RuntimeEvent::Broker( + pallet_broker::Event::::CoreCountRequested { core_count }, + ) => { + log::info!("{core_count:?}"); + + core_count == 62 + }, + _ => false, + } + })); + + Ok(()) + } +} + +// Incorrect potential renewals in state + +const INCORRECT_RENEWAL_IDS: [(u16, u32); 6] = + [(6, 292605), (5, 292605), (13, 292605), (15, 292605), (47, 292605), (44, 292605)]; + +// Hardcoded para ids and their end timeslice. +// Taken from https://github.com/SBalaguer/coretime-migration/blob/master/polkadot-output-200924.json +const POTENTIAL_RENEWALS: [(u32, u32); 5] = + [(2048, 283680), (3375, 283680), (3358, 283680), (2053, 283680), (2056, 283680)]; + +#[cfg(feature = "try-runtime")] +const LEASES: [(u32, u32); 46] = [ + (2094, 298800), + (2040, 298800), + (2035, 359280), + (3344, 344160), + (3370, 389520), + (3367, 374400), + (2086, 389520), + (2032, 359280), + (3333, 298800), + (2051, 389520), + (2106, 298800), + (3369, 313920), + (2008, 389520), + (2025, 359280), + (2000, 313920), + (2092, 404640), + (2002, 359280), + (3359, 389520), + (2030, 374400), + (3378, 404640), + (2104, 404640), + (2046, 374400), + (3345, 344160), + (3340, 344160), + (3338, 329040), + (2004, 329040), + (3377, 389520), + (3373, 389520), + (2031, 344160), + (3389, 404640), + (3366, 374400), + (2037, 374400), + (2034, 359280), + (2090, 404640), + (3346, 344160), + (2012, 359280), + (3397, 404640), + (2043, 374400), + (2091, 404640), + (2093, 298800), + (2026, 344160), + (3388, 404640), + (2101, 298800), + (3354, 359280), + (2006, 344160), + (2013, 374400), +];