feat(tazjin/tgsa): add stable redirects to telegram images

telegram occasionally changes the upstream urls at which images are
served, with old/new partially overlapping in time afaict.

with this commit, a new endpoint is added to tgsa at

  /img/$user/$post_id/$img_idx

which serves a redirect to the actual image url that telegram had
returned within the last hour (i.e. as per the cache).

next step is to use these image urls in the bbcode itself.

Change-Id: I9aa5cb56bc444cbe796868346c67f2e1e1b79413
Reviewed-on: https://cl.tvl.fyi/c/depot/+/5610
Tested-by: BuildkiteCI
Reviewed-by: tazjin <tazjin@tvl.su>
This commit is contained in:
Vincent Ambo 2022-05-15 01:35:46 +02:00 committed by tazjin
parent f200b1265f
commit b308361a73

View file

@ -192,6 +192,7 @@ const CACHE_EXPIRY: Duration = Duration::from_secs(60 * 60);
struct TgPost { struct TgPost {
bbcode: String, bbcode: String,
at: Instant, at: Instant,
media: Vec<String>,
} }
type Cache = RwLock<HashMap<TgLink, TgPost>>; type Cache = RwLock<HashMap<TgLink, TgPost>>;
@ -209,10 +210,16 @@ fn fetch_with_cache(cache: &Cache, link: &TgLink) -> Result<TgPost> {
let mut writer = cache.write().unwrap(); let mut writer = cache.write().unwrap();
let embed = fetch_embed(&link)?; let embed = fetch_embed(&link)?;
let msg = parse_tgmessage(&embed)?; let mut msg = parse_tgmessage(&embed)?;
let bbcode = to_bbcode(&link, &msg);
let mut media = vec![];
media.append(&mut msg.photos);
media.append(&mut msg.videos);
let post = TgPost { let post = TgPost {
bbcode: to_bbcode(&link, &msg), bbcode,
media,
at: Instant::now(), at: Instant::now(),
}; };
@ -221,6 +228,43 @@ fn fetch_with_cache(cache: &Cache, link: &TgLink) -> Result<TgPost> {
Ok(post) Ok(post)
} }
fn handle_img_redirect(cache: &Cache, img_path: &str) -> Result<rouille::Response> {
// img_path:
//
// RWApodcast/113/1
// ^ ^ ^
// | | |
// | | image (0-indexed)
// | post ID
// username
let img_parts: Vec<&str> = img_path.split("/").collect();
if img_parts.len() != 3 {
println!("invalid image link: {}", img_path);
return Err(anyhow!("not a valid image link: {}", img_path));
}
let link = TgLink {
username: img_parts[0].into(),
message_id: img_parts[1].parse().context("failed to parse message_id")?,
};
let img_idx: usize = img_parts[2].parse().context("failed to parse img_idx")?;
let post = fetch_with_cache(cache, &link)?;
if img_idx >= post.media.len() {
return Err(anyhow!(
"there is no {}. image in {}/{}",
img_idx,
link.username,
link.message_id
));
}
Ok(rouille::Response::redirect_303(post.media[img_idx].clone()))
}
fn handle_tg_link(cache: &Cache, link: &TgLink) -> Result<rouille::Response> { fn handle_tg_link(cache: &Cache, link: &TgLink) -> Result<rouille::Response> {
let post = fetch_with_cache(cache, link)?; let post = fetch_with_cache(cache, link)?;
Ok(rouille::Response::text(post.bbcode)) Ok(rouille::Response::text(post.bbcode))
@ -232,8 +276,12 @@ fn main() {
let cache: Cache = RwLock::new(HashMap::new()); let cache: Cache = RwLock::new(HashMap::new());
rouille::start_server("0.0.0.0:8472", move |request| { rouille::start_server("0.0.0.0:8472", move |request| {
let response = { let response = loop {
match TgLink::parse(request.raw_url()) { if request.raw_url().starts_with("/img/") {
break handle_img_redirect(&cache, &request.raw_url()[5..]);
}
break match TgLink::parse(request.raw_url()) {
None => Ok(rouille::Response::text( None => Ok(rouille::Response::text(
r#"tgsa r#"tgsa
---- ----
@ -258,7 +306,7 @@ pm me on the forums if this makes you mad or something.
"#, "#,
)), )),
Some(link) => handle_tg_link(&cache, &link), Some(link) => handle_tg_link(&cache, &link),
} };
}; };
match response { match response {