RavenApiUsageTests.cs 8.7 KB

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