PromiseTests.cs 17 KB

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