ModuleTests.cs 14 KB

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