feat: 修改各处定义,初步实现读取收藏夹功能

This commit is contained in:
amtoaer
2024-03-27 00:39:52 +08:00
parent 748d0a7830
commit 45c6662d88
7 changed files with 215 additions and 115 deletions

View File

@@ -1,15 +1,12 @@
use std::collections::HashSet;
use std::path::Path;
use std::sync::Arc;
use entity::*;
use futures_util::{pin_mut, StreamExt};
use migration::OnConflict;
use sea_orm::entity::prelude::*;
use sea_orm::ActiveValue::Set;
use sea_orm::QuerySelect;
use crate::bilibili::{BiliClient, FavoriteList, FavoriteListInfo, VideoInfo};
use crate::bilibili::{BiliClient, FavoriteList, Video};
use crate::core::utils::{
create_video_pages, create_videos, exists_bvids_favtime, filter_videos, handle_favorite_info,
};
use crate::Result;
pub async fn process_favorite(
@@ -29,100 +26,23 @@ pub async fn process_favorite(
while let Some(videos_info) = video_stream.next().await {
let exist_bvids_pubtimes =
exists_bvids_favtime(&videos_info, fid, connection.as_ref()).await?;
let video_info_to_create = videos_info
let should_break = videos_info
.iter()
.filter(|v| !exist_bvids_pubtimes.contains(&(v.bvid.clone(), v.fav_time.to_string())))
.collect::<Vec<&VideoInfo>>();
let len = video_info_to_create.len();
if !video_info_to_create.is_empty() {
create_videos(video_info_to_create, &favorite_obj, connection.as_ref()).await?;
// 出现 bvid 和 fav_time 都相同的记录,说明已经到达了上次处理到的位置
.any(|v| exist_bvids_pubtimes.contains(&(v.bvid.clone(), v.fav_time.naive_utc())));
create_videos(&videos_info, &favorite_obj, connection.as_ref()).await?;
let all_unprocessed_videos =
filter_videos(&videos_info, &favorite_obj, true, true, connection.as_ref()).await?;
if !all_unprocessed_videos.is_empty() {
for video in all_unprocessed_videos {
let bili_video = Video::new(bili_client.clone(), video.bvid.clone());
let pages = bili_video.get_pages().await?;
create_video_pages(&pages, &video, connection.as_ref()).await?;
}
}
if videos_info.len() != len {
if should_break {
break;
}
}
Ok(())
}
// 根据获得的收藏夹信息,插入或更新数据库中的收藏夹,并返回收藏夹对象
async fn handle_favorite_info(
info: &FavoriteListInfo,
connection: &DatabaseConnection,
) -> Result<favorite::Model> {
Ok(favorite::Entity::insert(favorite::ActiveModel {
f_id: Set(info.id as i32),
name: Set(info.title.to_string()),
path: Set("/home/amtoaer/Documents/code/rust/bili-sync/video".to_string()),
enabled: Set(true),
..Default::default()
})
.on_conflict(
OnConflict::column(favorite::Column::FId)
.update_column(favorite::Column::Name)
.update_column(favorite::Column::Path)
.update_column(favorite::Column::Enabled)
.to_owned(),
)
.exec_with_returning(connection)
.await?)
}
// 获取数据库中存在的与该视频 favorite_id 和 bvid 重合的视频
async fn exists_bvids_favtime(
videos_info: &[VideoInfo],
fid: i32,
connection: &DatabaseConnection,
) -> Result<HashSet<(String, String)>> {
let bvids = videos_info
.iter()
.map(|v| v.bvid.clone())
.collect::<Vec<String>>();
let exist_bvid_favtime = video::Entity::find()
.filter(
video::Column::FavoriteId
.eq(fid)
.and(video::Column::Bvid.is_in(bvids)),
)
.select_only()
.columns([video::Column::Bvid, video::Column::Favtime])
.all(connection)
.await?
.into_iter()
.map(|v| (v.bvid, v.favtime))
.collect::<HashSet<(String, String)>>();
Ok(exist_bvid_favtime)
}
async fn create_videos(
videos_info: Vec<&VideoInfo>,
favorite_obj: &favorite::Model,
connection: &DatabaseConnection,
) -> Result<u64> {
let video_models = videos_info
.iter()
.map(move |v| video::ActiveModel {
favorite_id: Set(favorite_obj.id),
bvid: Set(v.bvid.clone()),
path: Set(Path::new(favorite_obj.path.as_str())
.join(&v.title)
.to_str()
.unwrap()
.to_string()),
name: Set(v.title.clone()),
category: Set(v.vtype.to_string()),
intro: Set(v.intro.clone()),
cover: Set(v.cover.clone()),
ctime: Set(v.ctime.to_string()),
pubtime: Set(v.pubtime.to_string()),
favtime: Set(v.fav_time.to_string()),
downloaded: Set(false),
valid: Set(true),
tags: Set("[]".to_string()),
single_page: Set(false),
..Default::default()
})
.collect::<Vec<video::ActiveModel>>();
Ok(video::Entity::insert_many(video_models)
.exec_without_returning(connection)
.await?)
}