-
Notifications
You must be signed in to change notification settings - Fork 58
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
9 changed files
with
312 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,198 @@ | ||
use std::collections::HashSet; | ||
use std::path::Path; | ||
use std::pin::Pin; | ||
|
||
use anyhow::Result; | ||
use bili_sync_entity::*; | ||
use bili_sync_migration::OnConflict; | ||
use filenamify::filenamify; | ||
use futures::Stream; | ||
use sea_orm::entity::prelude::*; | ||
use sea_orm::ActiveValue::Set; | ||
use sea_orm::{DatabaseConnection, QuerySelect, TransactionTrait}; | ||
|
||
use super::VideoListModel; | ||
use crate::bilibili::{BiliClient, BiliError, Collection, CollectionItem, CollectionType, Video, VideoInfo}; | ||
use crate::config::TEMPLATE; | ||
use crate::utils::id_time_key; | ||
use crate::utils::model::create_video_pages; | ||
use crate::utils::status::Status; | ||
|
||
pub async fn watch_later_from<'a>( | ||
path: &Path, | ||
bili_client: &'a BiliClient, | ||
connection: &DatabaseConnection, | ||
) -> Result<(Box<dyn VideoListModel>, Pin<Box<dyn Stream<Item = VideoInfo> + 'a>>)> { | ||
let watch_later = WatchLater::new(bili_client); | ||
watch_later::Entity::insert(watch_later::ActiveModel { | ||
id: Set(1), | ||
path: Set(path.to_string_lossy().to_string()), | ||
..Default::default() | ||
}) | ||
.on_conflict( | ||
OnConflict::column(watch_later::Column::Id) | ||
.update_column(watch_later::Column::Path) | ||
.to_owned(), | ||
) | ||
.exec(connection) | ||
.await?; | ||
Ok(( | ||
Box::new( | ||
watch_later::Entity::find() | ||
.filter(watch_later::Column::Id.eq(1)) | ||
.one(connection) | ||
.await? | ||
.unwrap(), | ||
), | ||
Box::pin(watch_later.into_video_stream()), | ||
)) | ||
} | ||
use async_trait::async_trait; | ||
|
||
#[async_trait] | ||
impl VideoListModel for watch_later::Model { | ||
async fn video_count(&self, connection: &DatabaseConnection) -> Result<u64> { | ||
Ok(video::Entity::find() | ||
.filter(video::Column::WatchLaterId.eq(self.id)) | ||
.count(connection) | ||
.await?) | ||
} | ||
|
||
async fn unfilled_videos(&self, connection: &DatabaseConnection) -> Result<Vec<video::Model>> { | ||
Ok(video::Entity::find() | ||
.filter( | ||
video::Column::WatchLaterId | ||
.eq(self.id) | ||
.and(video::Column::Valid.eq(true)) | ||
.and(video::Column::DownloadStatus.eq(0)) | ||
.and(video::Column::Category.eq(2)) | ||
.and(video::Column::SinglePage.is_null()), | ||
) | ||
.all(connection) | ||
.await?) | ||
} | ||
|
||
async fn unhandled_video_pages( | ||
&self, | ||
connection: &DatabaseConnection, | ||
) -> Result<Vec<(video::Model, Vec<page::Model>)>> { | ||
Ok(video::Entity::find() | ||
.filter( | ||
video::Column::WatchLaterId | ||
.eq(self.id) | ||
.and(video::Column::Valid.eq(true)) | ||
.and(video::Column::DownloadStatus.lt(Status::handled())) | ||
.and(video::Column::Category.eq(2)) | ||
.and(video::Column::SinglePage.is_not_null()), | ||
) | ||
.find_with_related(page::Entity) | ||
.all(connection) | ||
.await?) | ||
} | ||
|
||
async fn exist_labels( | ||
&self, | ||
videos_info: &[VideoInfo], | ||
connection: &DatabaseConnection, | ||
) -> Result<HashSet<String>> { | ||
let bvids = videos_info.iter().map(|v| v.bvid().to_string()).collect::<Vec<_>>(); | ||
Ok(video::Entity::find() | ||
.filter( | ||
video::Column::WatchLaterId | ||
.eq(self.id) | ||
.and(video::Column::Bvid.is_in(bvids)), | ||
) | ||
.select_only() | ||
.columns([video::Column::Bvid, video::Column::Pubtime]) | ||
.into_tuple() | ||
.all(connection) | ||
.await? | ||
.into_iter() | ||
.map(|(bvid, time)| id_time_key(&bvid, &time)) | ||
.collect::<HashSet<_>>()) | ||
} | ||
|
||
fn video_model_by_info(&self, video_info: &VideoInfo, base_model: Option<video::Model>) -> video::ActiveModel { | ||
let mut video_model = video_info.to_model(base_model); | ||
video_model.watch_later_id = Set(Some(self.id)); | ||
if let Some(fmt_args) = &video_info.to_fmt_args() { | ||
video_model.path = Set(Path::new(&self.path) | ||
.join(filenamify( | ||
TEMPLATE | ||
.render("video", fmt_args) | ||
.unwrap_or_else(|_| video_info.bvid().to_string()), | ||
)) | ||
.to_string_lossy() | ||
.to_string()); | ||
} | ||
video_model | ||
} | ||
|
||
async fn fetch_videos_detail( | ||
&self, | ||
bili_clent: &BiliClient, | ||
videos_model: Vec<video::Model>, | ||
connection: &DatabaseConnection, | ||
) -> Result<()> { | ||
for video_model in videos_model { | ||
let video = Video::new(bili_clent, video_model.bvid.clone()); | ||
let info: Result<_> = async { Ok((video.get_tags().await?, video.get_view_info().await?)) }.await; | ||
match info { | ||
Ok((tags, view_info)) => { | ||
let VideoInfo::View { pages, .. } = &view_info else { | ||
unreachable!("view_info must be VideoInfo::View") | ||
}; | ||
let txn = connection.begin().await?; | ||
// 将分页信息写入数据库 | ||
create_video_pages(pages, &video_model, &txn).await?; | ||
// 将页标记和 tag 写入数据库 | ||
let mut video_active_model = self.video_model_by_info(&view_info, Some(video_model)); | ||
video_active_model.single_page = Set(Some(pages.len() == 1)); | ||
video_active_model.tags = Set(Some(serde_json::to_value(tags).unwrap())); | ||
video_active_model.save(&txn).await?; | ||
txn.commit().await?; | ||
} | ||
Err(e) => { | ||
error!( | ||
"获取视频 {} - {} 的详细信息失败,错误为:{}", | ||
&video_model.bvid, &video_model.name, e | ||
); | ||
if let Some(BiliError::RequestFailed(-404, _)) = e.downcast_ref::<BiliError>() { | ||
let mut video_active_model: video::ActiveModel = video_model.into(); | ||
video_active_model.valid = Set(false); | ||
video_active_model.save(connection).await?; | ||
} | ||
continue; | ||
} | ||
}; | ||
} | ||
Ok(()) | ||
} | ||
|
||
fn log_fetch_video_start(&self) { | ||
info!("开始获取稍后再看的视频与分页信息..."); | ||
} | ||
|
||
fn log_fetch_video_end(&self) { | ||
info!("获取稍后再看的视频与分页信息完成"); | ||
} | ||
|
||
fn log_download_video_start(&self) { | ||
info!("开始下载稍后再看中所有未处理过的视频..."); | ||
} | ||
|
||
fn log_download_video_end(&self) { | ||
info!("下载稍后再看中未处理过的视频完成"); | ||
} | ||
|
||
fn log_refresh_video_start(&self) { | ||
info!("开始扫描稍后再看的新视频..."); | ||
} | ||
|
||
fn log_refresh_video_end(&self, got_count: usize, new_count: u64) { | ||
info!( | ||
"扫描稍后再看的新视频完成,获取了 {} 条新视频,其中有 {} 条新视频", | ||
got_count, new_count, | ||
); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,3 +6,4 @@ pub mod collection; | |
pub mod favorite; | ||
pub mod page; | ||
pub mod video; | ||
pub mod watch_later; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
//! `SeaORM` Entity. Generated by sea-orm-codegen 0.12.15 | ||
use sea_orm::entity::prelude::*; | ||
|
||
#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq)] | ||
#[sea_orm(table_name = "watch_later")] | ||
pub struct Model { | ||
#[sea_orm(primary_key)] | ||
pub id: i32, | ||
pub path: String, | ||
pub created_at: String, | ||
} | ||
|
||
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)] | ||
pub enum Relation {} | ||
|
||
impl ActiveModelBehavior for ActiveModel {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
88 changes: 88 additions & 0 deletions
88
crates/bili_sync_migration/src/m20240709_130914_watch_later.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
use sea_orm_migration::prelude::*; | ||
|
||
#[derive(DeriveMigrationName)] | ||
pub struct Migration; | ||
|
||
#[async_trait::async_trait] | ||
impl MigrationTrait for Migration { | ||
async fn up(&self, manager: &SchemaManager) -> Result<(), DbErr> { | ||
let db = manager.get_connection(); | ||
manager | ||
.create_table( | ||
Table::create() | ||
.table(WatchLater::Table) | ||
.if_not_exists() | ||
.col( | ||
ColumnDef::new(WatchLater::Id) | ||
.unsigned() | ||
.not_null() | ||
.auto_increment() | ||
.primary_key(), | ||
) | ||
.col(ColumnDef::new(WatchLater::Path).string().not_null()) | ||
.col( | ||
ColumnDef::new(WatchLater::CreatedAt) | ||
.timestamp() | ||
.default(Expr::current_timestamp()) | ||
.not_null(), | ||
) | ||
.to_owned(), | ||
) | ||
.await?; | ||
manager | ||
.drop_index( | ||
Index::drop() | ||
.table(Video::Table) | ||
.name("idx_video_cid_fid_bvid") | ||
.to_owned(), | ||
) | ||
.await?; | ||
manager | ||
.alter_table( | ||
Table::alter() | ||
.table(Video::Table) | ||
.add_column(ColumnDef::new(Video::WatchLaterId).unsigned().null()) | ||
.to_owned(), | ||
) | ||
.await?; | ||
db.execute_unprepared("CREATE UNIQUE INDEX `idx_video_unique` ON `video` (ifnull(`collection_id`, -1), ifnull(`favorite_id`, -1), ifnull(`watch_later_id`, -1), `bvid`)") | ||
.await?; | ||
Ok(()) | ||
} | ||
|
||
async fn down(&self, manager: &SchemaManager) -> Result<(), DbErr> { | ||
let db = manager.get_connection(); | ||
manager | ||
.drop_index(Index::drop().table(Video::Table).name("idx_video_unique").to_owned()) | ||
.await?; | ||
db.execute_unprepared("DELETE FROM video WHERE watch_later_id IS NOT NULL") | ||
.await?; | ||
manager | ||
.alter_table( | ||
Table::alter() | ||
.table(Video::Table) | ||
.drop_column(Video::WatchLaterId) | ||
.to_owned(), | ||
) | ||
.await?; | ||
db.execute_unprepared("CREATE UNIQUE INDEX `idx_video_cid_fid_bvid` ON `video` (ifnull(`collection_id`, -1), ifnull(`favorite_id`, -1), `bvid`)") | ||
.await?; | ||
manager | ||
.drop_table(Table::drop().table(WatchLater::Table).to_owned()) | ||
.await | ||
} | ||
} | ||
|
||
#[derive(DeriveIden)] | ||
enum WatchLater { | ||
Table, | ||
Id, | ||
Path, | ||
CreatedAt, | ||
} | ||
|
||
#[derive(DeriveIden)] | ||
enum Video { | ||
Table, | ||
WatchLaterId, | ||
} |