mirror of
https://git.asonix.dog/asonix/pict-rs.git
synced 2025-01-08 10:35:26 +00:00
Make migration concurrent
This commit is contained in:
parent
19893f0e41
commit
4d909ba32a
4 changed files with 54 additions and 21 deletions
|
@ -1949,7 +1949,8 @@ impl PictRsConfiguration {
|
||||||
|
|
||||||
if arc_repo.get("migrate-0.4").await?.is_none() {
|
if arc_repo.get("migrate-0.4").await?.is_none() {
|
||||||
if let Some(old_repo) = repo_04::open(&config.old_repo)? {
|
if let Some(old_repo) = repo_04::open(&config.old_repo)? {
|
||||||
repo::migrate_04(old_repo, &arc_repo, &store, &config).await?;
|
repo::migrate_04(old_repo, arc_repo.clone(), store.clone(), config.clone())
|
||||||
|
.await?;
|
||||||
arc_repo
|
arc_repo
|
||||||
.set("migrate-0.4", Arc::from(b"migrated".to_vec()))
|
.set("migrate-0.4", Arc::from(b"migrated".to_vec()))
|
||||||
.await?;
|
.await?;
|
||||||
|
@ -2005,7 +2006,8 @@ impl PictRsConfiguration {
|
||||||
|
|
||||||
if arc_repo.get("migrate-0.4").await?.is_none() {
|
if arc_repo.get("migrate-0.4").await?.is_none() {
|
||||||
if let Some(old_repo) = repo_04::open(&config.old_repo)? {
|
if let Some(old_repo) = repo_04::open(&config.old_repo)? {
|
||||||
repo::migrate_04(old_repo, &arc_repo, &store, &config).await?;
|
repo::migrate_04(old_repo, arc_repo.clone(), store.clone(), config.clone())
|
||||||
|
.await?;
|
||||||
arc_repo
|
arc_repo
|
||||||
.set("migrate-0.4", Arc::from(b"migrated".to_vec()))
|
.set("migrate-0.4", Arc::from(b"migrated".to_vec()))
|
||||||
.await?;
|
.await?;
|
||||||
|
|
|
@ -74,7 +74,7 @@ pub(crate) trait FullRepo:
|
||||||
+ AliasRepo
|
+ AliasRepo
|
||||||
+ QueueRepo
|
+ QueueRepo
|
||||||
+ HashRepo
|
+ HashRepo
|
||||||
+ MigrationRepo
|
+ StoreMigrationRepo
|
||||||
+ AliasAccessRepo
|
+ AliasAccessRepo
|
||||||
+ VariantAccessRepo
|
+ VariantAccessRepo
|
||||||
+ ProxyRepo
|
+ ProxyRepo
|
||||||
|
@ -409,7 +409,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
pub(crate) trait MigrationRepo: BaseRepo {
|
pub(crate) trait StoreMigrationRepo: BaseRepo {
|
||||||
async fn is_continuing_migration(&self) -> Result<bool, RepoError>;
|
async fn is_continuing_migration(&self) -> Result<bool, RepoError>;
|
||||||
|
|
||||||
async fn mark_migrated(
|
async fn mark_migrated(
|
||||||
|
@ -424,9 +424,9 @@ pub(crate) trait MigrationRepo: BaseRepo {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
impl<T> MigrationRepo for Arc<T>
|
impl<T> StoreMigrationRepo for Arc<T>
|
||||||
where
|
where
|
||||||
T: MigrationRepo,
|
T: StoreMigrationRepo,
|
||||||
{
|
{
|
||||||
async fn is_continuing_migration(&self) -> Result<bool, RepoError> {
|
async fn is_continuing_migration(&self) -> Result<bool, RepoError> {
|
||||||
T::is_continuing_migration(self).await
|
T::is_continuing_migration(self).await
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
use tokio::task::JoinSet;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
config::Configuration,
|
config::Configuration,
|
||||||
details::Details,
|
details::Details,
|
||||||
|
@ -11,12 +13,15 @@ use crate::{
|
||||||
stream::IntoStreamer,
|
stream::IntoStreamer,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const MIGRATE_CONCURRENCY: usize = 32;
|
||||||
|
const GENERATOR_KEY: &str = "last-path";
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub(crate) async fn migrate_04<S: Store>(
|
pub(crate) async fn migrate_04<S: Store + 'static>(
|
||||||
old_repo: OldSledRepo,
|
old_repo: OldSledRepo,
|
||||||
new_repo: &ArcRepo,
|
new_repo: ArcRepo,
|
||||||
store: &S,
|
store: S,
|
||||||
config: &Configuration,
|
config: Configuration,
|
||||||
) -> Result<(), Error> {
|
) -> Result<(), Error> {
|
||||||
tracing::warn!("Running checks");
|
tracing::warn!("Running checks");
|
||||||
if let Err(e) = old_repo.health_check().await {
|
if let Err(e) = old_repo.health_check().await {
|
||||||
|
@ -39,15 +44,38 @@ pub(crate) async fn migrate_04<S: Store>(
|
||||||
|
|
||||||
let mut hash_stream = old_repo.hashes().await.into_streamer();
|
let mut hash_stream = old_repo.hashes().await.into_streamer();
|
||||||
|
|
||||||
|
let mut set = JoinSet::new();
|
||||||
|
|
||||||
let mut index = 0;
|
let mut index = 0;
|
||||||
while let Some(res) = hash_stream.next().await {
|
while let Some(res) = hash_stream.next().await {
|
||||||
index += 1;
|
|
||||||
if let Ok(hash) = res {
|
if let Ok(hash) = res {
|
||||||
let _ = migrate_hash_04(&old_repo, new_repo, store, config, hash).await;
|
set.spawn_local(migrate_hash_04(
|
||||||
|
old_repo.clone(),
|
||||||
|
new_repo.clone(),
|
||||||
|
store.clone(),
|
||||||
|
config.clone(),
|
||||||
|
hash.clone(),
|
||||||
|
));
|
||||||
} else {
|
} else {
|
||||||
tracing::warn!("Failed to read hash, skipping");
|
tracing::warn!("Failed to read hash, skipping");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
while set.len() >= MIGRATE_CONCURRENCY {
|
||||||
|
if let Some(_) = set.join_next().await {
|
||||||
|
index += 1;
|
||||||
|
|
||||||
|
if index % pct == 0 {
|
||||||
|
let percent = index / pct;
|
||||||
|
|
||||||
|
tracing::warn!("Migration {percent}% complete - {index}/{total_size}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
while let Some(_) = set.join_next().await {
|
||||||
|
index += 1;
|
||||||
|
|
||||||
if index % pct == 0 {
|
if index % pct == 0 {
|
||||||
let percent = index / pct;
|
let percent = index / pct;
|
||||||
|
|
||||||
|
@ -55,25 +83,28 @@ pub(crate) async fn migrate_04<S: Store>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(generator_state) = old_repo.get("last-path").await? {
|
if let Some(generator_state) = old_repo.get(GENERATOR_KEY).await? {
|
||||||
new_repo
|
new_repo
|
||||||
.set("last-path", generator_state.to_vec().into())
|
.set(GENERATOR_KEY, generator_state.to_vec().into())
|
||||||
.await?;
|
.await?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tracing::warn!("Migration complete");
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn migrate_hash_04<S: Store>(
|
async fn migrate_hash_04<S: Store>(
|
||||||
old_repo: &OldSledRepo,
|
old_repo: OldSledRepo,
|
||||||
new_repo: &ArcRepo,
|
new_repo: ArcRepo,
|
||||||
store: &S,
|
store: S,
|
||||||
config: &Configuration,
|
config: Configuration,
|
||||||
old_hash: sled::IVec,
|
old_hash: sled::IVec,
|
||||||
) -> Result<(), Error> {
|
) -> Result<(), Error> {
|
||||||
let mut hash_failures = 0;
|
let mut hash_failures = 0;
|
||||||
|
|
||||||
while let Err(e) = do_migrate_hash_04(old_repo, new_repo, store, config, old_hash.clone()).await
|
while let Err(e) =
|
||||||
|
do_migrate_hash_04(&old_repo, &new_repo, &store, &config, old_hash.clone()).await
|
||||||
{
|
{
|
||||||
hash_failures += 1;
|
hash_failures += 1;
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@ use crate::{
|
||||||
repo::{
|
repo::{
|
||||||
hash::Hash, Alias, AliasAccessRepo, AliasAlreadyExists, AliasRepo, BaseRepo, DeleteToken,
|
hash::Hash, Alias, AliasAccessRepo, AliasAlreadyExists, AliasRepo, BaseRepo, DeleteToken,
|
||||||
Details, FullRepo, HashAlreadyExists, HashRepo, Identifier, IdentifierRepo, JobId,
|
Details, FullRepo, HashAlreadyExists, HashRepo, Identifier, IdentifierRepo, JobId,
|
||||||
MigrationRepo, ProxyRepo, QueueRepo, RepoError, SettingsRepo, UploadId, UploadRepo,
|
ProxyRepo, QueueRepo, RepoError, SettingsRepo, StoreMigrationRepo, UploadId, UploadRepo,
|
||||||
UploadResult, VariantAccessRepo,
|
UploadResult, VariantAccessRepo,
|
||||||
},
|
},
|
||||||
serde_str::Serde,
|
serde_str::Serde,
|
||||||
|
@ -928,7 +928,7 @@ impl IdentifierRepo for SledRepo {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
impl MigrationRepo for SledRepo {
|
impl StoreMigrationRepo for SledRepo {
|
||||||
async fn is_continuing_migration(&self) -> Result<bool, RepoError> {
|
async fn is_continuing_migration(&self) -> Result<bool, RepoError> {
|
||||||
Ok(!self.migration_identifiers.is_empty())
|
Ok(!self.migration_identifiers.is_empty())
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue