mirror of
https://github.com/amtoaer/bili-sync.git
synced 2026-06-01 13:50:34 +08:00
119 lines
3.6 KiB
Rust
119 lines
3.6 KiB
Rust
use std::path::Path;
|
|
use std::pin::Pin;
|
|
|
|
use anyhow::{Context, Result};
|
|
use bili_sync_entity::*;
|
|
use futures::Stream;
|
|
use sea_orm::ActiveValue::Set;
|
|
use sea_orm::entity::prelude::*;
|
|
use sea_orm::sea_query::{OnConflict, SimpleExpr};
|
|
use sea_orm::{DatabaseConnection, Unchanged};
|
|
|
|
use crate::adapter::{_ActiveModel, VideoSource, VideoSourceEnum};
|
|
use crate::bilibili::{BiliClient, Collection, CollectionItem, CollectionType, VideoInfo};
|
|
|
|
impl VideoSource for collection::Model {
|
|
fn filter_expr(&self) -> SimpleExpr {
|
|
video::Column::CollectionId.eq(self.id)
|
|
}
|
|
|
|
fn set_relation_id(&self, video_model: &mut video::ActiveModel) {
|
|
video_model.collection_id = Set(Some(self.id));
|
|
}
|
|
|
|
fn path(&self) -> &Path {
|
|
Path::new(self.path.as_str())
|
|
}
|
|
|
|
fn get_latest_row_at(&self) -> DateTime {
|
|
self.latest_row_at
|
|
}
|
|
|
|
fn update_latest_row_at(&self, datetime: DateTime) -> _ActiveModel {
|
|
_ActiveModel::Collection(collection::ActiveModel {
|
|
id: Unchanged(self.id),
|
|
latest_row_at: Set(datetime),
|
|
..Default::default()
|
|
})
|
|
}
|
|
|
|
fn log_refresh_video_start(&self) {
|
|
info!("开始扫描{}「{}」..", CollectionType::from(self.r#type), self.name);
|
|
}
|
|
|
|
fn log_refresh_video_end(&self, count: usize) {
|
|
info!(
|
|
"扫描{}「{}」完成,获取到 {} 条新视频",
|
|
CollectionType::from(self.r#type),
|
|
self.name,
|
|
count,
|
|
);
|
|
}
|
|
|
|
fn log_fetch_video_start(&self) {
|
|
info!(
|
|
"开始填充{}「{}」视频详情..",
|
|
CollectionType::from(self.r#type),
|
|
self.name
|
|
);
|
|
}
|
|
|
|
fn log_fetch_video_end(&self) {
|
|
info!("填充{}「{}」视频详情完成", CollectionType::from(self.r#type), self.name);
|
|
}
|
|
|
|
fn log_download_video_start(&self) {
|
|
info!("开始下载{}「{}」视频..", CollectionType::from(self.r#type), self.name);
|
|
}
|
|
|
|
fn log_download_video_end(&self) {
|
|
info!("下载{}「{}」视频完成", CollectionType::from(self.r#type), self.name);
|
|
}
|
|
}
|
|
|
|
pub(super) async fn collection_from<'a>(
|
|
collection_item: &'a CollectionItem,
|
|
path: &Path,
|
|
bili_client: &'a BiliClient,
|
|
connection: &DatabaseConnection,
|
|
) -> Result<(
|
|
VideoSourceEnum,
|
|
Pin<Box<dyn Stream<Item = Result<VideoInfo>> + 'a + Send>>,
|
|
)> {
|
|
let collection = Collection::new(bili_client, collection_item);
|
|
let collection_info = collection.get_info().await?;
|
|
collection::Entity::insert(collection::ActiveModel {
|
|
s_id: Set(collection_info.sid),
|
|
m_id: Set(collection_info.mid),
|
|
r#type: Set(collection_info.collection_type.into()),
|
|
name: Set(collection_info.name.clone()),
|
|
path: Set(path.to_string_lossy().to_string()),
|
|
..Default::default()
|
|
})
|
|
.on_conflict(
|
|
OnConflict::columns([
|
|
collection::Column::SId,
|
|
collection::Column::MId,
|
|
collection::Column::Type,
|
|
])
|
|
.update_columns([collection::Column::Name, collection::Column::Path])
|
|
.to_owned(),
|
|
)
|
|
.exec(connection)
|
|
.await?;
|
|
Ok((
|
|
collection::Entity::find()
|
|
.filter(
|
|
collection::Column::SId
|
|
.eq(collection_item.sid.clone())
|
|
.and(collection::Column::MId.eq(collection_item.mid.clone()))
|
|
.and(collection::Column::Type.eq(Into::<i32>::into(collection_item.collection_type.clone()))),
|
|
)
|
|
.one(connection)
|
|
.await?
|
|
.context("collection not found")?
|
|
.into(),
|
|
Box::pin(collection.into_video_stream()),
|
|
))
|
|
}
|