1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
/*
This file is part of jellything (https://codeberg.org/metamuffin/jellything)
which is licensed under the GNU Affero General Public License (version 3); see /COPYING.
Copyright (C) 2023 metamuffin <metamuffin.org>
*/
use crate::{
database::Database,
routes::ui::{account::session::Session, error::MyResult, CacheControlFile},
};
use anyhow::{anyhow, Context};
use jellybase::{
cache::async_cache_file, federation::Federation, permission::NodePermissionExt,
AssetLocationExt,
};
pub use jellycommon::AssetRole;
use jellycommon::{AssetLocation, LocalTrack, SourceTrackKind, TrackSource};
use log::info;
use rocket::{get, http::ContentType, State};
use std::{path::PathBuf, str::FromStr};
use tokio::fs::File;
#[get("/n/<id>/asset?<role>&<width>")]
pub async fn r_item_assets(
session: Session,
db: &State<Database>,
id: &str,
role: AssetRole,
width: Option<usize>,
) -> MyResult<(ContentType, CacheControlFile)> {
let node = db
.node
.get(&id.to_string())?
.only_if_permitted(&session.user.permissions)
.ok_or(anyhow!("node does not exist"))?;
let mut asset = match role {
AssetRole::Backdrop => node.private.backdrop,
AssetRole::Poster => node.private.poster,
};
if let None = asset {
if let Some(parent) = &node.public.path.last() {
let parent = db.node.get(parent)?.ok_or(anyhow!("node does not exist"))?;
asset = match role {
AssetRole::Backdrop => parent.private.backdrop,
AssetRole::Poster => parent.private.poster,
};
}
};
let asset = asset.unwrap_or(AssetLocation::Assets(
PathBuf::from_str("fallback.avif").unwrap(),
));
Ok(asset_with_res(asset, width).await?)
}
// TODO this can create "federation recursion" because track selection cannot be relied on.
#[get("/n/<id>/thumbnail?<t>&<width>")]
pub async fn r_node_thumbnail(
session: Session,
db: &State<Database>,
fed: &State<Federation>,
id: &str,
t: f64,
width: Option<usize>,
) -> MyResult<(ContentType, CacheControlFile)> {
let node = db
.node
.get(&id.to_string())?
.only_if_permitted(&session.user.permissions)
.ok_or(anyhow!("node does not exist"))?;
let media = node.public.media.ok_or(anyhow!("no media"))?;
let (thumb_track_index, thumb_track) = media
.tracks
.iter()
.enumerate()
.find(|(_i, t)| matches!(t.kind, SourceTrackKind::Video { .. }))
.ok_or(anyhow!("no video track to create a thumbnail of"))?;
let source = node.private.source.ok_or(anyhow!("no source"))?;
let thumb_track_source = &source[thumb_track_index];
if t < 0. || t > media.duration {
Err(anyhow!("thumbnail instant not within media duration"))?
}
let step = 8.;
let t = (t / step).floor() * step;
let asset = match thumb_track_source {
TrackSource::Local(LocalTrack { path, .. }) => {
// the track selected might be different from thumb_track
jellytranscoder::thumbnail::create_thumbnail(path, t).await?
}
TrackSource::Remote(_) => {
let session = fed
.get_session(
thumb_track
.federated
.last()
.ok_or(anyhow!("federation broken"))?,
)
.await?;
async_cache_file(&["fed-thumb", id, &format!("{t}")], |out| {
session.node_thumbnail(out, id, 2048, t)
})
.await?
}
};
Ok(asset_with_res(asset, width).await?)
}
async fn asset_with_res(
asset: AssetLocation,
width: Option<usize>,
) -> MyResult<(ContentType, CacheControlFile)> {
// fit the resolution into a finite set so the maximum cache is finite too.
let width = 2usize.pow(width.unwrap_or(2048).clamp(128, 2048).ilog2());
let path = jellytranscoder::image::transcode(asset, 50., 5, width)
.await
.context("transcoding asset")?;
info!("loading asset from {path:?}");
Ok((
ContentType::AVIF,
CacheControlFile::new(File::open(path.path()).await.context("opening file")?).await,
))
}
|