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

Add basic rate limiter whitelist (specific IPs only) #573

Open
wants to merge 3 commits into
base: main
Choose a base branch
from

Conversation

zguig52
Copy link

@zguig52 zguig52 commented Nov 28, 2024

Types of changes

  • Bug fix (a non-breaking change which fixes an issue)
  • New feature (a non-breaking change which adds functionality)
  • Breaking change (fix or feature that would cause existing functionality to change)

Description

Partially resolves #517 (to be discussed further for more advances use cases)

Checklist:

  • My change requires a change to the documentation.
  • I have updated the documentation accordingly.
  • I have added tests to cover my changes (if not applicable, please state why)

@vejja
Copy link
Collaborator

vejja commented Nov 28, 2024

Hi @zguig52
Thanks for this
Do you think it would be possible to add basic tests to test/rateLimiter.test.ts to ensure that the module behaves as expected for some basic configuration values? (e.g. undefined, single ip, multiple ips, empty array, etc.)

@zguig52
Copy link
Author

zguig52 commented Nov 28, 2024

Hi @vejja ,
I previously added a manual tests page, sorry, I hadn't seen the autotests folder.
I have added 3 pages + associated conf and tests. They all pass when I do yarn test, but I don't know how to get tests results details...
I would have liked to check timing impacts of the whitelist array presence regarding to no whitelist at all.

@zguig52
Copy link
Author

zguig52 commented Nov 28, 2024

NB: for the undefined value test, it is already performed with existing tests as it is the default value configured

@vejja
Copy link
Collaborator

vejja commented Nov 28, 2024

Hi @zguig52
You can test the pre-release package from the Stackblitz Continuous Release artefact here:
npm i https://pkg.pr.new/nuxt-modules/security/nuxt-security@524b003
I don't think the timing impacts will be measurable but let us know !

NB: The bot was unable to inline the PR preview because your branch is outside of this repo but I triggered it manually.

@vejja
Copy link
Collaborator

vejja commented Nov 28, 2024

@zguig52 : Before we release this, I'd like to get @Baroshem's advice here because I'm not extremely knowledgeable about the Rate Limiter.

Do we agree that we are allowing ip whitelists here, as opposed to domain names ?
In other words, this is unlikely to be used in production, where the requesting IP is in general unknown.
If this assumption is correct, the main use case here is in development mode when we want to whitelist localhost by IP (e.g. 127.0.0.1 usually) ?

Let me know if I'm correct

@zguig52
Copy link
Author

zguig52 commented Nov 28, 2024

@vejja , this is an IPs CLIENT whitelist that is managed here.

Main usages are for production:

  • internal other components calls (where you could whitelist your internal network IPs)
  • performance / load tests (where you could whitelist your traffic injectors IPs)

If you look in the function associated to the rateLimiter, it uses client IP to determine and store all calls performed to the endpoint by a specific client and determine if rates are normal:

function getIP (event: H3Event) {
  const ip = getRequestIP(event, { xForwardedFor: true }) || ''
  return ip
}

This is a h3 function that provide client IP https://github.com/unjs/h3/blob/77510f3c27e6884489189b300c52b5355751aefc/src/utils/request.ts#L297

So I reused this information to compare it to the whitelist and skip rateLimit checks and storage if it matches.

@zguig52
Copy link
Author

zguig52 commented Nov 28, 2024

npm i https://pkg.pr.new/nuxt-modules/security/nuxt-security@524b003

I am deploying it in a qualification environment to check if it behaves as expected with public clients IPs

@vejja
Copy link
Collaborator

vejja commented Nov 28, 2024

@vejja , this is an IPs CLIENT whitelist that is managed here.

Main usages are for production:

  • internal other components calls (where you could whitelist your internal network IPs)
  • performance / load tests (where you could whitelist your traffic injectors IPs)

If you look in the function associated to the rateLimiter, it uses client IP to determine and store all calls performed to the endpoint by a specific client and determine if rates are normal:

function getIP (event: H3Event) {
  const ip = getRequestIP(event, { xForwardedFor: true }) || ''
  return ip
}

This is a h3 function that provide client IP https://github.com/unjs/h3/blob/77510f3c27e6884489189b300c52b5355751aefc/src/utils/request.ts#L297

So I reused this information to compare it to the whitelist and skip rateLimit checks and storage if it matches.

Oh oh I think I missed your whole point effectively !
I initially thought that you wanted to whitelist XHR requests to the API made by some IP-whitelisted frontend client.

But if I understand correctly, the use cases are significantly larger than what I thought

  • For people who deploy with SSR: whenever the SSR server prefetches data from an internal API route, avoid hitting rate limits. Assuming the SSR server is on a known internal IP, then we can whitelist it.
  • For larger systems where third-party tools (e.g. testing suites) are used to hit the internal API routes, avoid hitting rate limits. Assuming the 3rd-party suite is a known IP, then we can whitelist.

Am I right here ?

@zguig52
Copy link
Author

zguig52 commented Nov 28, 2024

@vejja , that's it! As I started to comment on initial discussion associated, the best would be to be allowed defining networks and not only specific IPs, but it is a bit more complicated to implement and require more thinking!
#517

@zguig52
Copy link
Author

zguig52 commented Nov 28, 2024

npm i https://pkg.pr.new/nuxt-modules/security/nuxt-security@524b003

Just did some tests with this release, it works as expected (at least for a single public IP configured, which is for the moment my tests conditions). I used siege tool to overflow server API routes and it doesn't trigger the rateLimit.

When I am behind the whitelisted public IP, no 429. When I use another public IP, it trigger 429!

@vejja
Copy link
Collaborator

vejja commented Nov 28, 2024

@vejja , that's it! As I started to comment on initial discussion associated, the best would be to be allowed defining networks and not only specific IPs, but it is a bit more complicated to implement and require more thinking! #517

ok clear saw #517 also - would implementation break the type definition of whitelist ?

ok clear
saw #517 also - would implementation break the type definition of whitelist ?

@vejja
Copy link
Collaborator

vejja commented Nov 28, 2024

@Baroshem : whenever you think this is ready - I believe this is a feature upgrade, shall we bump to minor v2.2 ?

@zguig52
Copy link
Author

zguig52 commented Nov 29, 2024

saw #517 also - would implementation break the type definition of whitelist ?

I have reviewed a bit my comments on this, as I think we might be able to add more capabilities with the same configuration format, without breaking this first implementation (or maybe we already change before release for basic input config format to be able to change it without breaking changes later).

Copy link
Collaborator

@Baroshem Baroshem left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks great!

Thanks @zguig52 !

@vejja let's plan it for 2.2 release :)

@vejja
Copy link
Collaborator

vejja commented Dec 30, 2024

Hi @Baroshem
I think @zguig52 wanted to modify the type definition of whitelist to allow more flexible definitions of addresses.
Shall we wait then in order to avoid breaking changes down the line ?

@zguig52
Copy link
Author

zguig52 commented Jan 8, 2025

Hi @vejja and @Baroshem,
From our discussions here #517, I had concluded we are able to keep a simple string array configuration format and then, at startup time, analyse and sort / process internally different kinds of IP/net/ranges definitions!
For me it's all good to go for a first release of this feature, when I will have time, I will work on extending it with more configuration capabilities.

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

Successfully merging this pull request may close these issues.

Rate limiter whitelist
3 participants