discourse fetch: add rate limiting (#1321)

This implements rate limiting to the Discourse fetch logic, so that we
can actually load nontrivial servers without getting a 529 failure.

We could have used retry; I thought it was more polite to actually limit
the rate at which we make requests. However, to avoid seeing 529s in
practice, I left a bit of a buffer: we make only 55 requests per minute,
although 60 would be allowed.

If we want to improve Discourse loading time, we could boost up to the
full 60 request/min, but add in retries. (Or we could switch to retries
entirely.)

Test plan: This logic is untested, however my full discourse-plugin
branch uses it to do full Discourse loads without issue.
This commit is contained in:
Dandelion Mané 2019-08-23 17:51:35 +02:00 committed by GitHub
parent 08408a9706
commit 012f19eb48
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 36 additions and 5 deletions

View File

@ -6,6 +6,7 @@
"aphrodite": "^2.1.0", "aphrodite": "^2.1.0",
"base64url": "^3.0.1", "base64url": "^3.0.1",
"better-sqlite3": "^5.4.0", "better-sqlite3": "^5.4.0",
"bottleneck": "^2.19.5",
"chalk": "2.4.2", "chalk": "2.4.2",
"commonmark": "^0.29.0", "commonmark": "^0.29.0",
"d3-array": "^2.2.0", "d3-array": "^2.2.0",

View File

@ -12,6 +12,8 @@
import stringify from "json-stable-stringify"; import stringify from "json-stable-stringify";
import fetch from "isomorphic-fetch"; import fetch from "isomorphic-fetch";
import Bottleneck from "bottleneck";
import * as NullUtil from "../../util/null";
export type UserId = number; export type UserId = number;
export type PostId = number; export type PostId = number;
@ -90,17 +92,40 @@ export interface Discourse {
likesByUser(targetUsername: string, offset: number): Promise<LikeAction[]>; likesByUser(targetUsername: string, offset: number): Promise<LikeAction[]>;
} }
const MAX_API_REQUESTS_PER_MINUTE = 55;
export class Fetcher implements Discourse { export class Fetcher implements Discourse {
// We limit the rate of API requests, as documented here:
// https://meta.discourse.org/t/global-rate-limits-and-throttling-in-discourse/78612
// Note this limit is for admin API keys. If we change to user user API keys
// (would be convenient as the keys would be less sensitive), we will need to lower
// this rate limit by a factor of 3
// TODO: I've set the max requests per minute to 55 (below the stated limit
// of 60) to be a bit conservative, and avoid getting limited by the server.
// We could improve our throughput by increasing the requests per minute to the
// stated limit, and incorporating retry logic to account for the occasional 529.
+options: DiscourseFetchOptions; +options: DiscourseFetchOptions;
+_fetchImplementation: typeof fetch; +_fetchImplementation: typeof fetch;
constructor( constructor(
options: DiscourseFetchOptions, options: DiscourseFetchOptions,
// fetchImplementation shouldn't be provided by clients, but is convenient for testing. // fetchImplementation shouldn't be provided by clients, but is convenient for testing.
fetchImplementation?: typeof fetch fetchImplementation?: typeof fetch,
// Used to avoid going over the Discourse API rate limit
minTimeMs?: number
) { ) {
this.options = options; this.options = options;
this._fetchImplementation = fetchImplementation || fetch; const minTime = NullUtil.orElse(
minTimeMs,
(1000 * 60) / MAX_API_REQUESTS_PER_MINUTE
);
// n.b. the rate limiting isn't programmatically tested. However, it's easy
// to tell when it's broken: try to load a nontrivial Discourse server, and see
// if you get a 429 failure.
const limiter = new Bottleneck({minTime});
const unlimitedFetch = NullUtil.orElse(fetchImplementation, fetch);
this._fetchImplementation = limiter.wrap(unlimitedFetch);
} }
_fetch(endpoint: string): Promise<Response> { _fetch(endpoint: string): Promise<Response> {

View File

@ -27,7 +27,7 @@ describe("plugins/discourse/fetch", () => {
throw new Error(`couldn't load snapshot for ${file}`); throw new Error(`couldn't load snapshot for ${file}`);
} }
} }
const snapshotFetcher = () => new Fetcher(options, snapshotFetch); const snapshotFetcher = () => new Fetcher(options, snapshotFetch, 0);
it("loads LatestTopicId from snapshot", async () => { it("loads LatestTopicId from snapshot", async () => {
const topicId = await snapshotFetcher().latestTopicId(); const topicId = await snapshotFetcher().latestTopicId();
@ -55,7 +55,7 @@ describe("plugins/discourse/fetch", () => {
return Promise.resolve(resp); return Promise.resolve(resp);
}; };
const fetcherWithStatus = (status: number) => const fetcherWithStatus = (status: number) =>
new Fetcher(options, fakeFetch(status)); new Fetcher(options, fakeFetch(status), 0);
function expectError(name, f, status) { function expectError(name, f, status) {
it(`${name} errors on ${String(status)}`, () => { it(`${name} errors on ${String(status)}`, () => {
const fetcher = fetcherWithStatus(status); const fetcher = fetcherWithStatus(status);
@ -96,7 +96,7 @@ describe("plugins/discourse/fetch", () => {
fetchOptions = _options; fetchOptions = _options;
return Promise.resolve(new Response("", {status: 404})); return Promise.resolve(new Response("", {status: 404}));
}; };
await new Fetcher(options, fakeFetch).post(1337); await new Fetcher(options, fakeFetch, 0).post(1337);
if (fetchOptions == null) { if (fetchOptions == null) {
throw new Error("fetchOptions == null"); throw new Error("fetchOptions == null");
} }

View File

@ -1799,6 +1799,11 @@ boolbase@~1.0.0:
resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e" resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e"
integrity sha1-aN/1++YMUes3cl6p4+0xDcwed24= integrity sha1-aN/1++YMUes3cl6p4+0xDcwed24=
bottleneck@^2.19.5:
version "2.19.5"
resolved "https://registry.yarnpkg.com/bottleneck/-/bottleneck-2.19.5.tgz#5df0b90f59fd47656ebe63c78a98419205cadd91"
integrity sha512-VHiNCbI1lKdl44tGrhNfU3lup0Tj/ZBMJB5/2ZbNXRCPuRCO7ed2mgcK4r17y+KB2EfuYuRaVlwNbAeaWGSpbw==
brace-expansion@^1.0.0, brace-expansion@^1.1.7: brace-expansion@^1.0.0, brace-expansion@^1.1.7:
version "1.1.11" version "1.1.11"
resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"