123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 |
- using Jint.Native;
- using Jint.Native.Object;
- using Jint.Runtime;
- // obsolete GetCompletionValue
- #pragma warning disable 618
- namespace Jint.Tests.Runtime;
- public class PromiseTests
- {
- [Fact]
- public void RegisterPromise_CalledWithinExecute_ResolvesCorrectly()
- {
- Action<JsValue> resolveFunc = null;
- var engine = new Engine();
- engine.SetValue("f", new Func<JsValue>(() =>
- {
- var (promise, resolve, _) = engine.RegisterPromise();
- resolveFunc = resolve;
- return promise;
- }));
- var promise = engine.Evaluate("f();");
- resolveFunc(66);
- Assert.Equal(66, promise.UnwrapIfPromise());
- }
- [Fact]
- public void RegisterPromise_CalledWithinExecute_RejectsCorrectly()
- {
- Action<JsValue> rejectFunc = null;
- var engine = new Engine();
- engine.SetValue("f", new Func<JsValue>(() =>
- {
- var (promise, _, reject) = engine.RegisterPromise();
- rejectFunc = reject;
- return promise;
- }));
- engine.Execute("f();");
- var completion = engine.Evaluate("f();");
- rejectFunc("oops!");
- var ex = Assert.Throws<PromiseRejectedException>(() => { completion.UnwrapIfPromise(); });
- Assert.Equal("oops!", ex.RejectedValue.AsString());
- }
- [Fact]
- public void RegisterPromise_UsedWithRace_WorksFlawlessly()
- {
- var engine = new Engine();
- Action<JsValue> resolve1 = null;
- engine.SetValue("f1", new Func<JsValue>(() =>
- {
- var (promise, resolve, _) = engine.RegisterPromise();
- resolve1 = resolve;
- return promise;
- }));
- Action<JsValue> resolve2 = null;
- engine.SetValue("f2", new Func<JsValue>(() =>
- {
- var (promise, resolve, _) = engine.RegisterPromise();
- resolve2 = resolve;
- return promise;
- }));
- var completion = engine.Evaluate("Promise.race([f1(), f2()]);");
- resolve1("first");
- // still not finished but the promise is fulfilled
- Assert.Equal("first", completion.UnwrapIfPromise());
- resolve2("second");
- // completion value hasn't changed
- Assert.Equal("first", completion.UnwrapIfPromise());
- }
- [Fact]
- public void Execute_ConcurrentNormalExecuteCall_WorksFine()
- {
- var engine = new Engine();
- engine.SetValue("f", new Func<JsValue>(() => engine.RegisterPromise().Promise));
- engine.Execute("f();");
- Assert.Equal(true, engine.Evaluate(" 1 + 1 === 2"));
- }
- [Fact]
- public void PromiseCtorWithNoResolver_Throws()
- {
- var engine = new Engine();
- Assert.Throws<JavaScriptException>(() => { engine.Execute("new Promise();"); });
- }
- [Fact]
- public void PromiseCtorWithInvalidResolver_Throws()
- {
- var engine = new Engine();
- Assert.Throws<JavaScriptException>(() => { engine.Execute("new Promise({});"); });
- }
- [Fact]
- public void PromiseCtorWithValidResolver_DoesNotThrow()
- {
- var engine = new Engine();
- engine.Execute("new Promise((resolve, reject)=>{});");
- }
- [Fact]
- public void PromiseCtor_ReturnsPromiseJsValue()
- {
- var engine = new Engine();
- var promise = engine.Evaluate("new Promise((resolve, reject)=>{});");
- Assert.IsType<JsPromise>(promise);
- }
- [Fact]
- public void PromiseResolveViaResolver_ReturnsCorrectValue()
- {
- var engine = new Engine();
- var res = engine.Evaluate("new Promise((resolve, reject)=>{resolve(66);});").UnwrapIfPromise();
- Assert.Equal(66, res);
- }
- [Fact]
- public void PromiseResolveViaStatic_ReturnsCorrectValue()
- {
- var engine = new Engine();
- Assert.Equal(66, engine.Evaluate("Promise.resolve(66);").UnwrapIfPromise());
- }
- [Fact]
- public void PromiseRejectViaResolver_ThrowsPromiseRejectedException()
- {
- var engine = new Engine();
- var ex = Assert.Throws<PromiseRejectedException>(() =>
- {
- engine.Evaluate("new Promise((resolve, reject)=>{reject('Could not connect');});").UnwrapIfPromise();
- });
- Assert.Equal("Could not connect", ex.RejectedValue.AsString());
- }
- [Fact]
- public void PromiseRejectViaStatic_ThrowsPromiseRejectedException()
- {
- var engine = new Engine();
- var ex = Assert.Throws<PromiseRejectedException>(() =>
- {
- engine.Evaluate("Promise.reject('Could not connect');").UnwrapIfPromise();
- });
- Assert.Equal("Could not connect", ex.RejectedValue.AsString());
- }
- [Fact]
- public void PromiseChainedThen_HandlerCalledWithCorrectValue()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => 44).then(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal(44, res);
- }
- [Fact]
- public void PromiseThen_ReturnsNewPromiseInstance()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "var promise1 = new Promise((resolve, reject) => { resolve(1); }); var promise2 = promise1.then(); promise1 === promise2").UnwrapIfPromise();
- Assert.Equal(false, res);
- }
- [Fact]
- public void PromiseThen_CalledCorrectlyOnResolve()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal(66, res);
- }
- [Fact]
- public void PromiseResolveChainedWithHandler_ResolvedAsUndefined()
- {
- var engine = new Engine();
- Assert.Equal(JsValue.Undefined, engine.Evaluate("Promise.resolve(33).then(() => {});").UnwrapIfPromise());
- }
- [Fact]
- public void PromiseChainedThenWithUndefinedCallback_PassesThroughValueCorrectly()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then().then(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal(66, res);
- }
- [Fact]
- public void PromiseChainedThenWithCallbackReturningUndefined_PassesThroughUndefinedCorrectly()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => {}).then(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal(JsValue.Undefined, res);
- }
- [Fact]
- public void PromiseChainedThenThrowsError_ChainedCallsCatchWithThrownError()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => { throw 'Thrown Error'; }).catch(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal("Thrown Error", res);
- }
- [Fact]
- public void PromiseChainedThenReturnsResolvedPromise_ChainedCallsThenWithPromiseValue()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => Promise.resolve(55)).then(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal(55, res);
- }
- [Fact]
- public void PromiseChainedThenReturnsRejectedPromise_ChainedCallsCatchWithPromiseValue()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).then(() => Promise.reject('Error Message')).catch(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal("Error Message", res);
- }
- [Fact]
- public void PromiseCatch_CalledCorrectlyOnReject()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).catch(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal("Could not connect", res);
- }
- [Fact]
- public void PromiseThenWithCatch_CalledCorrectlyOnReject()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).then(undefined, result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal("Could not connect", res);
- }
- [Fact]
- public void PromiseChainedWithHandler_ResolvedAsUndefined()
- {
- var engine = new Engine();
- Assert.Equal(JsValue.Undefined, engine.Evaluate("Promise.reject('error').catch(() => {});").UnwrapIfPromise());
- }
- [Fact]
- public void PromiseChainedCatchThen_ThenCallWithUndefined()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).catch(ex => {}).then(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal(JsValue.Undefined, res);
- }
- [Fact]
- public void PromiseChainedCatchWithUndefinedHandler_CatchChainedCorrectly()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerReject('Could not connect')}).catch().catch(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal("Could not connect", res);
- }
- [Fact]
- public void PromiseChainedFinally_HandlerCalled()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(66)}).finally(() => resolve(16)); });").UnwrapIfPromise();
- Assert.Equal(16, res);
- }
- [Fact]
- public void PromiseFinally_ReturnsNewPromiseInstance()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "var promise1 = new Promise((resolve, reject) => { resolve(1); }); var promise2 = promise1.finally(); promise1 === promise2");
- Assert.Equal(false, res);
- }
- [Fact]
- public void PromiseFinally_ResolvesWithCorrectValue()
- {
- var engine = new Engine();
- Assert.Equal(2, engine.Evaluate("Promise.resolve(2).finally(() => {})").UnwrapIfPromise());
- }
- [Fact]
- public void PromiseFinallyWithNoCallback_ResolvesWithCorrectValue()
- {
- var engine = new Engine();
- Assert.Equal(2, engine.Evaluate("Promise.resolve(2).finally()").UnwrapIfPromise());
- }
- [Fact]
- public void PromiseFinallyChained_ResolvesWithCorrectValue()
- {
- var engine = new Engine();
- Assert.Equal(2, engine.Evaluate("Promise.resolve(2).finally(() => 6).finally(() => 9);").UnwrapIfPromise());
- }
- [Fact]
- public void PromiseFinallyWhichThrows_ResolvesWithError()
- {
- var engine = new Engine();
- var res = engine.Evaluate(
- "new Promise((resolve, reject) => { new Promise((innerResolve, innerReject) => {innerResolve(5)}).finally(() => {throw 'Could not connect';}).catch(result => resolve(result)); });").UnwrapIfPromise();
- Assert.Equal("Could not connect", res);
- }
- [Fact]
- public void PromiseAll_BadIterable_Rejects()
- {
- var engine = new Engine();
- Assert.Throws<PromiseRejectedException>(() => { engine.Evaluate("Promise.all();").UnwrapIfPromise(); });
- }
- [Fact]
- public void PromiseAll_ArgsAreNotPromises_ResolvesCorrectly()
- {
- var engine = new Engine();
- Assert.Equal(new object[] {1d, 2d, 3d}, engine.Evaluate("Promise.all([1,2,3]);").UnwrapIfPromise().ToObject());
- }
- [Fact]
- public void PromiseAll_MixturePromisesNoPromises_ResolvesCorrectly()
- {
- var engine = new Engine();
- Assert.Equal(new object[] {1d, 2d, 3d},
- engine.Evaluate("Promise.all([1,Promise.resolve(2),3]);").UnwrapIfPromise().ToObject());
- }
- [Fact]
- public void PromiseAll_MixturePromisesNoPromisesOneRejects_ResolvesCorrectly()
- {
- var engine = new Engine();
- Assert.Throws<PromiseRejectedException>(() =>
- {
- engine.Evaluate("Promise.all([1,Promise.resolve(2),3, Promise.reject('Cannot connect')]);").UnwrapIfPromise();
- });
- }
- [Fact]
- public void PromiseRace_NoArgs_Rejects()
- {
- var engine = new Engine();
- Assert.Throws<PromiseRejectedException>(() => { engine.Evaluate("Promise.race();").UnwrapIfPromise(); });
- }
- [Fact]
- public void PromiseRace_InvalidIterator_Rejects()
- {
- var engine = new Engine();
- Assert.Throws<PromiseRejectedException>(() => { engine.Evaluate("Promise.race({});").UnwrapIfPromise(); });
- }
- [Fact]
- public void PromiseRaceNoPromises_ResolvesCorrectly()
- {
- var engine = new Engine();
- Assert.Equal(12d, engine.Evaluate("Promise.race([12,2,3]);").UnwrapIfPromise().ToObject());
- }
- [Fact]
- public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly()
- {
- var engine = new Engine();
- Assert.Equal(12d, engine.Evaluate("Promise.race([12,Promise.resolve(2),3]);").UnwrapIfPromise().ToObject());
- }
- [Fact]
- public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly2()
- {
- var engine = new Engine();
- Assert.Equal(2d, engine.Evaluate("Promise.race([Promise.resolve(2),6,3]);").UnwrapIfPromise().ToObject());
- }
- [Fact]
- public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly3()
- {
- var engine = new Engine();
- var res = engine.Evaluate("Promise.race([new Promise((resolve,reject)=>{}),Promise.resolve(55),3]);").UnwrapIfPromise();
- Assert.Equal(55d, res.ToObject());
- }
- [Fact]
- public void PromiseRaceMixturePromisesNoPromises_ResolvesCorrectly4()
- {
- var engine = new Engine();
- Assert.Throws<PromiseRejectedException>(() =>
- {
- engine.Evaluate(
- "Promise.race([new Promise((resolve,reject)=>{}),Promise.reject('Could not connect'),3]);").UnwrapIfPromise();
- });
- }
- [Fact]
- public void PromiseRegression_SingleElementArrayWithClrDictionaryInPromiseAll()
- {
- var engine = new Engine();
- var dictionary = new Dictionary<string, object>
- {
- { "Value 1", 1 },
- { "Value 2", "a string" }
- };
- engine.SetValue("clrDictionary", dictionary);
- var resultAsObject = engine
- .Evaluate(@"
- const promiseArray = [clrDictionary];
- return Promise.all(promiseArray);") // Returning and array through Promise.any()
- .UnwrapIfPromise()
- .ToObject();
- var result = (object[]) resultAsObject;
- Assert.Single(result);
- Assert.IsType<Dictionary<string, object>>(result[0]);
- }
- [Fact]
- public void ManualPromise_HasCorrectStackTrace()
- {
- using var engine = new Engine();
- string logMessage = null;
- var promise = engine.RegisterPromise();
- engine.SetValue("log", new Action<JsValue>((error) => {
- logMessage = (error as ObjectInstance)["stack"].ToString();
- }));
- engine.SetValue("getPromise", new Func<JsValue>(() => promise.Promise));
- engine.Execute( "const thePromise = getPromise(); thePromise.then(() => new Error()).then(e => log(e));" );
- // Calling this method will execute the JavaScript again.
- promise.Resolve(JsValue.Undefined);
- Assert.Equal("at <anonymous>:1:56", logMessage?.Trim());
- }
- }
|