RavenApiUsageTests.cs 9.3 KB

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