-
-
Notifications
You must be signed in to change notification settings - Fork 44
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
Support retriable Chelonia actions #1768
Changes from 6 commits
c45ba09
421c340
21131ee
637e5a6
d357522
6192b3a
4ca17b0
9d14577
4222f8f
3b57a3c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,219 @@ | ||
'use strict' | ||
|
||
import sbp from '@sbp/sbp' | ||
import '@sbp/okturtles.events' | ||
import { PERSISTENT_ACTION_FAILURE, PERSISTENT_ACTION_SUCCESS, PERSISTENT_ACTION_TOTAL_FAILURE } from './events.js' | ||
|
||
type SbpInvocation = any[] | ||
type UUIDV4 = string | ||
|
||
type PersistentActionOptions = { | ||
errorInvocation?: SbpInvocation, | ||
// Maximum number of tries, default: Infinity. | ||
maxAttempts: number, | ||
// How many seconds to wait between retries. | ||
retrySeconds: number, | ||
skipCondition?: SbpInvocation, | ||
totalFailureInvocation?: SbpInvocation | ||
} | ||
|
||
type PersistentActionStatus = {| | ||
attempting: boolean, | ||
failedAttemptsSoFar: number, | ||
lastError: string, | ||
nextRetry: string, | ||
resolved: boolean | ||
|} | ||
|
||
const defaultOptions = { | ||
maxAttempts: Number.POSITIVE_INFINITY, | ||
retrySeconds: 30 | ||
} | ||
const tag = '[chelonia.persistentActions]' | ||
|
||
class PersistentAction { | ||
id: UUIDV4 | ||
invocation: SbpInvocation | ||
options: PersistentActionOptions | ||
status: PersistentActionStatus | ||
timer: TimeoutID | void | ||
|
||
constructor (invocation: SbpInvocation, options: PersistentActionOptions = {}) { | ||
// $FlowFixMe: Cannot resolve name `crypto`. | ||
this.id = crypto.randomUUID() | ||
this.invocation = invocation | ||
this.options = { ...defaultOptions, ...options } | ||
this.status = { | ||
attempting: false, | ||
failedAttemptsSoFar: 0, | ||
lastError: '', | ||
nextRetry: '', | ||
resolved: false | ||
} | ||
} | ||
|
||
async attempt (): Promise<void> { | ||
// Bail out if the action is already attempting or resolved. | ||
if (this.status.attempting || this.status.resolved) return | ||
if (await this.trySBP(this.options.skipCondition)) { | ||
this.cancel() | ||
return | ||
} | ||
try { | ||
this.status.attempting = true | ||
const result = await sbp(...this.invocation) | ||
this.status.attempting = false | ||
this.handleSuccess(result) | ||
} catch (error) { | ||
this.status.attempting = false | ||
this.handleError(error) | ||
} | ||
} | ||
|
||
cancel (): void { | ||
this.timer && clearTimeout(this.timer) | ||
this.status.nextRetry = '' | ||
this.status.resolved = true | ||
} | ||
|
||
async handleError (error: Error): Promise<void> { | ||
const { id, options, status } = this | ||
// Update relevant status fields before calling any optional code. | ||
status.failedAttemptsSoFar++ | ||
status.lastError = error.message | ||
const anyAttemptLeft = options.maxAttempts > status.failedAttemptsSoFar | ||
if (!anyAttemptLeft) status.resolved = true | ||
status.nextRetry = anyAttemptLeft && !status.resolved | ||
? new Date(Date.now() + options.retrySeconds * 1e3).toISOString() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Using The second issue is the, like for the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The
|
||
: '' | ||
// Perform any optional SBP invocation. | ||
sbp('okTurtles.events/emit', PERSISTENT_ACTION_FAILURE, { error, id }) | ||
await this.trySBP(options.errorInvocation) | ||
if (!anyAttemptLeft) { | ||
sbp('okTurtles.events/emit', PERSISTENT_ACTION_TOTAL_FAILURE, { error, id }) | ||
await this.trySBP(options.totalFailureInvocation) | ||
} | ||
// Schedule a retry if appropriate. | ||
if (status.nextRetry) { | ||
// Note: there should be no older active timeout to clear. | ||
this.timer = setTimeout(() => this.attempt(), this.options.retrySeconds * 1e3) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 1e3 seems like a pretty low value There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ditto |
||
} | ||
} | ||
|
||
handleSuccess (result: any): void { | ||
const { id, status } = this | ||
status.lastError = '' | ||
status.nextRetry = '' | ||
status.resolved = true | ||
sbp('okTurtles.events/emit', PERSISTENT_ACTION_SUCCESS, { id, result }) | ||
} | ||
|
||
async trySBP (invocation: SbpInvocation | void): Promise<any> { | ||
try { | ||
return invocation ? await sbp(...invocation) : undefined | ||
} catch (error) { | ||
console.error(tag, error.message) | ||
} | ||
} | ||
} | ||
|
||
// SBP API | ||
|
||
sbp('sbp/selectors/register', { | ||
'chelonia.persistentActions/_init' (): void { | ||
this.actionsByID = Object.create(null) | ||
// Necessary for now as _init cannot be async. Becomes true when 'configure' has been called. | ||
this.ready = false | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I can't find the place to set There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Indeed, good catch. BTW I've since replaced it by a direct check for availability of |
||
sbp('okTurtles.events/on', PERSISTENT_ACTION_SUCCESS, ({ id }) => { | ||
sbp('chelonia.persistentActions/cancel', id) | ||
}) | ||
sbp('okTurtles.events/on', PERSISTENT_ACTION_TOTAL_FAILURE, ({ id }) => { | ||
sbp('chelonia.persistentActions/cancel', id) | ||
}) | ||
}, | ||
|
||
// Cancels a specific action by its ID. | ||
// The action won't be retried again, but an async action cannot be aborted if its promise is stil attempting. | ||
'chelonia.persistentActions/cancel' (id: UUIDV4): void { | ||
if (id in this.actionsByID) { | ||
this.actionsByID[id].cancel() | ||
delete this.actionsByID[id] | ||
// Likely no need to await this call. | ||
sbp('chelonia.persistentActions/save') | ||
} | ||
}, | ||
|
||
// TODO: validation | ||
'chelonia.persistentActions/configure' ({ databaseKey, options = {} }: { databaseKey: string; options: Object }): void { | ||
if (!databaseKey) throw new TypeError(`${tag} 'databaseKey' is required`) | ||
this.databaseKey = databaseKey | ||
for (const key in options) { | ||
if (key in defaultOptions) { | ||
defaultOptions[key] = options[key] | ||
} else { | ||
throw new TypeError(`${tag} Unknown option: ${key}`) | ||
taoeffect marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
} | ||
}, | ||
|
||
'chelonia.persistentActions/enqueue' (...args): UUIDV4[] { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think it's preferable to declare possible types of
I think the second format isn't necessary. And also There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Good point about the
Here I just followed the issue requirements.
An invocation is required in any case, so I don't think There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Oh, it was a good thing @Silver-IT brought this up, because I missed this in my review. This is not what the args look like according to the issue:
Rather, they look like this:
So to answer @Silver-IT's question: the reason for allowing both types is because it makes the syntax when enqueing invocations simpler. In most cases you can just pass in the sbpInvocation like so: But when you want to adjust the defaults for a specific invocation, you use the second syntax.
EDIT: oops, my mistake, the code seems to correctly handle this. |
||
if (!this.ready) throw new Error(`${tag} Not ready yet.`) | ||
const ids: UUIDV4[] = [] | ||
for (const arg of args) { | ||
const action = Array.isArray(arg) | ||
? new PersistentAction(arg) | ||
: new PersistentAction(arg.invocation, arg) | ||
this.actionsByID[action.id] = action | ||
ids.push(action.id) | ||
} | ||
// Likely no need to await this call. | ||
sbp('chelonia.persistentActions/save') | ||
for (const id of ids) this.actionsByID[id].attempt() | ||
return ids | ||
}, | ||
|
||
// Forces retrying an existing persisted action given its ID. | ||
// Note: 'failedAttemptsSoFar' will still be increased upon failure. | ||
'chelonia.persistentActions/forceRetry' (id: UUIDV4): void { | ||
if (id in this.actionsByID) { | ||
this.actionsByID[id].attempt() | ||
} | ||
}, | ||
|
||
// Loads and tries every stored persistent action under the configured database key. | ||
async 'chelonia.persistentActions/load' (): Promise<void> { | ||
const storedActions = JSON.parse((await sbp('chelonia/db/get', this.databaseKey)) ?? '[]') | ||
for (const { id, invocation, options } of storedActions) { | ||
this.actionsByID[id] = new PersistentAction(invocation, options) | ||
// Use the stored ID instead of the autogenerated one. | ||
// TODO: find a cleaner alternative. | ||
this.actionsByID[id].id = id | ||
Comment on lines
+198
to
+199
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why not to add There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That's a possible choice, but I felt like it was not that great either |
||
} | ||
this.ready = true | ||
sbp('chelonia.persistentActions/retryAll') | ||
}, | ||
|
||
// Retry all existing persisted actions. | ||
// TODO: add some delay between actions so as not to spam the server, | ||
// or have a way to issue them all at once in a single network call. | ||
'chelonia.persistentActions/retryAll' (): void { | ||
for (const id in this.actionsByID) { | ||
sbp('chelonia.persistentActions/forceRetry', id) | ||
} | ||
}, | ||
|
||
// Updates the database version of the attempting action list. | ||
'chelonia.persistentActions/save' (): Promise<Error | void> { | ||
return sbp( | ||
'chelonia/db/set', | ||
this.databaseKey, | ||
JSON.stringify(Object.values(this.actionsByID)) | ||
) | ||
}, | ||
|
||
'chelonia.persistentActions/status' () { | ||
return Object.values(this.actionsByID) | ||
// $FlowFixMe: `PersistentAction` is incompatible with mixed | ||
.map((action: PersistentAction) => ({ id: action.id, invocation: action.invocation, ...action.status })) | ||
} | ||
}) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
!status.resolved
could be ignored, I think.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
status.resolved
can be set independently by.cancel()
, therefore checking foranyAttemptLeft
was not enough