summaryrefslogtreecommitdiff
path: root/src/main.rs
diff options
context:
space:
mode:
authorJesse Morgan <jesse@jesterpm.net>2020-09-18 09:22:03 -0700
committerJesse Morgan <jesse@jesterpm.net>2020-09-18 09:22:03 -0700
commit89734a74faed3c330bed569ac36b6480da8ed8d9 (patch)
tree278b74f181e2866a439720f69c9d9b5699276c03 /src/main.rs
parent001d2d3dccafea7b368cd6545a5a0b000a84ee76 (diff)
First pass at a photo resizing endpoint.
This is currently fetching the images from the media_url, since I ripped it out of another project. I still need to change it to fetch from S3 directly. This is also using actix_web::client instead of reqwest. I should probably switch the oauth module to do the same.
Diffstat (limited to 'src/main.rs')
-rw-r--r--src/main.rs172
1 files changed, 10 insertions, 162 deletions
diff --git a/src/main.rs b/src/main.rs
index d5da4ac..2c81ccd 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -1,171 +1,15 @@
-use actix_multipart::Multipart;
-use actix_web::http::header;
-use actix_web::{middleware, web, App, HttpRequest, HttpResponse, HttpServer};
-
-use chrono::Utc;
-
-use futures::{StreamExt, TryStreamExt};
-
-use rand::distributions::Alphanumeric;
-use rand::{thread_rng, Rng};
+use actix_web::client::Client;
+use actix_web::{middleware, web, App, HttpServer};
use rusoto_core::Region;
-use rusoto_s3::{PutObjectRequest, S3Client, S3};
+use rusoto_s3::S3Client;
use serde::{Deserialize, Serialize};
-use std::collections::HashMap;
-use std::fmt::Display;
-use std::iter;
-
+mod media;
+mod micropub;
mod oauth;
-// To make the timepart shorter, we'll offset it with a custom epoch.
-const EPOCH: i64 = 631152000;
-
-#[derive(Serialize, Deserialize)]
-struct MicropubError {
- error: String,
- #[serde(skip_serializing_if = "Option::is_none")]
- error_description: Option<String>,
-}
-
-impl MicropubError {
- pub fn new<S>(err: S) -> Self
- where
- S: Into<String>,
- {
- MicropubError {
- error: err.into(),
- error_description: None,
- }
- }
-
- pub fn with_description<S, D>(err: S, description: D) -> Self
- where
- S: Into<String>,
- D: Display,
- {
- MicropubError {
- error: err.into(),
- error_description: Some(format!("{}", description)),
- }
- }
-}
-
-fn random_id() -> String {
- let now = Utc::now();
-
- // Generate the time part
- let ts = now.timestamp() - EPOCH;
- let offset = (ts.leading_zeros() / 8) as usize;
- let time_part = base32::encode(
- base32::Alphabet::RFC4648 { padding: false },
- &ts.to_be_bytes()[offset..],
- );
-
- // Generate the random part
- let mut rng = thread_rng();
- let random_part: String = iter::repeat(())
- .map(|()| rng.sample(Alphanumeric))
- .take(7)
- .collect();
-
- format!("{}-{}", time_part, random_part)
-}
-
-async fn handle_upload(req: HttpRequest, mut payload: Multipart) -> HttpResponse {
- let site = req
- .app_data::<web::Data<SiteConfig>>()
- .expect("Missing SiteConfig?");
- let s3_client = req
- .app_data::<web::Data<S3Client>>()
- .expect("Missing S3Client?");
- let verification_service = req
- .app_data::<web::Data<oauth::VerificationService>>()
- .expect("Missing VerificationService?");
-
- let auth_header = match req
- .headers()
- .get(header::AUTHORIZATION)
- .and_then(|s| s.to_str().ok())
- {
- Some(auth_header) => auth_header,
- None => return HttpResponse::Unauthorized().json(MicropubError::new("unauthorized")),
- };
-
- let access_token = match verification_service.validate(auth_header).await {
- Ok(token) => token,
- Err(e) => {
- return HttpResponse::Unauthorized()
- .json(MicropubError::with_description("unauthorized", e))
- }
- };
-
- if !access_token.scopes().any(|s| s == "media") {
- return HttpResponse::Unauthorized().json(MicropubError::new("unauthorized"));
- }
-
- // iterate over multipart stream
- if let Ok(Some(field)) = payload.try_next().await {
- let content_disp = field.content_disposition().unwrap();
- let content_type = field.content_type().clone();
- let filename = content_disp.get_filename();
- let ext = filename.and_then(|f| f.rsplit('.').next());
- let (classification, sep, suffix) = match content_type.type_() {
- mime::IMAGE => ("photo", '.', ext),
- mime::AUDIO => ("audio", '.', ext),
- mime::VIDEO => ("video", '.', ext),
- _ => ("file", '/', filename),
- };
-
- // This will be the key in S3.
- let key = match suffix {
- Some(ext) => format!("{}/{}{}{}", classification, random_id(), sep, ext),
- None => format!("{}/{}", classification, random_id()),
- };
-
- // This will be the publicly accessible URL for the file.
- let url = format!("{}/{}", site.media_url, key);
-
- let mut metadata: HashMap<String, String> = HashMap::new();
- metadata.insert(
- "client-id".to_string(),
- access_token.client_id().to_string(),
- );
- metadata.insert("author".to_string(), access_token.me().to_string());
- if let Some(f) = filename {
- metadata.insert("filename".to_string(), f.to_string());
- }
-
- let body = field
- .map(|b| b.map(|b| b.to_vec()))
- .try_concat()
- .await
- .unwrap();
-
- let put_request = PutObjectRequest {
- bucket: site.s3_bucket().to_owned(),
- key,
- body: Some(body.into()),
- metadata: Some(metadata),
- content_type: Some(content_type.to_string()),
- ..Default::default()
- };
-
- match s3_client.put_object(put_request).await {
- Ok(_) => {
- return HttpResponse::Created()
- .header(header::LOCATION, url)
- .finish()
- }
- Err(e) => return HttpResponse::InternalServerError().body(format!("{}", e)),
- };
- }
-
- HttpResponse::BadRequest().finish()
-}
-
#[derive(Serialize, Deserialize, Clone)]
#[serde(rename_all = "PascalCase")]
pub struct SiteConfig {
@@ -216,10 +60,14 @@ async fn main() -> std::io::Result<()> {
HttpServer::new(move || {
App::new()
.wrap(middleware::Logger::default())
+ .data(Client::new())
.data(site_config.clone())
.data(s3_client.clone())
.data(oauth::VerificationService::new(token_endpoint.clone()))
- .service(web::resource("/micropub/media").route(web::post().to(handle_upload)))
+ .service(
+ web::resource("/micropub/media").route(web::post().to(micropub::handle_upload)),
+ )
+ .configure(media::configure)
})
.bind(bind)?
.run()