Redirect on invalid UTF-8 in /p/

Previously, uploading a paste with invalid UTF-8 and then viewing it
with the pretty URL would cause a panic.
With this change, it simply redirects to the raw URL.
This commit is contained in:
Leonora Tindall
2022-01-31 20:49:05 -06:00
committed by Leonora Tindall
parent ea144a1024
commit 5aafe2500a
5 changed files with 63 additions and 9 deletions

View File

@@ -0,0 +1,31 @@
use rocket::{
request::Request,
response::{Redirect, Responder, Result},
};
use rocket_dyn_templates::Template;
pub enum MaybeRedirect {
Redirect(Box<Redirect>),
Template(Box<Template>),
}
impl From<Redirect> for MaybeRedirect {
fn from(other: Redirect) -> Self {
Self::Redirect(Box::new(other))
}
}
impl From<Template> for MaybeRedirect {
fn from(other: Template) -> Self {
Self::Template(Box::new(other))
}
}
impl<'r, 'o: 'r> Responder<'r, 'o> for MaybeRedirect {
fn respond_to(self, req: &'r Request<'_>) -> Result<'o> {
match self {
Self::Template(t) => t.respond_to(req),
Self::Redirect(r) => r.respond_to(req),
}
}
}

View File

@@ -1,3 +1,4 @@
pub mod maybe_redirect;
pub mod paste_id;
pub mod pretty;
pub mod pretty_syntax;

View File

@@ -10,16 +10,16 @@ static SYNTAXES: &[u8] =
static THEMES: &[u8] =
include_bytes!("../../resources/themes/ayu_dark.tmTheme");
pub fn get_pretty_body(path: &Path, ext: &str) -> String {
pub fn get_pretty_body(path: &Path, ext: &str) -> std::io::Result<String> {
let ss: SyntaxSet = syntect::dumps::from_binary(SYNTAXES);
let mut theme_cursor = std::io::Cursor::new(THEMES);
let theme = ThemeSet::load_from_reader(&mut theme_cursor).unwrap();
let content = fs::read_to_string(path).unwrap();
let content = fs::read_to_string(path)?;
let syntax = ss
.find_syntax_by_token(ext)
.unwrap_or_else(|| ss.find_syntax_plain_text());
highlighted_html_for_string(&content, &ss, syntax, &theme)
Ok(highlighted_html_for_string(&content, &ss, syntax, &theme))
}

View File

@@ -1,17 +1,28 @@
use rocket::response::Redirect;
use rocket_dyn_templates::Template;
use std::collections::HashMap;
use std::io::ErrorKind::InvalidData;
use std::path::Path;
use crate::get_upload_dir;
use crate::models::maybe_redirect::MaybeRedirect;
use crate::models::paste_id::PasteId;
use crate::models::pretty::get_pretty_body;
#[get("/p/<id>", rank = 2)]
pub async fn pretty_retrieve(id: PasteId<'_>) -> Option<Template> {
pub async fn pretty_retrieve(id: PasteId<'_>) -> Option<MaybeRedirect> {
let filepath = Path::new(&get_upload_dir()).join(format!("{id}", id = id));
let contents = get_pretty_body(&filepath, &String::from("txt"));
let contents = match get_pretty_body(&filepath, &String::from("txt")) {
Ok(v) => v,
Err(e) if e.kind() == InvalidData => {
return Some(Redirect::to(format!("/{}", id)).into());
}
_ => {
return None;
}
};
let mut map = HashMap::new();
map.insert("title", id.to_string());
@@ -19,7 +30,7 @@ pub async fn pretty_retrieve(id: PasteId<'_>) -> Option<Template> {
let rendered = Template::render("pretty.html", &map);
match tree_magic::match_filepath("text/plain", &filepath) {
true => Some(rendered),
true => Some(rendered.into()),
false => None,
}
}

View File

@@ -1,22 +1,33 @@
use rocket::response::Redirect;
use rocket_dyn_templates::Template;
use std::collections::HashMap;
use std::io::ErrorKind::InvalidData;
use std::path::Path;
use crate::get_upload_dir;
use crate::models::maybe_redirect::MaybeRedirect;
use crate::models::pretty::get_pretty_body;
use crate::models::pretty_syntax::PasteIdSyntax;
#[get("/p/<id_ext>", rank = 1)]
pub async fn pretty_retrieve_ext(
id_ext: PasteIdSyntax<'_>,
) -> Option<Template> {
) -> Option<MaybeRedirect> {
let id = id_ext.get_fname();
let ext = id_ext.get_ext();
let filepath = Path::new(&get_upload_dir()).join(id.to_string());
let contents = get_pretty_body(&filepath, &ext.to_string());
let contents = match get_pretty_body(&filepath, &ext.to_string()) {
Ok(v) => v,
Err(e) if e.kind() == InvalidData => {
return Some(Redirect::to(format!("/{}", id)).into());
}
_ => {
return None;
}
};
let mut map = HashMap::new();
map.insert("title", id.to_string());
@@ -24,7 +35,7 @@ pub async fn pretty_retrieve_ext(
let rendered = Template::render("pretty.html", &map);
match tree_magic::match_filepath("text/plain", &filepath) {
true => Some(rendered),
true => Some(rendered.into()),
false => None,
}
}