PromiseTests.cs 18 KB

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