RavenApiUsageTests.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. using Jint.Constraints;
  2. using Jint.Native;
  3. using Jint.Native.Function;
  4. using Jint.Runtime;
  5. using Jint.Runtime.Interop;
  6. namespace Jint.Tests.PublicInterface;
  7. /// <summary>
  8. /// Tests related to functionality that RavenDB needs exposed.
  9. /// </summary>
  10. public class RavenApiUsageTests
  11. {
  12. [Fact]
  13. public void CanBuildCustomScriptFunctionInstance()
  14. {
  15. var engine = new Engine();
  16. var properties = new Node[]
  17. {
  18. new ObjectProperty(PropertyKind.Init, new Identifier("field"),
  19. new MemberExpression(new Identifier("self"), new Identifier("field"), computed: false, optional: false), false, false, false)
  20. };
  21. var functionExp = new FunctionExpression(
  22. new Identifier("functionId"),
  23. NodeList.From<Node>(new Identifier("self")),
  24. new FunctionBody(NodeList.From<Statement>(new ReturnStatement(new ObjectExpression(NodeList.From(properties)))), strict: false),
  25. generator: false,
  26. async: false);
  27. var functionObject = new ScriptFunction(
  28. engine,
  29. functionExp,
  30. strict: false);
  31. Assert.NotNull(functionObject);
  32. }
  33. [Fact]
  34. public void CanChangeMaxStatementValue()
  35. {
  36. var engine = new Engine(options => options.MaxStatements(123));
  37. var constraint = engine.Constraints.Find<MaxStatementsConstraint>();
  38. Assert.NotNull(constraint);
  39. var oldMaxStatements = constraint.MaxStatements;
  40. constraint.MaxStatements = 321;
  41. Assert.Equal(123, oldMaxStatements);
  42. Assert.Equal(321, constraint.MaxStatements);
  43. }
  44. [Fact]
  45. public void CanGetPropertyDescriptor()
  46. {
  47. var engine = new Engine();
  48. var obj = new DirectoryInfo("the-path");
  49. var propertyDescriptor = ObjectWrapper.GetPropertyDescriptor(engine, obj, obj.GetType().GetProperty(nameof(DirectoryInfo.Name)));
  50. Assert.Equal("the-path", propertyDescriptor.Value);
  51. }
  52. [Fact]
  53. public void CanInjectConstructedObjects()
  54. {
  55. var engine = new Engine();
  56. var obj = new JsObject(engine);
  57. obj.FastSetDataProperty("name", "test");
  58. var array1 = new JsArray(engine, [
  59. JsNumber.Create(1),
  60. JsNumber.Create(2),
  61. JsNumber.Create(3)
  62. ]);
  63. engine.SetValue("array1", array1);
  64. TestArrayAccess(engine, array1, "array1");
  65. engine.SetValue("obj", obj);
  66. Assert.Equal("test", engine.Evaluate("obj.name"));
  67. engine.SetValue("emptyArray", new JsArray(engine));
  68. Assert.Equal(0, engine.Evaluate("emptyArray.length"));
  69. Assert.Equal(1, engine.Evaluate("emptyArray.push(1); return emptyArray.length"));
  70. engine.SetValue("emptyArray", new JsArray(engine, []));
  71. Assert.Equal(0, engine.Evaluate("emptyArray.length"));
  72. Assert.Equal(1, engine.Evaluate("emptyArray.push(1); return emptyArray.length"));
  73. engine.SetValue("date", new JsDate(engine, new DateTime(2022, 10, 20)));
  74. Assert.Equal(2022, engine.Evaluate("date.getFullYear()"));
  75. }
  76. private static void TestArrayAccess(Engine engine, JsArray array, string name)
  77. {
  78. Assert.Equal(1, engine.Evaluate($"{name}.findIndex(x => x === 2)"));
  79. Assert.Equal(2, array.GetOwnProperty("1").Value);
  80. array.Push(4);
  81. array.Push([5, 6]);
  82. var i = 0;
  83. foreach (var entry in array.GetEntries())
  84. {
  85. Assert.Equal(i.ToString(), entry.Key);
  86. Assert.Equal(i + 1, entry.Value);
  87. i++;
  88. }
  89. Assert.Equal(6, i);
  90. array[0] = "";
  91. array[1] = false;
  92. array[2] = null;
  93. Assert.Equal("", array[0]);
  94. Assert.Equal(false, array[1]);
  95. Assert.Equal(JsValue.Undefined, array[2]);
  96. Assert.Equal(4, array[3]);
  97. Assert.Equal(5, array[4]);
  98. Assert.Equal(6, array[5]);
  99. for (i = 0; i < 100; ++i)
  100. {
  101. array.Push(JsValue.Undefined);
  102. }
  103. Assert.Equal(106L, array.Length);
  104. Assert.True(array.All(x => x is JsNumber or JsUndefined or JsNumber or JsString or JsBoolean));
  105. }
  106. // Checks different ways how string can be checked for equality without the need to materialize lazy value
  107. [Fact]
  108. public void CanInheritCustomString()
  109. {
  110. var engine = new Engine();
  111. var str = new CustomString("the-value");
  112. engine.SetValue("str", str);
  113. var empty = new CustomString("");
  114. engine.SetValue("empty", empty);
  115. engine.SetValue("x", new CustomString("x", allowMaterialize: true));
  116. var obj = new JsObject(engine);
  117. obj.Set("name", new CustomString("the name"));
  118. engine.SetValue("obj", obj);
  119. var array = new JsArray(engine, Enumerable.Range(1, 100).Select(x => new CustomString(x.ToString())).ToArray<JsValue>());
  120. engine.SetValue("array", array);
  121. Assert.True(engine.Evaluate("str ? true : false").AsBoolean());
  122. Assert.False(engine.Evaluate("empty ? true : false").AsBoolean());
  123. Assert.True(engine.Evaluate("array.includes('2')").AsBoolean());
  124. Assert.True(engine.Evaluate("array.filter(x => x === '2').length > 0").AsBoolean());
  125. engine.SetValue("objArray", new JsArray(engine, [obj, obj]));
  126. Assert.True(engine.Evaluate("objArray.filter(x => x.name === 'the name').length === 2").AsBoolean());
  127. Assert.Equal(9, engine.Evaluate("str.length"));
  128. Assert.True(engine.Evaluate("str == 'the-value'").AsBoolean());
  129. Assert.True(engine.Evaluate("str === 'the-value'").AsBoolean());
  130. Assert.True(engine.Evaluate("str.indexOf('value-too-long') === -1").AsBoolean());
  131. Assert.True(engine.Evaluate("str.lastIndexOf('value-too-long') === -1").AsBoolean());
  132. Assert.False(engine.Evaluate("str.startsWith('value-too-long')").AsBoolean());
  133. Assert.False(engine.Evaluate("str.endsWith('value-too-long')").AsBoolean());
  134. Assert.False(engine.Evaluate("str.includes('value-too-long')").AsBoolean());
  135. Assert.True(engine.Evaluate("empty.trim() === ''").AsBoolean());
  136. Assert.True(engine.Evaluate("empty.trimStart() === ''").AsBoolean());
  137. Assert.True(engine.Evaluate("empty.trimEnd() === ''").AsBoolean());
  138. Assert.True(engine.Evaluate("str[1] === 'h'").AsBoolean());
  139. Assert.True(engine.Evaluate("str[x] === undefined").AsBoolean());
  140. }
  141. [Fact]
  142. public void CanDefineCustomNull()
  143. {
  144. var engine = new Engine();
  145. engine.SetValue("value", new CustomNull());
  146. Assert.Equal("foo", engine.Evaluate("value ? value + 'bar' : 'foo'"));
  147. }
  148. [Fact]
  149. public void CanDefineCustomUndefined()
  150. {
  151. var engine = new Engine();
  152. engine.SetValue("value", new CustomUndefined());
  153. Assert.Equal("foo", engine.Evaluate("value ? value + 'bar' : 'foo'"));
  154. }
  155. [Fact]
  156. public void CanResetCallStack()
  157. {
  158. var engine = new Engine();
  159. engine.Advanced.ResetCallStack();
  160. }
  161. [Fact]
  162. public void CanUseCustomReferenceResolver()
  163. {
  164. var engine = new Engine(options =>
  165. {
  166. options.ReferenceResolver = new MyReferenceResolver();
  167. });
  168. engine
  169. .Execute("""
  170. function output(doc) {
  171. var rows123 = [{}];
  172. var test = null;
  173. return {
  174. Rows : [{}].map(row=>({row:row, myRows:test.filter(x=>x)
  175. })).map(__rvn4=>({
  176. Custom:__rvn4.myRows[0].Custom,
  177. Custom2:__rvn4.myRows
  178. }))
  179. };
  180. }
  181. """);
  182. var result = engine.Evaluate("output()");
  183. var rows = result.AsObject()["Rows"];
  184. var custom = rows.AsArray()[0].AsObject()["Custom"];
  185. Assert.Equal(JsValue.Null, custom);
  186. }
  187. }
  188. file sealed class CustomString : JsString
  189. {
  190. private readonly string _value;
  191. private readonly bool _allowMaterialize;
  192. public CustomString(string value, bool allowMaterialize = false) : base(null)
  193. {
  194. _value = value;
  195. _allowMaterialize = allowMaterialize;
  196. }
  197. public override string ToString()
  198. {
  199. if (!_allowMaterialize)
  200. {
  201. // when called we know that we couldn't use fast paths
  202. throw new InvalidOperationException("I don't want to be materialized!");
  203. }
  204. return _value;
  205. }
  206. public override char this[int index] => _value[index];
  207. public override int Length => _value.Length;
  208. public override bool Equals(JsString obj)
  209. {
  210. return obj switch
  211. {
  212. CustomString customString => _value == customString._value,
  213. _ => _value == obj.ToString()
  214. };
  215. }
  216. protected override bool IsLooselyEqual(JsValue value)
  217. {
  218. return value switch
  219. {
  220. CustomString customString => _value == customString._value,
  221. JsString jsString => _value == jsString.ToString(),
  222. _ => base.IsLooselyEqual(value)
  223. };
  224. }
  225. public override int GetHashCode()
  226. {
  227. return _value.GetHashCode();
  228. }
  229. }
  230. file sealed class CustomNull : JsValue
  231. {
  232. public CustomNull() : base(Types.Null)
  233. {
  234. }
  235. public override object ToObject()
  236. {
  237. return null;
  238. }
  239. public override string ToString()
  240. {
  241. return "null";
  242. }
  243. }
  244. file sealed class CustomUndefined : JsValue
  245. {
  246. public CustomUndefined() : base(Types.Null)
  247. {
  248. }
  249. public override object ToObject()
  250. {
  251. return null;
  252. }
  253. public override string ToString()
  254. {
  255. return "null";
  256. }
  257. }
  258. file sealed class MyReferenceResolver : IReferenceResolver
  259. {
  260. public bool TryUnresolvableReference(Engine engine, Reference reference, out JsValue value)
  261. {
  262. JsValue referencedName = reference.ReferencedName;
  263. if (referencedName.IsString())
  264. {
  265. value = reference.IsPropertyReference ? JsValue.Undefined : JsValue.Null;
  266. return true;
  267. }
  268. throw new InvalidOperationException();
  269. }
  270. public bool TryPropertyReference(Engine engine, Reference reference, ref JsValue value)
  271. {
  272. return value.IsNull() || value.IsUndefined();
  273. }
  274. public bool TryGetCallable(Engine engine, object callee, out JsValue value)
  275. {
  276. value = new ClrFunction(engine, "function", static (_, _) => JsValue.Undefined);
  277. return true;
  278. }
  279. public bool CheckCoercible(JsValue value)
  280. {
  281. return true;
  282. }
  283. }