0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-04-01 02:41:39 -05:00

🐛 Fixed bug causing sites with many snippets to be rate limited (#18568)

refs TryGhost/Product#4022

- This block of code converts any snippets that are only in mobiledoc to lexical locally in the editor, then sends a PUT request for each snippet to update it on the server
- For sites with > 50 snippets, these PUT requests were triggering rate limits on Pro of 50 req/s
- This change is a temp fix to add a 100ms sleep between these requests to stay under the 50 req/s limit
- Longer term, we may introduce a migration that will convert all snippets, or modify the editor to somehow lazily convert the snippets when needed, but this temp fix should resolve the rate limiting issue for the time being
This commit is contained in:
Chris Raible 2023-10-11 04:06:03 -07:00 committed by GitHub
parent e10144c3a6
commit 909c8b53a8
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -817,22 +817,34 @@ export default class LexicalEditorController extends Controller {
this.syncMobiledocSnippets();
}
sleep(ms) {
return new Promise((r) => {
setTimeout(r, ms);
});
}
@action
async syncMobiledocSnippets() {
const snippets = this.store.peekAll('snippet');
// very early in the beta we had a bug where lexical snippets were saved with double-encoded JSON
// we fix that here by re-saving any snippets that are still in that state
const snippetFixSaves = [];
snippets.forEach((snippet) => {
for (let i = 0; i < snippets.length; i++) {
const snippet = snippets.objectAt(i);
if (typeof snippet.lexical === 'string') {
try {
snippet.lexical = JSON.parse(snippet.lexical);
snippet.mobiledoc = {};
snippetFixSaves.push(snippet.save());
await snippet.save();
// Temp fix: Timeout for 100 ms between requests to avoid hitting rate limit (50req/s)
// refs https://github.com/TryGhost/Product/issues/4022
await this.sleep(100);
} catch (e) {
snippet.lexical = null;
snippetFixSaves.push(snippet.save());
await snippet.save();
// Temp fix: Timeout for 100 ms between requests to avoid hitting rate limit (50req/s)
// refs https://github.com/TryGhost/Product/issues/4022
await this.sleep(100);
console.error(e); // eslint-disable-line no-console
@ -845,10 +857,10 @@ export default class LexicalEditorController extends Controller {
}
}
}
});
await Promise.all(snippetFixSaves);
}
snippets.forEach((snippet) => {
for (let i = 0; i < snippets.length; i++) {
const snippet = snippets.objectAt(i);
if (!snippet.lexical || snippet.lexical.syncedAt && moment.utc(snippet.lexical.syncedAt).isBefore(snippet.updatedAtUTC)) {
const serializedLexical = mobiledocToLexical(JSON.stringify(snippet.mobiledoc));
@ -875,8 +887,12 @@ export default class LexicalEditorController extends Controller {
// kick off a background save, we already have .lexical updated which is what we need
snippet.save();
// Temp fix: Timeout for 100 ms between requests to avoid hitting rate limit (50req/s)
// refs https://github.com/TryGhost/Product/issues/4022
await this.sleep(100);
}
});
}
}
/* Public methods --------------------------------------------------------*/