8 Commits

Author SHA1 Message Date
jordan@doyle.la
4eb9049a1a Bump version to v1.0.5 2020-07-23 15:18:22 +01:00
jordan@doyle.la
54f531a0a8 Fixes #16: New lines lost when only a \n is given 2020-07-23 15:15:57 +01:00
jordan@doyle.la
598baa2ace cargo update 2020-07-22 12:12:26 +01:00
Aaron
5b5c7dc2fb Update Rocket dependency branch to "master"
Signed-off-by: Aaron <admin@datahoarder.dev>
2020-07-19 21:06:31 +01:00
jordan@doyle.la
ebe0df5cc6 Upgrade rocket to latest version 2020-07-07 16:05:06 +01:00
Chris Martin
439f4bf2b4 Add viewport meta tag for better mobile interface 2020-06-13 11:15:41 +01:00
Shaarad Dalvi
377d4a129d Upated dockerfile to install libclang-dev
Build of `onig_sys` fails with missing dependency on clang.
2020-05-15 03:08:14 +01:00
jordan@doyle.la
9c53294b88 Move to async rocket 2020-04-13 11:47:50 +01:00
8 changed files with 1137 additions and 591 deletions

1647
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "bin" name = "bin"
version = "1.0.3" version = "1.0.5"
description = "a paste bin." description = "a paste bin."
repository = "https://github.com/w4/bin" repository = "https://github.com/w4/bin"
license = "WTFPL OR 0BSD" license = "WTFPL OR 0BSD"
@@ -10,13 +10,15 @@ edition = "2018"
[dependencies] [dependencies]
owning_ref = "0.4" owning_ref = "0.4"
linked-hash-map = "0.5" linked-hash-map = "0.5"
rocket = { version = "0.4.4", default-features = false } rocket = { git = "https://github.com/SergioBenitez/Rocket", branch = "master" }
askama = "0.9" askama = "0.9"
lazy_static = "1.4" lazy_static = "1.4"
rand = { version = "0.7", features = ["nightly"] } rand = { version = "0.7", features = ["nightly"] }
gpw = "0.1" gpw = "0.1"
syntect = "4.1" syntect = "4.1"
serde_derive = "1.0" serde_derive = "1.0"
tokio = { version = "0.2", features = ["sync", "macros"] }
async-trait = "0.1"
[profile.release] [profile.release]
lto = true lto = true

View File

@@ -1,6 +1,8 @@
FROM rust:1.34.2-slim-stretch AS builder FROM rust:1.34.2-slim-stretch AS builder
RUN rustup install nightly-x86_64-unknown-linux-gnu RUN rustup install nightly-x86_64-unknown-linux-gnu
RUN apt update && apt install -y libclang-dev
COPY . /sources COPY . /sources
WORKDIR /sources WORKDIR /sources
RUN cargo +nightly build --release RUN cargo +nightly build --release

View File

@@ -4,7 +4,7 @@ let src = fetchFromGitHub {
repo = "nixpkgs-mozilla"; repo = "nixpkgs-mozilla";
# commited 19/2/2020 # commited 19/2/2020
rev = "e912ed483e980dfb4666ae0ed17845c4220e5e7c"; rev = "e912ed483e980dfb4666ae0ed17845c4220e5e7c";
sha256 = "0cmvc9fnr38j3n0m4yf0k6s2x589w1rdby1qry1vh435v79gp95j"; sha256 = "08fvzb8w80bkkabc1iyhzd15f4sm7ra10jn32kfch5klgl0gj3j3";
}; };
in in
with import "${src.out}/rust-overlay.nix" pkgs pkgs; with import "${src.out}/rust-overlay.nix" pkgs pkgs;

View File

