Page MenuHomePhabricator

[native] implement method to run asynchronous cancellable task in CommCoreModule and return error via promise
ClosedPublic

Authored by kamil on Nov 9 2022, 3:07 AM.
Tags
None
Referenced Files
Unknown Object (File)
Thu, Dec 26, 5:40 PM
Unknown Object (File)
Mon, Dec 16, 9:00 PM
Unknown Object (File)
Mon, Dec 16, 9:00 PM
Unknown Object (File)
Mon, Dec 16, 9:00 PM
Unknown Object (File)
Mon, Dec 16, 8:59 PM
Unknown Object (File)
Mon, Dec 16, 8:59 PM
Unknown Object (File)
Mon, Dec 16, 8:59 PM
Unknown Object (File)
Mon, Dec 16, 8:59 PM

Details

Summary

context: ENG-2139

Introduced scheduleOrRunCancellable which will allow scheduling a task from the asynchronous call that could be canceled while the atomic flag in database singleton tasksCancelled will be set to true.

In this scenario an error could not be thrown because it will kill the database thread, it has to be returned via promise and handled in js.

Using the error message as constant TASK_CANCELLED not a meaningful string because later in javascript we will need to indicate an error only by detecting if its message is equal to a given string or contain a given string.

Test Plan
  1. Build app for both iOS and Android
  2. Add changes with CommCoreModule methods which will simplify testing: D5582
  3. Run this code to check if this works as supposed to:
try {
  await commCoreModule.printMessageAndWaitAsync('async task 1');
  await commCoreModule.printMessageAndWaitAsync('async task 2');
} catch (e) {
  console.log('caught: ', e);
}

try {
  await commCoreModule.cancelTasks();
} catch (e) {
  console.log('caught: ', e);
}

try {
  await commCoreModule.printMessageAndWaitAsync('async task 3');
  await commCoreModule.printMessageAndWaitAsync('async task 4');
} catch (e) {
  console.log('caught: ', e);
}

try {
  await commCoreModule.runTasks();
} catch (e) {
  console.log('caught: ', e);
}

try {
  await commCoreModule.printMessageAndWaitAsync('async task 5');
} catch (e) {
  console.log('caught: ', e);
}

result:

2022-11-09 14:24:10.079178+0100 Comm[88845:5799175] COMM: Starting async task: async task 1
2022-11-09 14:24:10.084171+0100 Comm[88845:5799175] COMM: Ending async task: async task 1
2022-11-09 14:24:10.097871+0100 Comm[88845:5799175] COMM: Starting async task: async task 2
2022-11-09 14:24:10.102069+0100 Comm[88845:5799175] COMM: Ending async task: async task 2
2022-11-09 14:24:10.114216+0100 Comm[88845:5799137] [javascript] 'caught: ', { message: 'TASK_CANCELLED' }
2022-11-09 14:24:10.115071+0100 Comm[88845:5799175] COMM: Starting async task: async task 5
2022-11-09 14:24:10.117837+0100 Comm[88845:5799175] COMM: Ending async task: async task 5

which demonstrates that we can not add a new task when taskCancelled: true, additionally we can remove await while calling commCoreModule.printMessageAndWaitAsync which will result that almost all (depending if the task will manage to execute) will throw an error that task was canceled - it's a scenario when tasks are in the queue but are removed without executing.

Diff Detail

Repository
rCOMM Comm
Lint
Lint Not Applicable
Unit
Tests Not Applicable

Event Timeline

kamil held this revision as a draft.
kamil published this revision for review.Nov 9 2022, 5:31 AM

Are there any callsites we need to update from scheduleOrRun to scheduleOrRunCancellable?

native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
49 ↗(On Diff #18256)

Can we reject with a JSIError instead of a string?

57–65 ↗(On Diff #18256)

Early exit please

85 ↗(On Diff #18256)

I would define this next to the other overload... doesn't make sense for enableMultithreading to be in between, in my opinion

native/ios/Comm/GlobalDBSingleton.mm
46 ↗(On Diff #18256)

Shouldn't this be this->scheduleOrRunCancellableCommonImpl(task, promise, jsInvoker)?

native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
57–65 ↗(On Diff #18256)

Agree with @ashoat . The order here is confusing as in D5577

native/ios/Comm/GlobalDBSingleton.mm
46 ↗(On Diff #18256)

Explanation here is the same as in D5577

use scheduleOrRunCancellableCommonImpl, improve readabillity

native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
49 ↗(On Diff #18256)

Unfortunately, I don't think so, this method accepts only string: docs

85 ↗(On Diff #18256)

Agree, My oversight

native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
49 ↗(On Diff #18256)

Interesting!! So jsi::JSError can only be used in synchronous calls? Kind of strange, but okay

marcin added a reviewer: tomek.
marcin added inline comments.
native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
47 ↗(On Diff #18319)

Same as in D5577, this string is repeated so many times it should be extracted into some constant or static attribute.

tomek requested changes to this revision.Nov 14 2022, 9:42 AM

Can we reduce the duplication of logic?

native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
48 ↗(On Diff #18376)

I don't think it is correct to capture promise by copy. Shouldn't we do this by reference? It feels like settling a copy of a promise shouldn't affect the original one...

57 ↗(On Diff #18376)

Are we sure that we want to capture all these values by copy?

This revision now requires changes to proceed.Nov 14 2022, 9:42 AM
native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
48 ↗(On Diff #18376)

it's a pointer to promise so copying seems okay

tomek added inline comments.
native/cpp/CommonCpp/NativeModules/InternalModules/GlobalDBSingleton.h
49–51 ↗(On Diff #18455)

I don't think there's a reason behind using explicit captures in one lambda and default by copy in another. I think we should use consistent syntax to avoid confusion.

48 ↗(On Diff #18376)

You're totally right, sorry for the confusion!

(Removing myself as blocking since I don't think I have enough context to give this a proper review)

This revision is now accepted and ready to land.Nov 17 2022, 9:21 AM

rebase before landing, use explicit capture