Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Proposal: Make IpInfo generic over localization provider. #47

Closed
Alextopher opened this issue Dec 22, 2023 · 4 comments
Closed

Proposal: Make IpInfo generic over localization provider. #47

Alextopher opened this issue Dec 22, 2023 · 4 comments

Comments

@Alextopher
Copy link
Contributor

Alextopher commented Dec 22, 2023

Proposed Change

Create a new abstraction for "localization providers" and make IpInfo generic over that type. If the spirit of this change is accepted we can also begin to improve the status-quo regarding excessive use of .clone() in IpInfo. This could look like making the following change (I'm working through experiments at https://github.com/Alextopher/ipinfo/tree/memory)

pub trait Localization {
    fn country_name(&self, country_code: &str) -> Option<&str>;
    fn is_eu(&self, country_code: &str) -> bool;
    fn flag(&self, country_code: &str) -> Option<&CountryFlag>;
    fn currency(&self, country_code: &str) -> Option<&CountryCurrency>;
    fn continent(&self, country_code: &str) -> Option<&Continent>;
}

// Where `StaticTables` is a crate provided Localization. It is a good default and improvement on the status-quo 
// See: https://github.com/Alextopher/ipinfo/blob/memory/src/localization.rs#L78
pub struct IpInfo<T: Localization = StaticTables> {
    url: String,
    token: Option<String>,
    client: reqwest::Client,
    cache: LruCache<String, IpDetails>,
    localization: T,
}

Motivation

There are some key API guidelines that inspires this kind of change.

Using a trait and generics here minimizes our assumptions on how localization data can be provided. #42 called for the replacement of remote resources with tables baked into the library. If IpInfo was generic then we could support both workflows. C-GENERIC

IpInfo only uses shared references to localization data. Because of this we should only require users to provide shared references. Then we can empower users to handle memory management themselves. C-CALLER-CONTROL

Downsides

This proposal is suggesting making a breaking change. Using a good default type parameter (StaticTables) means anyone who hasn't used localization (and just used ..Default::default() won't have a trouble updating.

@max-ipinfo
Copy link
Contributor

Thanks for the suggestion @Alextopher. I'll talk to the team about this proposal and see if this is something that we want to add.

@talhahwahla
Copy link
Contributor

Taking up from here @max-ipinfo.

@Alextopher thanks again for all your contributions! My reading here, as you also mention, is that this will be a breaking change for users doing custom localization. Secondly, I suspect this will add extra work for users to achieve the same end. So for these two reasons I doubt that it will be worthwhile for users to migrate.

But nevertheless we can evaluate how big of an improvement we can achieve in performance. Do you have any before/after numbers around this?

@Alextopher
Copy link
Contributor Author

I don't have numbers around anymore. Nonetheless they are very large strings and hashmaps that need to be cloned every time you create an IpInfo client.

@talhahwahla
Copy link
Contributor

Agree with you on the cloning overhead and would have eagerly moved forward with your proposal if we could somehow ensure backwards compatibility for all users. I think we can close this issue for now -- we will keep this idea in mind and revisit it if more users express interest.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

3 participants