PromiseTests.cs 16 KB

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