favicons
|
@ -113,8 +113,8 @@ async fn main() -> std::io::Result<()> {
|
||||||
.configure(v1::services)
|
.configure(v1::services)
|
||||||
.configure(widget::services)
|
.configure(widget::services)
|
||||||
.configure(docs::services)
|
.configure(docs::services)
|
||||||
.configure(static_assets::services)
|
|
||||||
.configure(pages::services)
|
.configure(pages::services)
|
||||||
|
.configure(static_assets::services)
|
||||||
.app_data(get_json_err())
|
.app_data(get_json_err())
|
||||||
})
|
})
|
||||||
.bind(SETTINGS.server.get_ip())
|
.bind(SETTINGS.server.get_ip())
|
||||||
|
|
52
src/static_assets/filemap.rs
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2021 Aravinth Manivannan <realaravinth@batsense.net>
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
use cache_buster::Files;
|
||||||
|
|
||||||
|
pub struct FileMap {
|
||||||
|
pub files: Files,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FileMap {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
let map = include_str!("../cache_buster_data.json");
|
||||||
|
let files = Files::new(&map);
|
||||||
|
Self { files }
|
||||||
|
}
|
||||||
|
pub fn get<'a>(&'a self, path: &'a str) -> Option<&'a str> {
|
||||||
|
// let file_path = self.files.get(path);
|
||||||
|
let file_path = self.files.get_full_path(path);
|
||||||
|
|
||||||
|
if file_path.is_some() {
|
||||||
|
let file_path = &file_path.unwrap()[1..];
|
||||||
|
return Some(file_path);
|
||||||
|
} else {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn filemap_works() {
|
||||||
|
let files = super::FileMap::new();
|
||||||
|
let css = files.get("./static-assets/bundle/bundle.css").unwrap();
|
||||||
|
println!("{}", css);
|
||||||
|
assert!(css.contains("/static/bundle/bundle"));
|
||||||
|
}
|
||||||
|
}
|
26
src/static_assets/mod.rs
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2021 Aravinth Manivannan <realaravinth@batsense.net>
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
pub mod filemap;
|
||||||
|
pub mod static_files;
|
||||||
|
|
||||||
|
pub use filemap::FileMap;
|
||||||
|
|
||||||
|
pub fn services(cfg: &mut actix_web::web::ServiceConfig) {
|
||||||
|
cfg.service(static_files::static_files);
|
||||||
|
cfg.service(static_files::favicons);
|
||||||
|
}
|
|
@ -18,7 +18,7 @@ use std::borrow::Cow;
|
||||||
|
|
||||||
use actix_web::body::Body;
|
use actix_web::body::Body;
|
||||||
use actix_web::{get, http::header, web, HttpResponse, Responder};
|
use actix_web::{get, http::header, web, HttpResponse, Responder};
|
||||||
use cache_buster::Files;
|
use log::debug;
|
||||||
use mime_guess::from_path;
|
use mime_guess::from_path;
|
||||||
use rust_embed::RustEmbed;
|
use rust_embed::RustEmbed;
|
||||||
|
|
||||||
|
@ -28,7 +28,7 @@ use crate::CACHE_AGE;
|
||||||
#[folder = "static/"]
|
#[folder = "static/"]
|
||||||
struct Asset;
|
struct Asset;
|
||||||
|
|
||||||
pub fn handle_embedded_file(path: &str) -> HttpResponse {
|
fn handle_assets(path: &str) -> HttpResponse {
|
||||||
match Asset::get(path) {
|
match Asset::get(path) {
|
||||||
Some(content) => {
|
Some(content) => {
|
||||||
let body: Body = match content {
|
let body: Body = match content {
|
||||||
|
@ -48,37 +48,41 @@ pub fn handle_embedded_file(path: &str) -> HttpResponse {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[get("/static/{_:.*}")]
|
#[get("/static/{_:.*}")]
|
||||||
async fn dist(path: web::Path<String>) -> impl Responder {
|
pub async fn static_files(path: web::Path<String>) -> impl Responder {
|
||||||
handle_embedded_file(&path.0)
|
handle_assets(&path.0)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn services(cfg: &mut web::ServiceConfig) {
|
|
||||||
cfg.service(dist);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct FileMap {
|
|
||||||
files: Files,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl FileMap {
|
#[derive(RustEmbed)]
|
||||||
pub fn new() -> Self {
|
#[folder = "static-assets/favicons/"]
|
||||||
let map = include_str!("cache_buster_data.json");
|
struct Favicons;
|
||||||
let files = Files::new(&map);
|
|
||||||
Self { files }
|
|
||||||
}
|
|
||||||
pub fn get<'a>(&'a self, path: &'a str) -> Option<&'a str> {
|
|
||||||
// let file_path = self.files.get(path);
|
|
||||||
let file_path = self.files.get_full_path(path);
|
|
||||||
|
|
||||||
if file_path.is_some() {
|
fn handle_favicons(path: &str) -> HttpResponse {
|
||||||
let file_path = &file_path.unwrap()[1..];
|
match Favicons::get(path) {
|
||||||
return Some(file_path);
|
Some(content) => {
|
||||||
} else {
|
let body: Body = match content {
|
||||||
return None;
|
Cow::Borrowed(bytes) => bytes.into(),
|
||||||
|
Cow::Owned(bytes) => bytes.into(),
|
||||||
|
};
|
||||||
|
|
||||||
|
HttpResponse::Ok()
|
||||||
|
.set(header::CacheControl(vec![header::CacheDirective::MaxAge(
|
||||||
|
CACHE_AGE,
|
||||||
|
)]))
|
||||||
|
.content_type(from_path(path).first_or_octet_stream().as_ref())
|
||||||
|
.body(body)
|
||||||
}
|
}
|
||||||
|
None => HttpResponse::NotFound().body("404 Not Found"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[get("/{file}")]
|
||||||
|
pub async fn favicons(path: web::Path<String>) -> impl Responder {
|
||||||
|
debug!("searching favicons");
|
||||||
|
handle_favicons(&path.0)
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use actix_web::http::StatusCode;
|
use actix_web::http::StatusCode;
|
||||||
|
@ -86,10 +90,11 @@ mod tests {
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
use crate::tests::*;
|
||||||
|
|
||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn static_assets_work() {
|
async fn static_assets_work() {
|
||||||
let mut app = test::init_service(App::new().configure(services)).await;
|
let mut app = get_app!().await;
|
||||||
|
|
||||||
let resp = test::call_service(
|
let resp = test::call_service(
|
||||||
&mut app,
|
&mut app,
|
||||||
|
@ -99,11 +104,21 @@ mod tests {
|
||||||
assert_eq!(resp.status(), StatusCode::OK);
|
assert_eq!(resp.status(), StatusCode::OK);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[actix_rt::test]
|
||||||
fn filemap_works() {
|
async fn favicons_work() {
|
||||||
let files = super::FileMap::new();
|
|
||||||
let css = files.get("./static-assets/bundle/bundle.css").unwrap();
|
assert!(Favicons::get("favicon.ico").is_some());
|
||||||
println!("{}", css);
|
|
||||||
assert!(css.contains("/static/bundle/bundle"));
|
//let mut app = test::init_service(App::new().configure(services)).await;
|
||||||
|
let mut app = get_app!().await;
|
||||||
|
|
||||||
|
let resp = test::call_service(
|
||||||
|
&mut app,
|
||||||
|
test::TestRequest::get().uri("/favicon.ico").to_request(),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
assert_eq!(resp.status(), StatusCode::OK);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -48,6 +48,21 @@ macro_rules! post_request {
|
||||||
|
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! get_app {
|
macro_rules! get_app {
|
||||||
|
() => {
|
||||||
|
test::init_service(
|
||||||
|
App::new()
|
||||||
|
.wrap(get_identity_service())
|
||||||
|
.wrap(actix_middleware::NormalizePath::new(
|
||||||
|
actix_middleware::normalize::TrailingSlash::Trim,
|
||||||
|
))
|
||||||
|
.configure(crate::api::v1::services)
|
||||||
|
.configure(crate::widget::services)
|
||||||
|
.configure(crate::docs::services)
|
||||||
|
.configure(crate::pages::services)
|
||||||
|
.configure(crate::static_assets::services)
|
||||||
|
)
|
||||||
|
|
||||||
|
};
|
||||||
($data:expr) => {
|
($data:expr) => {
|
||||||
test::init_service(
|
test::init_service(
|
||||||
App::new()
|
App::new()
|
||||||
|
@ -55,9 +70,11 @@ macro_rules! get_app {
|
||||||
.wrap(actix_middleware::NormalizePath::new(
|
.wrap(actix_middleware::NormalizePath::new(
|
||||||
actix_middleware::normalize::TrailingSlash::Trim,
|
actix_middleware::normalize::TrailingSlash::Trim,
|
||||||
))
|
))
|
||||||
.configure(crate::api::v1::pow::services)
|
|
||||||
.configure(crate::api::v1::services)
|
.configure(crate::api::v1::services)
|
||||||
|
.configure(crate::widget::services)
|
||||||
|
.configure(crate::docs::services)
|
||||||
.configure(crate::pages::services)
|
.configure(crate::pages::services)
|
||||||
|
.configure(crate::static_assets::services)
|
||||||
.data($data.clone()),
|
.data($data.clone()),
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
BIN
static-assets/favicons/android-icon-144x144.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
static-assets/favicons/android-icon-192x192.png
Normal file
After Width: | Height: | Size: 58 KiB |
BIN
static-assets/favicons/android-icon-36x36.png
Normal file
After Width: | Height: | Size: 3 KiB |
BIN
static-assets/favicons/android-icon-48x48.png
Normal file
After Width: | Height: | Size: 4.5 KiB |
BIN
static-assets/favicons/android-icon-72x72.png
Normal file
After Width: | Height: | Size: 8.8 KiB |
BIN
static-assets/favicons/android-icon-96x96.png
Normal file
After Width: | Height: | Size: 15 KiB |
BIN
static-assets/favicons/apple-icon-114x114.png
Normal file
After Width: | Height: | Size: 20 KiB |
BIN
static-assets/favicons/apple-icon-120x120.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
static-assets/favicons/apple-icon-144x144.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
static-assets/favicons/apple-icon-152x152.png
Normal file
After Width: | Height: | Size: 34 KiB |
BIN
static-assets/favicons/apple-icon-180x180.png
Normal file
After Width: | Height: | Size: 47 KiB |
BIN
static-assets/favicons/apple-icon-57x57.png
Normal file
After Width: | Height: | Size: 5.9 KiB |
BIN
static-assets/favicons/apple-icon-60x60.png
Normal file
After Width: | Height: | Size: 6.4 KiB |
BIN
static-assets/favicons/apple-icon-72x72.png
Normal file
After Width: | Height: | Size: 8.8 KiB |
BIN
static-assets/favicons/apple-icon-76x76.png
Normal file
After Width: | Height: | Size: 9.6 KiB |
BIN
static-assets/favicons/apple-icon-precomposed.png
Normal file
After Width: | Height: | Size: 58 KiB |
BIN
static-assets/favicons/apple-icon.png
Normal file
After Width: | Height: | Size: 58 KiB |
2
static-assets/favicons/browserconfig.xml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<browserconfig><msapplication><tile><square70x70logo src="/ms-icon-70x70.png"/><square150x150logo src="/ms-icon-150x150.png"/><square310x310logo src="/ms-icon-310x310.png"/><TileColor>#ffffff</TileColor></tile></msapplication></browserconfig>
|
BIN
static-assets/favicons/favicon-16x16.png
Normal file
After Width: | Height: | Size: 1.6 KiB |
BIN
static-assets/favicons/favicon-32x32.png
Normal file
After Width: | Height: | Size: 2.6 KiB |
BIN
static-assets/favicons/favicon-96x96.png
Normal file
After Width: | Height: | Size: 15 KiB |
BIN
static-assets/favicons/favicon.ico
Normal file
After Width: | Height: | Size: 1.1 KiB |
41
static-assets/favicons/manifest.json
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
{
|
||||||
|
"name": "App",
|
||||||
|
"icons": [
|
||||||
|
{
|
||||||
|
"src": "\/android-icon-36x36.png",
|
||||||
|
"sizes": "36x36",
|
||||||
|
"type": "image\/png",
|
||||||
|
"density": "0.75"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "\/android-icon-48x48.png",
|
||||||
|
"sizes": "48x48",
|
||||||
|
"type": "image\/png",
|
||||||
|
"density": "1.0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "\/android-icon-72x72.png",
|
||||||
|
"sizes": "72x72",
|
||||||
|
"type": "image\/png",
|
||||||
|
"density": "1.5"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "\/android-icon-96x96.png",
|
||||||
|
"sizes": "96x96",
|
||||||
|
"type": "image\/png",
|
||||||
|
"density": "2.0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "\/android-icon-144x144.png",
|
||||||
|
"sizes": "144x144",
|
||||||
|
"type": "image\/png",
|
||||||
|
"density": "3.0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "\/android-icon-192x192.png",
|
||||||
|
"sizes": "192x192",
|
||||||
|
"type": "image\/png",
|
||||||
|
"density": "4.0"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
BIN
static-assets/favicons/ms-icon-144x144.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
static-assets/favicons/ms-icon-150x150.png
Normal file
After Width: | Height: | Size: 33 KiB |
BIN
static-assets/favicons/ms-icon-310x310.png
Normal file
After Width: | Height: | Size: 111 KiB |
BIN
static-assets/favicons/ms-icon-70x70.png
Normal file
After Width: | Height: | Size: 8.4 KiB |
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../../components/headers.html"); .>
|
<. include!("../../components/headers/index.html"); .>
|
||||||
<div class="tmp-layout">
|
<div class="tmp-layout">
|
||||||
<main class="auth-main">
|
<main class="auth-main">
|
||||||
<div class="auth-inner-container">
|
<div class="auth-inner-container">
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../../components/headers.html"); .>
|
<. include!("../../components/headers/index.html"); .>
|
||||||
<div class="tmp-layout">
|
<div class="tmp-layout">
|
||||||
<main class="auth-main">
|
<main class="auth-main">
|
||||||
<div class="auth-inner-container">
|
<div class="auth-inner-container">
|
||||||
|
|
17
templates/components/headers/favicon.html
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
<link rel="apple-touch-icon" sizes="57x57" href="/apple-icon-57x57.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="60x60" href="/apple-icon-60x60.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="72x72" href="/apple-icon-72x72.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="76x76" href="/apple-icon-76x76.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="114x114" href="/apple-icon-114x114.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/apple-icon-120x120.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="144x144" href="/apple-icon-144x144.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="152x152" href="/apple-icon-152x152.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/apple-icon-180x180.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="192x192" href="/android-icon-192x192.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="96x96" href="/favicon-96x96.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png">
|
||||||
|
<link rel="manifest" href="/manifest.json">
|
||||||
|
<meta name="msapplication-TileColor" content="#ffffff">
|
||||||
|
<meta name="msapplication-TileImage" content="/ms-icon-144x144.png">
|
||||||
|
<meta name="theme-color" content="#ffffff">
|
|
@ -44,5 +44,6 @@
|
||||||
<meta property="og:site_name"
|
<meta property="og:site_name"
|
||||||
content="mCaptcha is an AGPL'd, privacy focued, proof-of-work based CAPTCHA System"
|
content="mCaptcha is an AGPL'd, privacy focued, proof-of-work based CAPTCHA System"
|
||||||
/>
|
/>
|
||||||
</head>
|
|
||||||
|
|
||||||
|
<. include!("./favicon.html"); .>
|
||||||
|
</head>
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../components/headers.html"); .>
|
<. include!("../components/headers/index.html"); .>
|
||||||
<div class="inner-container">
|
<div class="inner-container">
|
||||||
<div class="error-box">
|
<div class="error-box">
|
||||||
<h1 class="error-title"><.= title .></h1>
|
<h1 class="error-title"><.= title .></h1>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../components/headers.html"); .>
|
<. include!("../components/headers/index.html"); .>
|
||||||
<. include!("./navbar/index.html"); .>
|
<. include!("./navbar/index.html"); .>
|
||||||
<div class="tmp-layout">
|
<div class="tmp-layout">
|
||||||
<. include!("./header/index.html"); .>
|
<. include!("./header/index.html"); .>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../../../components/headers.html"); .>
|
<. include!("../../../components/headers/index.html"); .>
|
||||||
<. include!("../../navbar/index.html"); .>
|
<. include!("../../navbar/index.html"); .>
|
||||||
<div class="tmp-layout">
|
<div class="tmp-layout">
|
||||||
<. include!("../../header/index.html"); .>
|
<. include!("../../header/index.html"); .>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../../../components/headers.html"); .>
|
<. include!("../../../components/headers/index.html"); .>
|
||||||
<. include!("../../navbar/index.html"); .>
|
<. include!("../../navbar/index.html"); .>
|
||||||
<div class="tmp-layout">
|
<div class="tmp-layout">
|
||||||
<. include!("../../header/index.html"); .>
|
<. include!("../../header/index.html"); .>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../../../components/headers.html"); .>
|
<. include!("../../../components/headers/index.html"); .>
|
||||||
<. include!("../../navbar/index.html"); .>
|
<. include!("../../navbar/index.html"); .>
|
||||||
<div class="tmp-layout">
|
<div class="tmp-layout">
|
||||||
<. include!("../../header/index.html"); .>
|
<. include!("../../header/index.html"); .>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<. include!("../components/raw-headers.html"); .>
|
<. include!("../components/headers/raw-headers.html"); .>
|
||||||
<body>
|
<body>
|
||||||
<form class="contaienr">
|
<form class="contaienr">
|
||||||
<label class="checkbox-label" for="verification">
|
<label class="checkbox-label" for="verification">
|
||||||
|
|