mirror of
				https://github.com/dani-garcia/vaultwarden.git
				synced 2025-10-26 16:00:02 +02:00 
			
		
		
		
	Merge pull request #1583 from BlackDex/icon-updates
Updated icon fetching.
This commit is contained in:
		| @@ -37,6 +37,7 @@ static CLIENT: Lazy<Client> = Lazy::new(|| { | |||||||
|  |  | ||||||
| // Build Regex only once since this takes a lot of time. | // Build Regex only once since this takes a lot of time. | ||||||
| static ICON_REL_REGEX: Lazy<Regex> = Lazy::new(|| Regex::new(r"(?i)icon$|apple.*icon").unwrap()); | static ICON_REL_REGEX: Lazy<Regex> = Lazy::new(|| Regex::new(r"(?i)icon$|apple.*icon").unwrap()); | ||||||
|  | static ICON_REL_BLACKLIST: Lazy<Regex> = Lazy::new(|| Regex::new(r"(?i)mask-icon").unwrap()); | ||||||
| static ICON_SIZE_REGEX: Lazy<Regex> = Lazy::new(|| Regex::new(r"(?x)(\d+)\D*(\d+)").unwrap()); | static ICON_SIZE_REGEX: Lazy<Regex> = Lazy::new(|| Regex::new(r"(?x)(\d+)\D*(\d+)").unwrap()); | ||||||
|  |  | ||||||
| // Special HashMap which holds the user defined Regex to speedup matching the regex. | // Special HashMap which holds the user defined Regex to speedup matching the regex. | ||||||
| @@ -52,7 +53,9 @@ fn icon(domain: String) -> Cached<Content<Vec<u8>>> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     match get_icon(&domain) { |     match get_icon(&domain) { | ||||||
|         Some(i) => Cached::ttl(Content(ContentType::new("image", "x-icon"), i), CONFIG.icon_cache_ttl()), |         Some((icon, icon_type)) => { | ||||||
|  |             Cached::ttl(Content(ContentType::new("image", icon_type), icon), CONFIG.icon_cache_ttl()) | ||||||
|  |         }, | ||||||
|         _ => Cached::ttl(Content(ContentType::new("image", "png"), FALLBACK_ICON.to_vec()), CONFIG.icon_cache_negttl()), |         _ => Cached::ttl(Content(ContentType::new("image", "png"), FALLBACK_ICON.to_vec()), CONFIG.icon_cache_negttl()), | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -243,7 +246,7 @@ fn is_domain_blacklisted(domain: &str) -> bool { | |||||||
|     is_blacklisted |     is_blacklisted | ||||||
| } | } | ||||||
|  |  | ||||||
| fn get_icon(domain: &str) -> Option<Vec<u8>> { | fn get_icon(domain: &str) -> Option<(Vec<u8>, String)> { | ||||||
|     let path = format!("{}/{}.png", CONFIG.icon_cache_folder(), domain); |     let path = format!("{}/{}.png", CONFIG.icon_cache_folder(), domain); | ||||||
|  |  | ||||||
|     // Check for expiration of negatively cached copy |     // Check for expiration of negatively cached copy | ||||||
| @@ -252,7 +255,11 @@ fn get_icon(domain: &str) -> Option<Vec<u8>> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     if let Some(icon) = get_cached_icon(&path) { |     if let Some(icon) = get_cached_icon(&path) { | ||||||
|         return Some(icon); |         let icon_type =  match get_icon_type(&icon) { | ||||||
|  |             Some(x) => x, | ||||||
|  |             _ => "x-icon", | ||||||
|  |         }; | ||||||
|  |         return Some((icon, icon_type.to_string())); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if CONFIG.disable_icon_download() { |     if CONFIG.disable_icon_download() { | ||||||
| @@ -261,9 +268,9 @@ fn get_icon(domain: &str) -> Option<Vec<u8>> { | |||||||
|  |  | ||||||
|     // Get the icon, or None in case of error |     // Get the icon, or None in case of error | ||||||
|     match download_icon(&domain) { |     match download_icon(&domain) { | ||||||
|         Ok(icon) => { |         Ok((icon, icon_type)) => { | ||||||
|             save_icon(&path, &icon); |             save_icon(&path, &icon); | ||||||
|             Some(icon) |             Some((icon, icon_type.unwrap_or("x-icon").to_string())) | ||||||
|         } |         } | ||||||
|         Err(e) => { |         Err(e) => { | ||||||
|             error!("Error downloading icon: {:?}", e); |             error!("Error downloading icon: {:?}", e); | ||||||
| @@ -324,7 +331,6 @@ fn icon_is_expired(path: &str) -> bool { | |||||||
|     expired.unwrap_or(true) |     expired.unwrap_or(true) | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug)] |  | ||||||
| struct Icon { | struct Icon { | ||||||
|     priority: u8, |     priority: u8, | ||||||
|     href: String, |     href: String, | ||||||
| @@ -348,7 +354,7 @@ fn get_favicons_node(node: &std::rc::Rc<markup5ever_rcdom::Node>, icons: &mut Ve | |||||||
|                 let attr_name = attr.name.local.as_ref(); |                 let attr_name = attr.name.local.as_ref(); | ||||||
|                 let attr_value = attr.value.as_ref(); |                 let attr_value = attr.value.as_ref(); | ||||||
|  |  | ||||||
|                 if attr_name == "rel" && ICON_REL_REGEX.is_match(attr_value) { |                 if attr_name == "rel" && ICON_REL_REGEX.is_match(attr_value) && !ICON_REL_BLACKLIST.is_match(attr_value) { | ||||||
|                     has_rel = true; |                     has_rel = true; | ||||||
|                 } else if attr_name == "href" { |                 } else if attr_name == "href" { | ||||||
|                     href = Some(attr_value); |                     href = Some(attr_value); | ||||||
| @@ -597,7 +603,7 @@ fn parse_sizes(sizes: Option<&str>) -> (u16, u16) { | |||||||
|     (width, height) |     (width, height) | ||||||
| } | } | ||||||
|  |  | ||||||
| fn download_icon(domain: &str) -> Result<Vec<u8>, Error> { | fn download_icon(domain: &str) -> Result<(Vec<u8>, Option<&str>), Error> { | ||||||
|     if is_domain_blacklisted(domain) { |     if is_domain_blacklisted(domain) { | ||||||
|         err!("Domain is blacklisted", domain) |         err!("Domain is blacklisted", domain) | ||||||
|     } |     } | ||||||
| @@ -605,6 +611,7 @@ fn download_icon(domain: &str) -> Result<Vec<u8>, Error> { | |||||||
|     let icon_result = get_icon_url(&domain)?; |     let icon_result = get_icon_url(&domain)?; | ||||||
|  |  | ||||||
|     let mut buffer = Vec::new(); |     let mut buffer = Vec::new(); | ||||||
|  |     let mut icon_type: Option<&str> = None; | ||||||
|  |  | ||||||
|     use data_url::DataUrl; |     use data_url::DataUrl; | ||||||
|  |  | ||||||
| @@ -616,17 +623,31 @@ fn download_icon(domain: &str) -> Result<Vec<u8>, Error> { | |||||||
|                 Ok((body, _fragment)) => { |                 Ok((body, _fragment)) => { | ||||||
|                     // Also check if the size is atleast 67 bytes, which seems to be the smallest png i could create |                     // Also check if the size is atleast 67 bytes, which seems to be the smallest png i could create | ||||||
|                     if body.len() >= 67 { |                     if body.len() >= 67 { | ||||||
|  |                         // Check if the icon type is allowed, else try an icon from the list. | ||||||
|  |                         icon_type = get_icon_type(&body); | ||||||
|  |                         if icon_type.is_none() { | ||||||
|  |                             debug!("Icon from {} data:image uri, is not a valid image type", domain); | ||||||
|  |                             continue; | ||||||
|  |                         } | ||||||
|  |                         info!("Extracted icon from data:image uri for {}", domain); | ||||||
|                         buffer = body; |                         buffer = body; | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 _ => warn!("data uri is invalid"), |                 _ => warn!("Extracted icon from data:image uri is invalid"), | ||||||
|             }; |             }; | ||||||
|         } else { |         } else { | ||||||
|             match get_page_with_cookies(&icon.href, &icon_result.cookies, &icon_result.referer) { |             match get_page_with_cookies(&icon.href, &icon_result.cookies, &icon_result.referer) { | ||||||
|                 Ok(mut res) => { |                 Ok(mut res) => { | ||||||
|                     info!("Downloaded icon from {}", icon.href); |  | ||||||
|                     res.copy_to(&mut buffer)?; |                     res.copy_to(&mut buffer)?; | ||||||
|  |                     // Check if the icon type is allowed, else try an icon from the list. | ||||||
|  |                     icon_type = get_icon_type(&buffer); | ||||||
|  |                     if icon_type.is_none() { | ||||||
|  |                         buffer.clear(); | ||||||
|  |                         debug!("Icon from {}, is not a valid image type", icon.href); | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     info!("Downloaded icon from {}", icon.href); | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
|                 _ => warn!("Download failed for {}", icon.href), |                 _ => warn!("Download failed for {}", icon.href), | ||||||
| @@ -635,10 +656,10 @@ fn download_icon(domain: &str) -> Result<Vec<u8>, Error> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     if buffer.is_empty() { |     if buffer.is_empty() { | ||||||
|         err!("Empty response") |         err!("Empty response downloading icon") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Ok(buffer) |     Ok((buffer, icon_type)) | ||||||
| } | } | ||||||
|  |  | ||||||
| fn save_icon(path: &str, icon: &[u8]) { | fn save_icon(path: &str, icon: &[u8]) { | ||||||
| @@ -650,7 +671,18 @@ fn save_icon(path: &str, icon: &[u8]) { | |||||||
|             create_dir_all(&CONFIG.icon_cache_folder()).expect("Error creating icon cache"); |             create_dir_all(&CONFIG.icon_cache_folder()).expect("Error creating icon cache"); | ||||||
|         } |         } | ||||||
|         Err(e) => { |         Err(e) => { | ||||||
|             info!("Icon save error: {:?}", e); |             warn!("Icon save error: {:?}", e); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | fn get_icon_type(bytes: &[u8]) -> Option<&'static str> { | ||||||
|  |     match bytes { | ||||||
|  |         [137, 80, 78, 71, ..] => Some("png"), | ||||||
|  |         [0, 0, 1, 0, ..] => Some("x-icon"), | ||||||
|  |         [82, 73, 70, 70, ..] => Some("webp"), | ||||||
|  |         [255, 216, 255, ..] => Some("jpeg"), | ||||||
|  |         [66, 77, ..] => Some("bmp"), | ||||||
|  |         _ => None | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user