PromiseTests.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. using Jint.Native;
  2. using Jint.Native.Object;
  3. using Jint.Native.Promise;
  4. using Jint.Runtime;
  5. // obsolete GetCompletionValue
  6. #pragma warning disable 618
  7. namespace Jint.Tests.Runtime;
  8. public class PromiseTests
  9. {
  10. [Fact]
  11. public void RegisterPromise_CalledWithinExecute_ResolvesCorrectly()
  12. {
  13. Action<JsValue> resolveFunc = null;
  14. var engine = new Engine();
  15. engine.SetValue("f", new Func<JsValue>(() =>
  16. {
  17. var (promise, resolve, _) = engine.RegisterPromise();
  18. resolveFunc = resolve;
  19. return promise;
  20. }));
  21. var promise = engine.Evaluate("f();");
  22. resolveFunc(66);
  23. Assert.Equal(66, promise.UnwrapIfPromise());
  24. }
  25. [Fact]
  26. public void RegisterPromise_CalledWithinExecute_RejectsCorrectly()
  27. {
  28. Action<JsValue> rejectFunc = null;
  29. var engine = new Engine();
  30. engine.SetValue("f", new Func<JsValue>(() =>
  31. {
  32. var (promise, _, reject) = engine.RegisterPromise();
  33. rejectFunc = reject;
  34. return promise;
  35. }));
  36. engine.Execute("f();");
  37. var completion = engine.Evaluate("f();");
  38. rejectFunc("oops!");
  39. var ex = Assert.Throws<PromiseRejectedException>(() => { completion.UnwrapIfPromise(); });
  40. Assert.Equal("oops!", ex.RejectedValue.AsString());
  41. }
  42. [Fact]
  43. public void RegisterPromise_UsedWithRace_WorksFlawlessly()
  44. {
  45. var engine = new Engine();
  46. Action<JsValue> resolve1 = null;
  47. engine.SetValue("f1", new Func<JsValue>(() =>
  48. {
  49. var (promise, resolve, _) = engine.RegisterPromise();
  50. resolve1 = resolve;
  51. return promise;
  52. }));
  53. Action<JsValue> resolve2 = null;
  54. engine.SetValue("f2", new Func<JsValue>(() =>
  55. {
  56. var (promise, resolve, _) = engine.RegisterPromise();
  57. resolve2 = resolve;
  58. return promise;
  59. }));
  60. var completion = engine.Evaluate("Promise.race([f1(), f2()]);");
  61. resolve1("first");
  62. // still not finished but the promise is fulfilled
  63. Assert.Equal("first", completion.UnwrapIfPromise());
  64. resolve2("second");
  65. // completion value hasn't changed
  66. Assert.Equal("first", completion.UnwrapIfPromise());
  67. }
  68. [Fact]
  69. public void Execute_ConcurrentNormalExecuteCall_WorksFine()
  70. {
  71. var engine = new Engine();
  72. engine.SetValue("f", new Func<JsValue>(() => engine.RegisterPromise().Promise));
  73. engine.Execute("f();");
  74. Assert.Equal(true, engine.Evaluate(" 1 + 1 === 2"));
  75. }
  76. [Fact(Timeout = 5000)]
  77. public void PromiseCtorWithNoResolver_Throws()
  78. {
  79. var engine = new Engine();
  80. Assert.Throws<JavaScriptException>(() => { engine.Execute("new Promise();"); });
  81. }
  82. [Fact(Timeout = 5000)]
  83. public void PromiseCtorWithInvalidResolver_Throws()
  84. {
  85. var engine = new Engine();
  86. Assert.Throws<JavaScriptException>(() => { engine.Execute("new Promise({});"); });
  87. }
  88. [Fact(Timeout = 5000)]
  89. public void PromiseCtorWithValidResolver_DoesNotThrow()
  90. {
  91. var engine = new Engine();
  92. engine.Execute("new Promise((resolve, reject)=>{});");
  93. }
  94. [Fact(Timeout = 5000)]
  95. public void PromiseCtor_ReturnsPromiseJsValue()
  96. {
  97. var engine = new Engine();
  98. var promise = engine.Evaluate("new Promise((resolve, reject)=>{});");
  99. Assert.IsType<JsPromise>(promise);
  100. }
  101. [Fact(Timeout = 5000)]
  102. public void PromiseResolveViaResolver_ReturnsCorrectValue()
  103. {
  104. var engine = new Engine();
  105. var res = engine.Evaluate("new Promise((resolve, reject)=>{resolve(66);});").UnwrapIfPromise();
  106. Assert.Equal(66, res);
  107. }
  108. [Fact(Timeout = 5000)]
  109. public void PromiseResolveViaStatic_ReturnsCorrectValue()
  110. {
  111. var engine = new Engine();
  112. Assert.Equal(66, engine.Evaluate("Promise.resolve(66);").UnwrapIfPromise());
  113. }
  114. [Fact(Timeout = 5000)]
  115. public void PromiseRejectViaResolver_ThrowsPromiseRejectedException()
  116. {
  117. var engine = new Engine();
  118. var ex = Assert.Throws<PromiseRejectedException>(() =>
  119. {
  120. engine.Evaluate("new Promise((resolve, reject)=>{reject('Could not connect');});").UnwrapIfPromise();
  121. });
  122. Assert.Equal("Could not connect", ex.RejectedValue.AsString());
  123. }
  124. [Fact(Timeout = 5000)]
  125. public void PromiseRejectViaStatic_ThrowsPromiseRejectedException()
  126. {
  127. var engine = new Engine();
  128. var ex = Assert.Throws<PromiseRejectedException>(() =>
  129. {
  130. engine.Evaluate("Promise.reject('Could not connect');").UnwrapIfPromise();
  131. });
  132. Assert.Equal("Could not connect", ex.RejectedValue.AsString());
  133. }
  134. [Fact(Timeout = 5000)]
  135. public void PromiseChainedThen_HandlerCalledWithCorrectValue()
  136. {
  137. var engine = new Engine();
  138. var res = engine.Evaluate(
  139. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => 44).then(result => resolve(result)); });").UnwrapIfPromise();
  140. Assert.Equal(44, res);
  141. }
  142. [Fact(Timeout = 5000)]
  143. public void PromiseThen_ReturnsNewPromiseInstance()
  144. {
  145. var engine = new Engine();
  146. var res = engine.Evaluate(
  147. "var promise1 = new Promise((resolve, reject) => { resolve(1); }); var promise2 = promise1.then(); promise1 === promise2").UnwrapIfPromise();
  148. Assert.Equal(false, res);
  149. }
  150. [Fact(Timeout = 5000)]
  151. public void PromiseThen_CalledCorrectlyOnResolve()
  152. {
  153. var engine = new Engine();
  154. var res = engine.Evaluate(
  155. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(result => resolve(result)); });").UnwrapIfPromise();
  156. Assert.Equal(66, res);
  157. }
  158. [Fact(Timeout = 5000)]
  159. public void PromiseResolveChainedWithHandler_ResolvedAsUndefined()
  160. {
  161. var engine = new Engine();
  162. Assert.Equal(JsValue.Undefined, engine.Evaluate("Promise.resolve(33).then(() => {});").UnwrapIfPromise());
  163. }
  164. [Fact(Timeout = 5000)]
  165. public void PromiseChainedThenWithUndefinedCallback_PassesThroughValueCorrectly()
  166. {
  167. var engine = new Engine();
  168. var res = engine.Evaluate(
  169. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then().then(result => resolve(result)); });").UnwrapIfPromise();
  170. Assert.Equal(66, res);
  171. }
  172. [Fact(Timeout = 5000)]
  173. public void PromiseChainedThenWithCallbackReturningUndefined_PassesThroughUndefinedCorrectly()
  174. {
  175. var engine = new Engine();
  176. var res = engine.Evaluate(
  177. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => {}).then(result => resolve(result)); });").UnwrapIfPromise();
  178. Assert.Equal(JsValue.Undefined, res);
  179. }
  180. [Fact(Timeout = 5000)]
  181. public void PromiseChainedThenThrowsError_ChainedCallsCatchWithThrownError()
  182. {
  183. var engine = new Engine();
  184. var res = engine.Evaluate(
  185. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => { throw 'Thrown Error'; }).catch(result => resolve(result)); });").UnwrapIfPromise();
  186. Assert.Equal("Thrown Error", res);
  187. }
  188. [Fact(Timeout = 5000)]
  189. public void PromiseChainedThenReturnsResolvedPromise_ChainedCallsThenWithPromiseValue()
  190. {
  191. var engine = new Engine();
  192. var res = engine.Evaluate(
  193. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => Promise.resolve(55)).then(result => resolve(result)); });").UnwrapIfPromise();
  194. Assert.Equal(55, res);
  195. }
  196. [Fact(Timeout = 5000)]
  197. public void PromiseChainedThenReturnsRejectedPromise_ChainedCallsCatchWithPromiseValue()
  198. {
  199. var engine = new Engine();
  200. var res = engine.Evaluate(
  201. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => Promise.reject('Error Message')).catch(result => resolve(result)); });").UnwrapIfPromise();
  202. Assert.Equal("Error Message", res);
  203. }
  204. [Fact(Timeout = 5000)]
  205. public void PromiseCatch_CalledCorrectlyOnReject()
  206. {
  207. var engine = new Engine();
  208. var res = engine.Evaluate(
  209. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).catch(result => resolve(result)); });").UnwrapIfPromise();
  210. Assert.Equal("Could not connect", res);
  211. }
  212. [Fact(Timeout = 5000)]
  213. public void PromiseThenWithCatch_CalledCorrectlyOnReject()
  214. {
  215. var engine = new Engine();
  216. var res = engine.Evaluate(
  217. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).then(undefined, result => resolve(result)); });").UnwrapIfPromise();
  218. Assert.Equal("Could not connect", res);
  219. }
  220. [Fact(Timeout = 5000)]
  221. public void PromiseChainedWithHandler_ResolvedAsUndefined()
  222. {
  223. var engine = new Engine();
  224. Assert.Equal(JsValue.Undefined, engine.Evaluate("Promise.reject('error').catch(() => {});").UnwrapIfPromise());
  225. }
  226. [Fact(Timeout = 5000)]
  227. public void PromiseChainedCatchThen_ThenCallWithUndefined()
  228. {
  229. var engine = new Engine();
  230. var res = engine.Evaluate(
  231. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).catch(ex => {}).then(result => resolve(result)); });").UnwrapIfPromise();
  232. Assert.Equal(JsValue.Undefined, res);
  233. }
  234. [Fact(Timeout = 5000)]
  235. public void PromiseChainedCatchWithUndefinedHandler_CatchChainedCorrectly()
  236. {
  237. var engine = new Engine();
  238. var res = engine.Evaluate(
  239. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).catch().catch(result => resolve(result)); });").UnwrapIfPromise();
  240. Assert.Equal("Could not connect", res);
  241. }
  242. [Fact(Timeout = 5000)]
  243. public void PromiseChainedFinally_HandlerCalled()
  244. {
  245. var engine = new Engine();
  246. var res = engine.Evaluate(
  247. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).finally(() => resolve(16)); });").UnwrapIfPromise();
  248. Assert.Equal(16, res);
  249. }
  250. [Fact(Timeout = 5000)]
  251. public void PromiseFinally_ReturnsNewPromiseInstance()
  252. {
  253. var engine = new Engine();
  254. var res = engine.Evaluate(
  255. "var promise1 = new Promise((resolve, reject) => { resolve(1); }); var promise2 = promise1.finally(); promise1 === promise2");
  256. Assert.Equal(false, res);
  257. }
  258. [Fact(Timeout = 5000)]
  259. public void PromiseFinally_ResolvesWithCorrectValue()
  260. {
  261. var engine = new Engine();
  262. Assert.Equal(2, engine.Evaluate("Promise.resolve(2).finally(() => {})").UnwrapIfPromise());
  263. }
  264. [Fact(Timeout = 5000)]
  265. public void PromiseFinallyWithNoCallback_ResolvesWithCorrectValue()
  266. {
  267. var engine = new Engine();
  268. Assert.Equal(2, engine.Evaluate("Promise.resolve(2).finally()").UnwrapIfPromise());
  269. }
  270. [Fact(Timeout = 5000)]
  271. public void PromiseFinallyChained_ResolvesWithCorrectValue()
  272. {
  273. var engine = new Engine();
  274. Assert.Equal(2, engine.Evaluate("Promise.resolve(2).finally(() => 6).finally(() => 9);").UnwrapIfPromise());
  275. }
  276. [Fact(Timeout = 5000)]
  277. public void PromiseFinallyWhichThrows_ResolvesWithError()
  278. {
  279. var engine = new Engine();
  280. var res = engine.Evaluate(
  281. "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(5)}).finally(() => {throw 'Could not connect';}).catch(result => resolve(result)); });").UnwrapIfPromise();
  282. Assert.Equal("Could not connect", res);
  283. }
  284. [Fact(Timeout = 5000)]
  285. public void PromiseAll_BadIterable_Rejects()
  286. {
  287. var engine = new Engine();
  288. Assert.Throws<PromiseRejectedException>(() => { engine.Evaluate("Promise.all();").UnwrapIfPromise(); });
  289. }
  290. [Fact(Timeout = 5000)]
  291. public void PromiseAll_ArgsAreNotPromises_ResolvesCorrectly()
  292. {
  293. var engine = new Engine();
  294. Assert.Equal(new object[] {1d, 2d, 3d}, engine.Evaluate("Promise.all([1,2,3]);").UnwrapIfPromise().ToObject());
  295. }
  296. [Fact(Timeout = 5000)]
  297. public void PromiseAll_MixturePromisesNoPromises_ResolvesCorrectly()
  298. {
  299. var engine = new Engine();
  300. Assert.Equal(new object[] {1d, 2d, 3d},
  301. engine.Evaluate("Promise.all([1,Promise.resolve(2),3]);").UnwrapIfPromise().ToObject());
  302. }
  303. [Fact(Timeout = 5000)]
  304. public void PromiseAll_MixturePromisesNoPromisesOneRejects_ResolvesCorrectly()
  305. {
  306. var engine = new Engine();
  307. Assert.Throws<PromiseRejectedException>(() =>
  308. {
  309. engine.Evaluate("Promise.all([1,Promise.resolve(2),3, Promise.reject('Cannot connect')]);").UnwrapIfPromise();
  310. });
  311. }
  312. [Fact(Timeout = 5000)]
  313. public void PromiseRace_NoArgs_Rejects()
  314. {
  315. var engine = new Engine();
  316. Assert.Throws<PromiseRejectedException>(() => { engine.Evaluate("Promise.race();").UnwrapIfPromise(); });
  317. }
  318. [Fact(Timeout = 5000)]
  319. public void PromiseRace_InvalidIterator_Rejects()
  320. {
  321. var engine = new Engine();
  322. Assert.Throws<PromiseRejectedException>(() => { engine.Evaluate("Promise.race({});").UnwrapIfPromise(); });
  323. }
  324. [Fact(Timeout = 5000)]
  325. public void PromiseRaceNoPromises_ResolvesCorrectly()
  326. {
  327. var engine = new Engine();
  328. Assert.Equal(12d, engine.Evaluate("Promise.race([12,2,3]);").UnwrapIfPromise().ToObject());
  329. }
  330. [Fact(Timeout = 5000)]
  331. public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly()
  332. {
  333. var engine = new Engine();
  334. Assert.Equal(12d, engine.Evaluate("Promise.race([12,Promise.resolve(2),3]);").UnwrapIfPromise().ToObject());
  335. }
  336. [Fact(Timeout = 5000)]
  337. public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly2()
  338. {
  339. var engine = new Engine();
  340. Assert.Equal(2d, engine.Evaluate("Promise.race([Promise.resolve(2),6,3]);").UnwrapIfPromise().ToObject());
  341. }
  342. [Fact(Timeout = 5000)]
  343. public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly3()
  344. {
  345. var engine = new Engine();
  346. var res = engine.Evaluate("Promise.race([new Promise((resolve,reject)=>{}),Promise.resolve(55),3]);").UnwrapIfPromise();
  347. Assert.Equal(55d, res.ToObject());
  348. }
  349. [Fact(Timeout = 5000)]
  350. public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly4()
  351. {
  352. var engine = new Engine();
  353. Assert.Throws<PromiseRejectedException>(() =>
  354. {
  355. engine.Evaluate(
  356. "Promise.race([new Promise((resolve,reject)=>{}),Promise.reject('Could not connect'),3]);").UnwrapIfPromise();
  357. });
  358. }
  359. [Fact(Timeout = 5000)]
  360. public void PromiseRegression_SingleElementArrayWithClrDictionaryInPromiseAll()
  361. {
  362. var engine = new Engine();
  363. var dictionary = new Dictionary<string, object>
  364. {
  365. { "Value 1", 1 },
  366. { "Value 2", "a string" }
  367. };
  368. engine.SetValue("clrDictionary", dictionary);
  369. var resultAsObject = engine
  370. .Evaluate(@"
  371. const promiseArray = [clrDictionary];
  372. return Promise.all(promiseArray);") // Returning and array through Promise.any()
  373. .UnwrapIfPromise()
  374. .ToObject();
  375. var result = (object[]) resultAsObject;
  376. Assert.Single(result);
  377. Assert.IsType<Dictionary<string, object>>(result[0]);
  378. }
  379. [Fact]
  380. public void ManualPromise_HasCorrectStackTrace()
  381. {
  382. using var engine = new Engine();
  383. string logMessage = null;
  384. var promise = engine.RegisterPromise();
  385. engine.SetValue("log", new Action<JsValue>((error) => {
  386. logMessage = (error as ObjectInstance)["stack"].ToString();
  387. }));
  388. engine.SetValue("getPromise", new Func<JsValue>(() => promise.Promise));
  389. engine.Execute( "const thePromise = getPromise(); thePromise.then(() => new Error()).then(e => log(e));" );
  390. // Calling this method will execute the JavaScript again.
  391. promise.Resolve(JsValue.Undefined);
  392. Assert.Equal("at <anonymous>:1:56", logMessage?.Trim());
  393. }
  394. }