@@ -8,11 +8,14 @@ use rand::{thread_rng, Rng};
use linked_hash_map::LinkedHashMap; use linked_hash_map::LinkedHashMap;
use owning_ref::RwLockReadGuardRef; use owning_ref::OwningRef;
use std::cell::RefCell; use std::cell::RefCell;
use std::env; use std::env;
use std::sync::{PoisonError, RwLock};
use tokio::sync::{RwLock, RwLockReadGuard};
type RwLockReadGuardRef<'a, T, U = T> = OwningRef<Box<RwLockReadGuard<'a, T>>, U>;
lazy_static! { lazy_static! {
static ref ENTRIES: RwLock<LinkedHashMap<String, String>> = RwLock::new(LinkedHashMap::new()); static ref ENTRIES: RwLock<LinkedHashMap<String, String>> = RwLock::new(LinkedHashMap::new());
@@ -27,13 +30,13 @@ lazy_static! {
/// `ENTRIES.len() - BIN_BUFFER_SIZE` elements will be popped off the front of the map. /// `ENTRIES.len() - BIN_BUFFER_SIZE` elements will be popped off the front of the map.
/// ///
/// During the purge, `ENTRIES` is locked and the current thread will block. /// During the purge, `ENTRIES` is locked and the current thread will block.
fn purge_old() { async fn purge_old() {
let entries_len = ENTRIES.read().unwrap_or_else(PoisonError::into_inner).len(); let entries_len = ENTRIES.read().await.len();
if entries_len > *BUFFER_SIZE { if entries_len > *BUFFER_SIZE {
let to_remove = entries_len - *BUFFER_SIZE; let to_remove = entries_len - *BUFFER_SIZE;
let mut entries = ENTRIES.write().unwrap_or_else(PoisonError::into_inner); let mut entries = ENTRIES.write().await;
for _ in 0..to_remove { for _ in 0..to_remove {
entries.pop_front(); entries.pop_front();
@@ -44,6 +47,7 @@ fn purge_old() {
/// Generates a 'pronounceable' random ID using gpw /// Generates a 'pronounceable' random ID using gpw
pub fn generate_id() -> String { pub fn generate_id() -> String {
thread_local!(static KEYGEN: RefCell<gpw::PasswordGenerator> = RefCell::new(gpw::PasswordGenerator::default())); thread_local!(static KEYGEN: RefCell<gpw::PasswordGenerator> = RefCell::new(gpw::PasswordGenerator::default()));
KEYGEN.with(|k| k.borrow_mut().next()).unwrap_or_else(|| { KEYGEN.with(|k| k.borrow_mut().next()).unwrap_or_else(|| {
thread_rng() thread_rng()
.sample_iter(&Alphanumeric) .sample_iter(&Alphanumeric)
@@ -53,19 +57,21 @@ pub fn generate_id() -> String {
} }
/// Stores a paste under the given id /// Stores a paste under the given id
pub fn store_paste(id: String, content: String) { pub async fn store_paste(id: String, content: String) {
purge_old(); purge_old().await;
ENTRIES ENTRIES
.write() .write()
.unwrap_or_else(PoisonError::into_inner) .await
.insert(id, content); .insert(id, content);
} }
/// Get a paste by id. /// Get a paste by id.
/// ///
/// Returns `None` if the paste doesn't exist. /// Returns `None` if the paste doesn't exist.
pub fn get_paste(id: &str) -> Option<RwLockReadGuardRef<LinkedHashMap<String, String>, String>> { pub async fn get_paste(id: &str) -> Option<RwLockReadGuardRef<'_, LinkedHashMap<String, String>, String>> {
let or = RwLockReadGuardRef::new(ENTRIES.read().unwrap_or_else(PoisonError::into_inner)); // need to box the guard until owning_ref understands Pin is a stable address
let or = RwLockReadGuardRef::new(Box::new(ENTRIES.read().await));
if or.contains_key(id) { if or.contains_key(id) {
Some(or.map(|x| x.get(id).unwrap())) Some(or.map(|x| x.get(id).unwrap()))

View File

@@ -25,7 +25,8 @@ use rocket::response::Redirect;
use rocket::Data; use rocket::Data;
use std::borrow::Cow; use std::borrow::Cow;
use std::io::Read;
use tokio::io::AsyncReadExt;
/// ///
/// Homepage /// Homepage
@@ -53,22 +54,24 @@ struct IndexForm {
} }
#[post("/", data = "<input>")] #[post("/", data = "<input>")]
fn submit(input: Form<IndexForm>) -> Redirect { async fn submit(input: Form<IndexForm>) -> Redirect {
let id = generate_id(); let id = generate_id();
let uri = uri!(show_paste: &id); let uri = uri!(show_paste: &id);
store_paste(id, input.into_inner().val); store_paste(id, input.into_inner().val).await;
Redirect::to(uri) Redirect::to(uri)
} }
#[put("/", data = "<input>")] #[put("/", data = "<input>")]
fn submit_raw(input: Data, host: HostHeader) -> std::io::Result<String> { async fn submit_raw(input: Data, host: HostHeader<'_>) -> Result<String, Status> {
let mut data = String::new(); let mut data = String::new();
input.open().take(1024 * 1000).read_to_string(&mut data)?; input.open().take(1024 * 1000)
.read_to_string(&mut data).await
.map_err(|_| Status::InternalServerError)?;
let id = generate_id(); let id = generate_id();
let uri = uri!(show_paste: &id); let uri = uri!(show_paste: &id);
store_paste(id, data); store_paste(id, data).await;
match *host { match *host {
Some(host) => Ok(format!("https://{}{}", host, uri)), Some(host) => Ok(format!("https://{}{}", host, uri)),
@@ -87,14 +90,12 @@ struct ShowPaste<'a> {
} }
#[get("/<key>")] #[get("/<key>")]
fn show_paste(key: String, plaintext: IsPlaintextRequest) -> Result<Content<String>, Status> { async fn show_paste(key: String, plaintext: IsPlaintextRequest) -> Result<Content<String>, Status> {
let mut splitter = key.splitn(2, '.'); let mut splitter = key.splitn(2, '.');
let key = splitter.next().ok_or_else(|| Status::NotFound)?; let key = splitter.next().ok_or_else(|| Status::NotFound)?;
let ext = splitter.next(); let ext = splitter.next();
// get() returns a read-only lock, we're not going to be writing to this key let entry = &*get_paste(key).await.ok_or_else(|| Status::NotFound)?;
// again so we can hold this for as long as we want
let entry = &*get_paste(key).ok_or_else(|| Status::NotFound)?;
if *plaintext { if *plaintext {
Ok(Content(ContentType::Plain, entry.to_string())) Ok(Content(ContentType::Plain, entry.to_string()))
@@ -110,7 +111,7 @@ fn show_paste(key: String, plaintext: IsPlaintextRequest) -> Result<Content<Stri
// Add <code> tags to enable line numbering with CSS // Add <code> tags to enable line numbering with CSS
let html = format!( let html = format!(
"<code>{}</code>", "<code>{}</code>",
code_highlighted.replace("\n", "</code><code>") code_highlighted.replace("\n", "\n</code><code>")
); );
let content = MarkupDisplay::new_safe(Cow::Borrowed(&html), AskamaHtml); let content = MarkupDisplay::new_safe(Cow::Borrowed(&html), AskamaHtml);
@@ -123,8 +124,14 @@ fn show_paste(key: String, plaintext: IsPlaintextRequest) -> Result<Content<Stri
} }
} }
fn main() { #[tokio::main]
rocket::ignite() async fn main() {
let result = rocket::ignite()
.mount("/", routes![index, submit, submit_raw, show_paste]) .mount("/", routes![index, submit, submit_raw, show_paste])
.launch(); .launch()
.await;
if let Err(e) = result {
eprintln!("Failed to launch Rocket: {:#?}", e);
}
} }

View File

@@ -3,6 +3,8 @@ use std::ops::Deref;
use rocket::request::{FromRequest, Outcome}; use rocket::request::{FromRequest, Outcome};
use rocket::Request; use rocket::Request;
use async_trait::async_trait;
/// Holds a value that determines whether or not this request wanted a plaintext response. /// Holds a value that determines whether or not this request wanted a plaintext response.
/// ///
/// We assume anything with the text/plain Accept or Content-Type headers want plaintext, /// We assume anything with the text/plain Accept or Content-Type headers want plaintext,
@@ -17,10 +19,11 @@ impl Deref for IsPlaintextRequest {
} }
} }
#[async_trait]
impl<'a, 'r> FromRequest<'a, 'r> for IsPlaintextRequest { impl<'a, 'r> FromRequest<'a, 'r> for IsPlaintextRequest {
type Error = (); type Error = ();
fn from_request(request: &'a Request<'r>) -> Outcome<IsPlaintextRequest, ()> { async fn from_request(request: &'a Request<'r>) -> Outcome<IsPlaintextRequest, ()> {
if let Some(format) = request.format() { if let Some(format) = request.format() {
if format.is_plain() { if format.is_plain() {
return Outcome::Success(IsPlaintextRequest(true)); return Outcome::Success(IsPlaintextRequest(true));
@@ -54,10 +57,11 @@ impl<'a> Deref for HostHeader<'a> {
} }
} }
#[async_trait]
impl<'a, 'r> FromRequest<'a, 'r> for HostHeader<'a> { impl<'a, 'r> FromRequest<'a, 'r> for HostHeader<'a> {
type Error = (); type Error = ();
fn from_request(request: &'a Request<'r>) -> Outcome<HostHeader<'a>, ()> { async fn from_request(request: &'a Request<'r>) -> Outcome<HostHeader<'a>, ()> {
Outcome::Success(HostHeader(request.headers().get_one("Host"))) Outcome::Success(HostHeader(request.headers().get_one("Host")))
} }
} }

View File

@@ -2,6 +2,8 @@
<html> <html>
<head> <head>
<meta charset="UTF-8"> <meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>bin.</title> <title>bin.</title>
<link rel="help" href="https://github.com/w4/bin"> <link rel="help" href="https://github.com/w4/bin">