0
0
mirror of https://github.com/PostHog/posthog.git synced 2024-11-29 03:04:16 +01:00
posthog/plugin-server/tests/utils/token-bucket.test.ts
Tomás Farías Santana 1e94d8e138
feat(ingestion-slowlane): Re-route events in plugin-server on capacity exceeded (#14211)
* feat(ingestion-slowlane): Add token-bucket utility

* feat(ingestion-slowlane): Re-route overflow events

* fix: Import missing stringToBoolean

* fix(ingestion-slowlane): Flip around kafka topics according to mode

* refactor(ingestion-slowlane): Use dash instead of underscore in filename

* fix(ingestion-slowlane): Do not increase tokens beyond bucket capacity

* feat(ingestion-slowlane): Add ingestion-overflow mode/capability/consumer

* feat(ingestion-slowlane): Add ingestion warning for capacity overflow

* test(ingestion-slowlane): Add test for ingestion of overflow events

* fix(ingestion-slowlane): Rate limit warnings to 1 per hour

* test(ingestion-slowlane): Add a couple more tests for overflow re-route

* fix(slowlane-ingestion): Look at batch topic to determine message topic

* refactor(slowlane-ingestion): Use refactored consumer model

* fix(slowlane-ingestion): Undo topic requirement in eachMessageIngestion

* refactor(slowlane-ingestion): Only produce events if ingestionOverflow is also enabled

* refactor(slowlane-ingestion): Use an env variable to determine if ingestionOverflow is enabled

* chore(slowlane-ingestion): Add a comment explaining env variable
2023-02-16 14:30:13 +01:00

156 lines
5.4 KiB
TypeScript

import { BucketKeyMissingError, Limiter, Storage } from '../../src/utils/token-bucket'
describe('Storage', () => {
describe('replenish()', () => {
afterEach(() => {
jest.useRealTimers()
})
it('adds capacity to new key', () => {
const key = 'test'
const storage = new Storage(10, 1)
const now = new Date('2023-02-08T08:00:00')
jest.useFakeTimers().setSystemTime(now)
storage.replenish(key)
expect(storage.buckets.has(key)).toEqual(true)
expect(storage.buckets.get(key)![0]).toEqual(10)
expect(storage.buckets.get(key)![1]).toEqual(now.valueOf())
})
it('adds rate to existing key', () => {
const key = 'test'
const storage = new Storage(10, 1)
const now = new Date('2023-02-08T08:00:00')
jest.useFakeTimers().setSystemTime(now)
storage.replenish(key)
storage.consume(key, 5)
now.setSeconds(now.getSeconds() + 2)
jest.setSystemTime(now)
storage.replenish(key)
expect(storage.buckets.has(key)).toEqual(true)
expect(storage.buckets.get(key)![0]).toEqual(7)
expect(storage.buckets.get(key)![1]).toEqual(now.valueOf())
})
it('adds rate to existing key with argument now', () => {
const key = 'test'
const storage = new Storage(10, 1)
const now = Date.now()
storage.replenish(key, now)
storage.consume(key, 9)
storage.replenish(key, now + 2000)
expect(storage.buckets.has(key)).toEqual(true)
expect(storage.buckets.get(key)![0]).toEqual(3)
expect(storage.buckets.get(key)![1]).toEqual(now + 2000)
})
it('does not add more than capacity tokens', () => {
const key = 'test'
const storage = new Storage(10, 1)
const now = new Date('2023-02-08T08:00:00')
storage.replenish(key, now.valueOf())
expect(storage.buckets.get(key)![0]).toEqual(10)
expect(storage.buckets.get(key)![1]).toEqual(now.valueOf())
// 20 seconds would exceed capacity of 10 tokens at 1 token/sec.
storage.replenish(key, now.valueOf() + 20000)
expect(storage.buckets.has(key)).toEqual(true)
expect(storage.buckets.get(key)![0]).toEqual(10)
expect(storage.buckets.get(key)![1]).toEqual(now.valueOf() + 20000)
})
})
describe('consume()', () => {
it('consumes when tokens are less than capacity', () => {
const key = 'test'
const storage = new Storage(10, 1)
storage.replenish(key)
expect(storage.consume(key, 9)).toEqual(true)
expect(storage.buckets.get(key)![0]).toEqual(1)
})
it('rejects when tokens are more than capacity', () => {
const key = 'test'
const storage = new Storage(10, 1)
storage.replenish(key)
expect(storage.consume(key, 11)).toEqual(false)
expect(storage.buckets.get(key)![0]).toEqual(10)
})
it('throws error on missing bucket key', () => {
const key = 'test'
const storage = new Storage(10, 1)
expect(storage.buckets.has(key)).toEqual(false)
expect(() => storage.consume(key, 1)).toThrow(BucketKeyMissingError)
})
})
})
describe('Limiter', () => {
describe('consume()', () => {
afterEach(() => {
jest.useRealTimers()
})
it('consumes when tokens available', () => {
const key = 'test'
const limiter = new Limiter(10, 2)
const now = new Date('2023-02-08T08:00:00')
jest.useFakeTimers().setSystemTime(now)
expect(limiter.consume(key, 9)).toEqual(true)
})
it('rejects when tokens run out', () => {
const key = 'test'
const limiter = new Limiter(10, 2)
const now = new Date('2023-02-08T08:00:00')
jest.useFakeTimers().setSystemTime(now)
expect(limiter.consume(key, 10)).toEqual(true)
// We are not advancing time, so no tokens should have been replenished
expect(limiter.consume(key, 1)).toEqual(false)
})
it('consumes when tokens have been replenished', () => {
const key = 'test'
const limiter = new Limiter(10, 2)
const now = new Date('2023-02-08T08:00:00')
jest.useFakeTimers().setSystemTime(now)
limiter.consume(key, 10)
expect(limiter.consume(key, 1)).toEqual(false)
jest.setSystemTime(now.valueOf() + 1000)
// Now that we have advanced 1 second, we can consume 1 token
expect(limiter.consume(key, 1)).toEqual(true)
})
it('consumes when tokens have been replenished with now argument', () => {
const key = 'test'
const limiter = new Limiter(10, 2)
const now = new Date('2023-02-08T08:00:00')
jest.useFakeTimers().setSystemTime(now)
limiter.consume(key, 10, now.valueOf())
expect(limiter.consume(key, 1)).toEqual(false)
// Even though we are not advancing time, we are passing the time to use with now
expect(limiter.consume(key, 1, now.valueOf() + 1000)).toEqual(true)
})
})
})