proxy github gists and render them on posts

This commit is contained in:
Aravinth Manivannan 2021-11-04 19:36:37 +05:30
parent 8e9b070d46
commit 072e30cba3
Signed by untrusted user: realaravinth
GPG key ID: AD9F0F08E855ED88
7 changed files with 167 additions and 1 deletions

7
Cargo.lock generated
View file

@ -1049,6 +1049,7 @@ dependencies = [
"serde_json", "serde_json",
"sled", "sled",
"url", "url",
"urlencoding",
] ]
[[package]] [[package]]
@ -2121,6 +2122,12 @@ dependencies = [
"percent-encoding", "percent-encoding",
] ]
[[package]]
name = "urlencoding"
version = "2.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "68b90931029ab9b034b300b797048cf23723400aa757e8a2bfb9d748102f9821"
[[package]] [[package]]
name = "vcpkg" name = "vcpkg"
version = "0.2.15" version = "0.2.15"

View file

@ -41,3 +41,4 @@ chrono = "0.4.19"
sled = "0.34.7" sled = "0.34.7"
bincode = "1.3.3" bincode = "1.3.3"
urlencoding = "2.1.0"

View file

@ -17,8 +17,9 @@
use std::ops::{Bound, RangeBounds}; use std::ops::{Bound, RangeBounds};
use actix_web::{http::header, web, HttpResponse, Responder}; use actix_web::{http::header, web, HttpResponse, Responder};
use reqwest::header::CONTENT_TYPE; use reqwest::header::{CONTENT_TYPE, USER_AGENT};
use sailfish::TemplateOnce; use sailfish::TemplateOnce;
use serde::{Deserialize, Serialize};
use crate::data::PostResp; use crate::data::PostResp;
use crate::AppData; use crate::AppData;
@ -30,6 +31,7 @@ pub mod routes {
pub index: &'static str, pub index: &'static str,
pub page: &'static str, pub page: &'static str,
pub asset: &'static str, pub asset: &'static str,
pub gist: &'static str,
} }
impl Proxy { impl Proxy {
@ -38,6 +40,7 @@ pub mod routes {
index: "/", index: "/",
page: "/{username}/{post}", page: "/{username}/{post}",
asset: "/asset/medium/{name}", asset: "/asset/medium/{name}",
gist: "/asset/github-gist",
} }
} }
pub fn get_page(&self, username: &str, post: &str) -> String { pub fn get_page(&self, username: &str, post: &str) -> String {
@ -49,6 +52,14 @@ pub mod routes {
pub fn get_medium_asset(&self, asset_name: &str) -> String { pub fn get_medium_asset(&self, asset_name: &str) -> String {
self.asset.replace("{name}", asset_name) self.asset.replace("{name}", asset_name)
} }
pub fn get_gist(&self, url: &str) -> String {
if let Some(gist_id) = url.split('/').last() {
format!("{}?gist={}", self.gist, urlencoding::encode(gist_id))
} else {
url.to_owned()
}
}
} }
} }
@ -141,6 +152,104 @@ async fn assets(path: web::Path<String>, data: AppData) -> impl Responder {
.body(res.bytes().await.unwrap()) .body(res.bytes().await.unwrap())
} }
#[derive(Deserialize, Serialize)]
struct GistQuery {
gist: String,
}
#[derive(Deserialize, Serialize, TemplateOnce)]
#[template(path = "gist.html")]
#[template(rm_whitespace = true)]
pub struct GistContent {
pub files: Vec<GistFile>,
pub html_url: String,
}
#[derive(TemplateOnce)]
#[template(path = "gist_error.html")]
#[template(rm_whitespace = true)]
pub struct GistContentError;
#[derive(Deserialize, Serialize)]
pub struct GistFile {
pub file_name: String,
pub content: String,
pub language: String,
pub raw_url: String,
}
impl GistFile {
pub fn get_html_content(&self) -> String {
let mut content = self.content.as_str();
if self.content.starts_with('"') {
content = self.content.slice(1..);
}
if content.ends_with('"') {
content = content.slice(..content.len() - 1);
}
content.replace("\\t", " ")
}
}
#[my_codegen::get(path = "crate::V1_API_ROUTES.proxy.gist")]
async fn get_gist(query: web::Query<GistQuery>, data: AppData) -> impl Responder {
const URL: &str = "https://api.github.com/gists/";
let url = format!("{}{}", URL, query.gist);
let resp = data
.client
.get(&url)
.header(USER_AGENT, "libmedium")
.send()
.await
.unwrap()
.json::<serde_json::Value>()
.await
.unwrap();
if let Some(files) = resp.get("files") {
if let serde_json::Value::Object(v) = files {
let mut files = Vec::with_capacity(v.len());
v.iter().for_each(|(name, file_obj)| {
let file = GistFile {
file_name: name.to_string(),
content: file_obj
.get("content")
.unwrap()
.as_str()
.unwrap()
.to_owned(),
language: file_obj
.get("language")
.unwrap()
.as_str()
.unwrap()
.to_owned(),
raw_url: file_obj
.get("raw_url")
.unwrap()
.as_str()
.unwrap()
.to_owned(),
};
files.push(file);
});
let gist = GistContent {
files,
html_url: resp.get("html_url").unwrap().to_string(),
};
return HttpResponse::Ok()
.content_type("text/html")
.body(gist.render_once().unwrap());
}
};
let err = GistContentError {};
HttpResponse::Ok()
.content_type("text/html")
.body(err.render_once().unwrap())
}
#[my_codegen::get(path = "crate::V1_API_ROUTES.proxy.page")] #[my_codegen::get(path = "crate::V1_API_ROUTES.proxy.page")]
async fn page(path: web::Path<(String, String)>, data: AppData) -> impl Responder { async fn page(path: web::Path<(String, String)>, data: AppData) -> impl Responder {
let post_id = path.1.split('-').last(); let post_id = path.1.split('-').last();
@ -162,6 +271,7 @@ async fn page(path: web::Path<(String, String)>, data: AppData) -> impl Responde
pub fn services(cfg: &mut web::ServiceConfig) { pub fn services(cfg: &mut web::ServiceConfig) {
cfg.service(assets); cfg.service(assets);
cfg.service(get_gist);
cfg.service(page); cfg.service(page);
cfg.service(index); cfg.service(index);
} }

18
templates/gist.html Normal file
View file

@ -0,0 +1,18 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Gist</title>
</head>
<body>
<. for file in files {.>
<code> <.= file.get_html_content() .> </code>
<.}.>
<a href="<.= &html_url[1..html_url.len() -1] .>" target="_blank">See on GitHub</a>
</body>
<style>
<. include!("./main.css"); .>
</style>
</html>

14
templates/gist_error.html Normal file
View file

@ -0,0 +1,14 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Gist</title>
</head>
<body>
Error
</body>
<style>
<. include!("./main.css"); .>
</style>
</html>

View file

@ -62,6 +62,17 @@ figcaption {
code { code {
font-family: monospace; font-family: monospace;
font-size: 15px; font-size: 15px;
white-space: pre-wrap;
background-color: #888;
color: #fff;
font-weight: 600;
}
iframe {
width: 100%;
display: block;
margin: auto;
min-height: 100px;
} }
@media screen and (max-width: 1200px) { @media screen and (max-width: 1200px) {

View file

@ -44,7 +44,12 @@
<h6><.= p.text .></h6> <h6><.= p.text .></h6>
<.} else if p.type_ == "IFRAME" {.> <.} else if p.type_ == "IFRAME" {.>
<. let src = &p.iframe.as_ref().unwrap().media_resource.as_ref().unwrap().href; .> <. let src = &p.iframe.as_ref().unwrap().media_resource.as_ref().unwrap().href; .>
<. if src.contains("gist.github.com"){.>
<iframe src="<.= crate::V1_API_ROUTES.proxy.get_gist(&src) .>" frameborder="0"></iframe>
<a href="<.= src .>">Click here to open gist on GitHub</a>
<.} else {.>
<iframe src="<.= src .>" frameborder="0"></iframe> <iframe src="<.= src .>" frameborder="0"></iframe>
<.}.>
<.}.> <.}.>
<.}.> <.}.>
</article> </article>