ModuleTests.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. using System.IO;
  2. using System.Reflection;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Linq;
  6. using System.Threading;
  7. using Jint.Native;
  8. using Jint.Runtime;
  9. using Xunit;
  10. namespace Jint.Tests.Runtime;
  11. public class ModuleTests
  12. {
  13. private readonly Engine _engine;
  14. public ModuleTests()
  15. {
  16. _engine = new Engine();
  17. }
  18. [Fact]
  19. public void ShouldExportNamed()
  20. {
  21. _engine.AddModule("my-module", @"export const value = 'exported value';");
  22. var ns = _engine.ImportModule("my-module");
  23. Assert.Equal("exported value", ns.Get("value").AsString());
  24. }
  25. [Fact]
  26. public void ShouldExportNamedListRenamed()
  27. {
  28. _engine.AddModule("my-module", @"const value1 = 1; const value2 = 2; export { value1 as renamed1, value2 as renamed2 }");
  29. var ns = _engine.ImportModule("my-module");
  30. Assert.Equal(1, ns.Get("renamed1").AsInteger());
  31. Assert.Equal(2, ns.Get("renamed2").AsInteger());
  32. }
  33. [Fact]
  34. public void ShouldExportDefault()
  35. {
  36. _engine.AddModule("my-module", @"export default 'exported value';");
  37. var ns = _engine.ImportModule("my-module");
  38. Assert.Equal("exported value", ns.Get("default").AsString());
  39. }
  40. [Fact]
  41. public void ShouldExportAll()
  42. {
  43. _engine.AddModule("module1", @"export const value = 'exported value';");
  44. _engine.AddModule("module2", @"export * from 'module1';");
  45. var ns = _engine.ImportModule("module2");
  46. Assert.Equal("exported value", ns.Get("value").AsString());
  47. }
  48. [Fact]
  49. public void ShouldImportNamed()
  50. {
  51. _engine.AddModule("imported-module", @"export const value = 'exported value';");
  52. _engine.AddModule("my-module", @"import { value } from 'imported-module'; export const exported = value;");
  53. var ns = _engine.ImportModule("my-module");
  54. Assert.Equal("exported value", ns.Get("exported").AsString());
  55. }
  56. [Fact]
  57. public void ShouldImportRenamed()
  58. {
  59. _engine.AddModule("imported-module", @"export const value = 'exported value';");
  60. _engine.AddModule("my-module", @"import { value as renamed } from 'imported-module'; export const exported = renamed;");
  61. var ns = _engine.ImportModule("my-module");
  62. Assert.Equal("exported value", ns.Get("exported").AsString());
  63. }
  64. [Fact]
  65. public void ShouldImportDefault()
  66. {
  67. _engine.AddModule("imported-module", @"export default 'exported value';");
  68. _engine.AddModule("my-module", @"import imported from 'imported-module'; export const exported = imported;");
  69. var ns = _engine.ImportModule("my-module");
  70. Assert.Equal("exported value", ns.Get("exported").AsString());
  71. }
  72. [Fact]
  73. public void ShouldImportAll()
  74. {
  75. _engine.AddModule("imported-module", @"export const value = 'exported value';");
  76. _engine.AddModule("my-module", @"import * as imported from 'imported-module'; export const exported = imported.value;");
  77. var ns = _engine.ImportModule("my-module");
  78. Assert.Equal("exported value", ns.Get("exported").AsString());
  79. }
  80. [Fact]
  81. public void ShouldImportDynamically()
  82. {
  83. var received = false;
  84. _engine.AddModule("imported-module", builder => builder.ExportFunction("signal", () => received = true));
  85. _engine.AddModule("my-module", @"import('imported-module').then(ns => { ns.signal(); });");
  86. _engine.ImportModule("my-module");
  87. _engine.RunAvailableContinuations();
  88. Assert.True(received);
  89. }
  90. [Fact]
  91. public void ShouldPropagateParseError()
  92. {
  93. _engine.AddModule("imported", @"export const invalid;");
  94. _engine.AddModule("my-module", @"import { invalid } from 'imported';");
  95. var exc = Assert.Throws<JavaScriptException>(() => _engine.ImportModule("my-module"));
  96. Assert.Equal("Error while loading module: error in module 'imported': Line 1: Missing initializer in const declaration", exc.Message);
  97. Assert.Equal("imported", exc.Location.Source);
  98. }
  99. [Fact]
  100. public void ShouldPropagateLinkError()
  101. {
  102. _engine.AddModule("imported", @"export invalid;");
  103. _engine.AddModule("my-module", @"import { value } from 'imported';");
  104. var exc = Assert.Throws<JavaScriptException>(() => _engine.ImportModule("my-module"));
  105. Assert.Equal("Error while loading module: error in module 'imported': Line 1: Unexpected identifier", exc.Message);
  106. Assert.Equal("imported", exc.Location.Source);
  107. }
  108. [Fact]
  109. public void ShouldPropagateExecuteError()
  110. {
  111. _engine.AddModule("my-module", @"throw new Error('imported successfully');");
  112. var exc = Assert.Throws<JavaScriptException>(() => _engine.ImportModule("my-module"));
  113. Assert.Equal("imported successfully", exc.Message);
  114. Assert.Equal("my-module", exc.Location.Source);
  115. }
  116. [Fact]
  117. public void ShouldPropagateThrowStatementThroughJavaScriptImport()
  118. {
  119. _engine.AddModule("imported-module", @"throw new Error('imported successfully');");
  120. _engine.AddModule("my-module", @"import 'imported-module';");
  121. var exc = Assert.Throws<JavaScriptException>(() => _engine.ImportModule("my-module"));
  122. Assert.Equal("imported successfully", exc.Message);
  123. }
  124. [Fact]
  125. public void ShouldAddModuleFromJsValue()
  126. {
  127. _engine.AddModule("my-module", builder => builder.ExportValue("value", JsString.Create("hello world")));
  128. var ns = _engine.ImportModule("my-module");
  129. Assert.Equal("hello world", ns.Get("value").AsString());
  130. }
  131. [Fact]
  132. public void ShouldAddModuleFromClrInstance()
  133. {
  134. _engine.AddModule("imported-module", builder => builder.ExportObject("value", new ImportedClass { Value = "instance value" }));
  135. _engine.AddModule("my-module", @"import { value } from 'imported-module'; export const exported = value.value;");
  136. var ns = _engine.ImportModule("my-module");
  137. Assert.Equal("instance value", ns.Get("exported").AsString());
  138. }
  139. [Fact]
  140. public void ShouldAllowInvokeUserDefinedClass()
  141. {
  142. _engine.AddModule("user", "export class UserDefined { constructor(v) { this._v = v; } hello(c) { return `hello ${this._v}${c}`; } }");
  143. var ctor = _engine.ImportModule("user").Get("UserDefined");
  144. var instance = _engine.Construct(ctor, JsString.Create("world"));
  145. var result = instance.GetMethod("hello").Call(instance, JsString.Create("!"));
  146. Assert.Equal("hello world!", result);
  147. }
  148. [Fact]
  149. public void ShouldAddModuleFromClrType()
  150. {
  151. _engine.AddModule("imported-module", builder => builder.ExportType<ImportedClass>());
  152. _engine.AddModule("my-module", @"import { ImportedClass } from 'imported-module'; export const exported = new ImportedClass().value;");
  153. var ns = _engine.ImportModule("my-module");
  154. Assert.Equal("hello world", ns.Get("exported").AsString());
  155. }
  156. [Fact]
  157. public void ShouldAddModuleFromClrFunction()
  158. {
  159. var received = new List<string>();
  160. _engine.AddModule("imported-module", builder => builder
  161. .ExportFunction("act_noargs", () => received.Add("act_noargs"))
  162. .ExportFunction("act_args", args => received.Add($"act_args:{args[0].AsString()}"))
  163. .ExportFunction("fn_noargs", () =>
  164. {
  165. received.Add("fn_noargs");
  166. return "ret";
  167. })
  168. .ExportFunction("fn_args", args =>
  169. {
  170. received.Add($"fn_args:{args[0].AsString()}");
  171. return "ret";
  172. })
  173. );
  174. _engine.AddModule("my-module", @"
  175. import * as fns from 'imported-module';
  176. export const result = [fns.act_noargs(), fns.act_args('ok'), fns.fn_noargs(), fns.fn_args('ok')];");
  177. var ns = _engine.ImportModule("my-module");
  178. Assert.Equal(new[] { "act_noargs", "act_args:ok", "fn_noargs", "fn_args:ok" }, received.ToArray());
  179. Assert.Equal(new[] { "undefined", "undefined", "ret", "ret" }, ns.Get("result").AsArray().Select(x => x.ToString()).ToArray());
  180. }
  181. private class ImportedClass
  182. {
  183. public string Value { get; set; } = "hello world";
  184. }
  185. [Fact]
  186. public void ShouldAllowExportMultipleImports()
  187. {
  188. _engine.AddModule("@mine/import1", builder => builder.ExportValue("value1", JsNumber.Create(1)));
  189. _engine.AddModule("@mine/import2", builder => builder.ExportValue("value2", JsNumber.Create(2)));
  190. _engine.AddModule("@mine", "export * from '@mine/import1'; export * from '@mine/import2'");
  191. _engine.AddModule("app", @"import { value1, value2 } from '@mine'; export const result = `${value1} ${value2}`");
  192. var ns = _engine.ImportModule("app");
  193. Assert.Equal("1 2", ns.Get("result").AsString());
  194. }
  195. [Fact]
  196. public void ShouldAllowNamedStarExport()
  197. {
  198. _engine.AddModule("imported-module", builder => builder.ExportValue("value1", 5));
  199. _engine.AddModule("my-module", "export * as ns from 'imported-module';");
  200. var ns = _engine.ImportModule("my-module");
  201. Assert.Equal(5, ns.Get("ns").Get("value1").AsNumber());
  202. }
  203. [Fact]
  204. public void ShouldAllowChaining()
  205. {
  206. _engine.AddModule("dependent-module", "export const dependency = 1;");
  207. _engine.AddModule("my-module", builder => builder
  208. .AddSource("import { dependency } from 'dependent-module';")
  209. .AddSource("export const output = dependency + 1;")
  210. .ExportValue("num", JsNumber.Create(-1))
  211. );
  212. var ns = _engine.ImportModule("my-module");
  213. Assert.Equal(2, ns.Get("output").AsInteger());
  214. Assert.Equal(-1, ns.Get("num").AsInteger());
  215. }
  216. [Fact]
  217. public void ShouldImportOnlyOnce()
  218. {
  219. var called = 0;
  220. _engine.AddModule("imported-module", builder => builder.ExportFunction("count", args => called++));
  221. _engine.AddModule("my-module", @"import { count } from 'imported-module'; count();");
  222. _engine.ImportModule("my-module");
  223. _engine.ImportModule("my-module");
  224. Assert.Equal(1, called);
  225. }
  226. [Fact]
  227. public void ShouldAllowSelfImport()
  228. {
  229. _engine.AddModule("my-globals", @"export const globals = { counter: 0 };");
  230. _engine.AddModule("my-module", @"
  231. import { globals } from 'my-globals';
  232. import {} from 'my-module';
  233. globals.counter++;
  234. export const count = globals.counter;
  235. ");
  236. var ns= _engine.ImportModule("my-module");
  237. Assert.Equal(1, ns.Get("count").AsInteger());
  238. }
  239. [Fact]
  240. public void ShouldAllowCyclicImport()
  241. {
  242. // https://tc39.es/ecma262/#sec-example-cyclic-module-record-graphs
  243. _engine.AddModule("B", @"import { a } from 'A'; export const b = 'b';");
  244. _engine.AddModule("A", @"import { b } from 'B'; export const a = 'a';");
  245. var nsA = _engine.ImportModule("A");
  246. var nsB = _engine.ImportModule("B");
  247. Assert.Equal("a", nsA.Get("a").AsString());
  248. Assert.Equal("b", nsB.Get("b").AsString());
  249. }
  250. [Fact]
  251. public void ShouldSupportConstraints()
  252. {
  253. var engine = new Engine(opts => opts.TimeoutInterval(TimeSpan.FromTicks(1)));
  254. engine.AddModule("sleep", builder => builder.ExportFunction("sleep", () => Thread.Sleep(100)));
  255. engine.AddModule("my-module", @"import { sleep } from 'sleep'; for(var i = 0; i < 100; i++) { sleep(); } export const result = 'ok';");
  256. Assert.Throws<TimeoutException>(() => engine.ImportModule("my-module"));
  257. }
  258. [Fact]
  259. public void CanLoadModuleImportsFromFiles()
  260. {
  261. var engine = new Engine(options => options.EnableModules(GetBasePath()));
  262. engine.AddModule("my-module", "import { User } from './modules/user.js'; export const user = new User('John', 'Doe');");
  263. var ns = engine.ImportModule("my-module");
  264. Assert.Equal("John Doe", ns["user"].Get("name").AsString());
  265. }
  266. [Fact]
  267. public void CanImportFromFile()
  268. {
  269. var engine = new Engine(options => options.EnableModules(GetBasePath()));
  270. var ns = engine.ImportModule("./modules/format-name.js");
  271. var result = engine.Invoke(ns.Get("formatName"), "John", "Doe").AsString();
  272. Assert.Equal("John Doe", result);
  273. }
  274. internal static string GetBasePath()
  275. {
  276. var assemblyDirectory = new DirectoryInfo(AppDomain.CurrentDomain.RelativeSearchPath ?? AppDomain.CurrentDomain.BaseDirectory);
  277. var current = assemblyDirectory;
  278. while (current is not null && current.Name != "Jint.Tests")
  279. {
  280. current = current.Parent;
  281. }
  282. if (current is null)
  283. {
  284. throw new NullReferenceException($"Could not find tests base path, assemblyPath: {assemblyDirectory}");
  285. }
  286. return Path.Combine(current.FullName, "Runtime", "Scripts");
  287. }
  288. }