PromiseTests.cs 18 KB

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