2018-02-09 19:00:55 -05:00
|
|
|
use std::io;
|
|
|
|
use std::io::prelude::*;
|
|
|
|
use std::fs::{create_dir_all, File};
|
|
|
|
|
|
|
|
use rocket::Route;
|
|
|
|
use rocket::response::Content;
|
|
|
|
use rocket::http::ContentType;
|
|
|
|
|
|
|
|
use reqwest;
|
|
|
|
|
|
|
|
use CONFIG;
|
|
|
|
|
|
|
|
pub fn routes() -> Vec<Route> {
|
|
|
|
routes![icon]
|
|
|
|
}
|
|
|
|
|
|
|
|
#[get("/<domain>/icon.png")]
|
|
|
|
fn icon(domain: String) -> Content<Vec<u8>> {
|
2018-02-17 12:48:42 -05:00
|
|
|
let icon_type = ContentType::new("image", "x-icon");
|
|
|
|
|
2018-02-09 19:00:55 -05:00
|
|
|
// Validate the domain to avoid directory traversal attacks
|
|
|
|
if domain.contains("/") || domain.contains("..") {
|
2018-02-17 12:48:42 -05:00
|
|
|
return Content(icon_type, get_fallback_icon());
|
2018-02-09 19:00:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
let url = format!("https://icons.bitwarden.com/{}/icon.png", domain);
|
|
|
|
|
|
|
|
// Get the icon, or fallback in case of error
|
|
|
|
let icon = match get_icon_cached(&domain, &url) {
|
|
|
|
Ok(icon) => icon,
|
2018-02-17 12:48:42 -05:00
|
|
|
Err(_) => return Content(icon_type, get_fallback_icon())
|
2018-02-09 19:00:55 -05:00
|
|
|
};
|
|
|
|
|
2018-02-17 12:48:42 -05:00
|
|
|
Content(icon_type, icon)
|
2018-02-09 19:00:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn get_icon(url: &str) -> Result<Vec<u8>, reqwest::Error> {
|
|
|
|
let mut res = reqwest::get(url)?;
|
|
|
|
|
|
|
|
res = match res.error_for_status() {
|
|
|
|
Err(e) => return Err(e),
|
|
|
|
Ok(res) => res
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut buffer: Vec<u8> = vec![];
|
|
|
|
res.copy_to(&mut buffer)?;
|
|
|
|
|
|
|
|
Ok(buffer)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_icon_cached(key: &str, url: &str) -> io::Result<Vec<u8>> {
|
|
|
|
create_dir_all(&CONFIG.icon_cache_folder)?;
|
|
|
|
let path = &format!("{}/{}.png", CONFIG.icon_cache_folder, key);
|
|
|
|
|
2018-02-14 18:53:11 -05:00
|
|
|
// Try to read the cached icon, and return it if it exists
|
2018-02-09 19:00:55 -05:00
|
|
|
match File::open(path) {
|
|
|
|
Ok(mut f) => {
|
|
|
|
let mut buffer = Vec::new();
|
|
|
|
|
|
|
|
if f.read_to_end(&mut buffer).is_ok() {
|
|
|
|
return Ok(buffer);
|
|
|
|
}
|
|
|
|
/* If error reading file continue */
|
|
|
|
}
|
|
|
|
Err(_) => { /* Continue */ }
|
|
|
|
}
|
|
|
|
|
|
|
|
println!("Downloading icon for {}...", key);
|
|
|
|
let icon = match get_icon(url) {
|
|
|
|
Ok(icon) => icon,
|
|
|
|
Err(_) => return Err(io::Error::new(io::ErrorKind::NotFound, ""))
|
|
|
|
};
|
|
|
|
|
2018-02-14 18:53:11 -05:00
|
|
|
// Save the currently downloaded icon
|
2018-02-09 19:00:55 -05:00
|
|
|
match File::create(path) {
|
2018-02-15 13:05:57 -05:00
|
|
|
Ok(mut f) => { f.write_all(&icon).expect("Error writing icon file"); }
|
2018-02-09 19:00:55 -05:00
|
|
|
Err(_) => { /* Continue */ }
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(icon)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_fallback_icon() -> Vec<u8> {
|
|
|
|
let fallback_icon = "https://raw.githubusercontent.com/bitwarden/web/master/src/images/fa-globe.png";
|
|
|
|
get_icon_cached("default", fallback_icon).unwrap()
|
|
|
|
}
|