2 stable releases

1.1.0 Aug 25, 2024
1.0.0 Apr 2, 2024

#207 in HTTP client

Download history 22/week @ 2024-09-25 34/week @ 2024-10-02 16/week @ 2024-10-09 11/week @ 2024-10-16 12/week @ 2024-10-23 23/week @ 2024-10-30 14/week @ 2024-11-06 26/week @ 2024-11-13 9/week @ 2024-11-20 4/week @ 2024-11-27 4/week @ 2024-12-04 8/week @ 2024-12-11 15/week @ 2024-12-18 32/week @ 2025-01-08

51 downloads per month

Custom license

30KB
207 lines

favicon-picker

Crates.io Crates.io (recent)

This is just a small library that allows you to get websites favicons.

This library is similar to site_icons but i just wanted something simple and flexible to use for various side-project.

Usage

Just add favicon-picker into your cargo dependencies or run:

cargo add favicon-picker

Example

Simple

use favicon_picker::get_favicons_from_url;
use reqwest::{header::{HeaderMap, HeaderValue, USER_AGENT}, Client};
use url::Url;

#[tokio::main(flavor = "current_thread")]
async fn main() -> anyhow::Result<()> {
    let client = Client::builder().default_headers({
        // the `User-Agent` is not required here
        let mut headers = HeaderMap::new();
        headers.append(USER_AGENT, HeaderValue::from_str("favicon-picker/1.0.0")?);
        headers
    }).build()?;
    let base_url = Url::parse("https://comic-walker.com/")?;
    for favs in get_favicons_from_url(&client, &base_url).await.map_err(|e| anyhow::Error::msg(e.to_string()))? {
        // I just printed the value to the standard output
        println!("{:#?}", favs);
    }
    Ok(())
}

Download

use std::{
    fs::{create_dir_all, File},
    io::{BufWriter, Write},
    path::Path,
};

use favicon_picker::get_favicons_from_url;
use reqwest::{
    header::{HeaderMap, HeaderValue, USER_AGENT},
    Client,
};
use url::Url;

#[tokio::main(flavor = "current_thread")]
async fn main() -> anyhow::Result<()> {
    let dowload_path = Path::new("output");
    create_dir_all(dowload_path)?;
    let client = Client::builder()
        .default_headers({
            // the `User-Agent` is not required here
            let mut headers = HeaderMap::new();
            headers.append(USER_AGENT, HeaderValue::from_str("favicon-picker/1.0.0")?);
            headers
        })
        .build()?;
    let base_url = Url::parse("https://comic-walker.com/")?;
    /// get existing favicons on the page
    for favs in get_favicons_from_url(&client, &base_url)
        .await
        .map_err(|e| anyhow::Error::msg(e.to_string()))?
    {
        // i just need the favicon file name here
        let file_name = favs
            .href
            .path_segments()
            .ok_or_else(|| anyhow::Error::msg("cannot be base"))?
            .last()
            .ok_or_else(|| anyhow::Error::msg("can't get the last url fragments"))?;
        let filepath = dowload_path.join(file_name);
        let data = favs.get_image_bytes(&client).await?;
        let mut file = BufWriter::new(File::create(filepath)?);
        file.write_all(&data)?;
        file.flush()?;
    }
    Ok(())
}

How does it work ??

When you call the favicon_picker::get_favicons_from_url, it will just fetch, and parse the existing HTML page. Then it will lookup to existing <link rel='icon'/> and that's it.

If it finds no favicon, it will give you the default favicon url which is http://<some-website-domain>/favicon.ico

Feature flags

  • non_exhaustive: Will make the Favicon struct #[non_exhaustive] Enabled by default
  • serde: Allows the Favicon struct to be serialized and deserialized with the serde framework
  • blocking: Allows you to use the reqwest::blocking::Client. Useful if you don't want to use async/await

Contribution

All pull request are welcome!!

License

GNU GENERAL PUBLIC LICENSE version 3

Dependencies

~6–18MB
~242K SLoC