Skip to content

Commit

Permalink
Allow bypass on ratelimit ip (bluesky-social#1613)
Browse files Browse the repository at this point in the history
allow bypass on ratelimit ip
  • Loading branch information
dholms authored Sep 15, 2023
1 parent 7f3d95b commit b8f4dd9
Show file tree
Hide file tree
Showing 3 changed files with 27 additions and 4 deletions.
15 changes: 15 additions & 0 deletions packages/pds/src/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ export interface ServerConfigValues {

rateLimitsEnabled: boolean
rateLimitBypassKey?: string
rateLimitBypassIps?: string[]
redisScratchAddress?: string
redisScratchPassword?: string

Expand Down Expand Up @@ -163,6 +164,15 @@ export class ServerConfig {

const rateLimitsEnabled = process.env.RATE_LIMITS_ENABLED === 'true'
const rateLimitBypassKey = nonemptyString(process.env.RATE_LIMIT_BYPASS_KEY)
const rateLimitBypassIpsStr = nonemptyString(
process.env.RATE_LIMIT_BYPASS_IPS,
)
const rateLimitBypassIps = rateLimitBypassIpsStr
? rateLimitBypassIpsStr.split(',').map((ipOrCidr) => {
const ip = ipOrCidr.split('/')[0]
return ip.trim()
})
: undefined
const redisScratchAddress = nonemptyString(
process.env.REDIS_SCRATCH_ADDRESS,
)
Expand Down Expand Up @@ -266,6 +276,7 @@ export class ServerConfig {
blobCacheLocation,
rateLimitsEnabled,
rateLimitBypassKey,
rateLimitBypassIps,
redisScratchAddress,
redisScratchPassword,
appUrlPasswordReset,
Expand Down Expand Up @@ -454,6 +465,10 @@ export class ServerConfig {
return this.cfg.rateLimitBypassKey
}

get rateLimitBypassIps() {
return this.cfg.rateLimitBypassIps
}

get redisScratchAddress() {
return this.cfg.redisScratchAddress
}
Expand Down
2 changes: 2 additions & 0 deletions packages/pds/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -268,12 +268,14 @@ export class PDS {
rlCreator = (opts: RateLimiterOpts) =>
RateLimiter.redis(redisScratch, {
bypassSecret: config.rateLimitBypassKey,
bypassIps: config.rateLimitBypassIps,
...opts,
})
} else {
rlCreator = (opts: RateLimiterOpts) =>
RateLimiter.memory({
bypassSecret: config.rateLimitBypassKey,
bypassIps: config.rateLimitBypassIps,
...opts,
})
}
Expand Down
14 changes: 10 additions & 4 deletions packages/xrpc-server/src/rate-limiter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,21 +20,24 @@ export type RateLimiterOpts = {
durationMs: number
points: number
bypassSecret?: string
bypassIps?: string[]
calcKey?: CalcKeyFn
calcPoints?: CalcPointsFn
failClosed?: boolean
}

export class RateLimiter implements RateLimiterI {
public limiter: RateLimiterAbstract
private byPassSecret?: string
private bypassSecret?: string
private bypassIps?: string[]
private failClosed?: boolean
public calcKey: CalcKeyFn
public calcPoints: CalcPointsFn

constructor(limiter: RateLimiterAbstract, opts: RateLimiterOpts) {
this.limiter = limiter
this.byPassSecret = opts.bypassSecret
this.bypassSecret = opts.bypassSecret
this.bypassIps = opts.bypassIps
this.calcKey = opts.calcKey ?? defaultKey
this.calcPoints = opts.calcPoints ?? defaultPoints
}
Expand Down Expand Up @@ -63,11 +66,14 @@ export class RateLimiter implements RateLimiterI {
opts?: { calcKey?: CalcKeyFn; calcPoints?: CalcPointsFn },
): Promise<RateLimiterStatus | RateLimitExceededError | null> {
if (
this.byPassSecret &&
ctx.req.header('x-ratelimit-bypass') === this.byPassSecret
this.bypassSecret &&
ctx.req.header('x-ratelimit-bypass') === this.bypassSecret
) {
return null
}
if (this.bypassIps && this.bypassIps.includes(ctx.req.ip)) {
return null
}
const key = opts?.calcKey ? opts.calcKey(ctx) : this.calcKey(ctx)
const points = opts?.calcPoints
? opts.calcPoints(ctx)
Expand Down

0 comments on commit b8f4dd9

Please sign in to comment.