NullPropagation.cs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. using Jint.Native;
  2. using Jint.Runtime;
  3. using Jint.Runtime.Interop;
  4. using Jint.Runtime.References;
  5. namespace Jint.Tests.Runtime
  6. {
  7. public class NullPropagation
  8. {
  9. public class NullPropagationReferenceResolver : IReferenceResolver
  10. {
  11. public bool TryUnresolvableReference(Engine engine, Reference reference, out JsValue value)
  12. {
  13. value = reference.Base;
  14. return true;
  15. }
  16. public bool TryPropertyReference(Engine engine, Reference reference, ref JsValue value)
  17. {
  18. return value.IsNull() || value.IsUndefined();
  19. }
  20. public bool TryGetCallable(Engine engine, object callee, out JsValue value)
  21. {
  22. if (callee is Reference reference)
  23. {
  24. var name = reference.ReferencedName.AsString();
  25. if (name == "filter")
  26. {
  27. value = new ClrFunctionInstance(engine, "map", (thisObj, values) => engine.Realm.Intrinsics.Array.ArrayCreate(0));
  28. return true;
  29. }
  30. }
  31. value = new ClrFunctionInstance(engine, "anonymous", (thisObj, values) => thisObj);
  32. return true;
  33. }
  34. public bool CheckCoercible(JsValue value)
  35. {
  36. return true;
  37. }
  38. }
  39. [Fact]
  40. public void CanCallFilterOnNull()
  41. {
  42. var engine = new Engine(cfg => cfg.SetReferencesResolver(new NullPropagationReferenceResolver()));
  43. const string Script = @"
  44. var input = {};
  45. var output = { Tags : input.Tags.filter(x=>x!=null) };
  46. ";
  47. engine.Execute(Script);
  48. var output = engine.GetValue("output").AsObject();
  49. Assert.True(output.Get("Tags").IsArray());
  50. }
  51. [Fact]
  52. public void NullPropagationTest()
  53. {
  54. var engine = new Engine(cfg => cfg.SetReferencesResolver(new NullPropagationReferenceResolver()));
  55. const string Script = @"
  56. var input = {
  57. Address : null
  58. };
  59. var address = input.Address;
  60. var city = input.Address.City;
  61. var length = input.Address.City.length;
  62. var output = {
  63. Count1 : input.Address.City.length,
  64. Count2 : this.XYZ.length
  65. };
  66. ";
  67. engine.Execute(Script);
  68. var address = engine.GetValue("address");
  69. var city = engine.GetValue("city");
  70. var length = engine.GetValue("length");
  71. var output = engine.GetValue("output").AsObject();
  72. Assert.Equal(JsValue.Null, address);
  73. Assert.Equal(JsValue.Null, city);
  74. Assert.Equal(JsValue.Null, length);
  75. Assert.Equal(JsValue.Null, output.Get("Count1"));
  76. Assert.Equal(JsValue.Undefined, output.Get("Count2"));
  77. }
  78. [Fact]
  79. public void NullPropagationFromArg()
  80. {
  81. var engine = new Engine(cfg => cfg.SetReferencesResolver(new NullPropagationReferenceResolver()));
  82. const string Script = @"
  83. function test(arg) {
  84. return arg.Name;
  85. }
  86. function test2(arg) {
  87. return arg.Name.toUpperCase();
  88. }
  89. ";
  90. engine.Execute(Script);
  91. var result = engine.Invoke("test", JsValue.Null);
  92. Assert.Equal(JsValue.Null, result);
  93. result = engine.Invoke("test2", JsValue.Null);
  94. Assert.Equal(JsValue.Null, result);
  95. }
  96. [Fact]
  97. public void NullPropagationShouldNotAffectOperators()
  98. {
  99. var engine = new Engine(cfg => cfg.SetReferencesResolver(new NullPropagationReferenceResolver()));
  100. var jsObject = engine.Realm.Intrinsics.Object.Construct(Arguments.Empty);
  101. jsObject.Set("NullField", JsValue.Null);
  102. var script = @"
  103. this.is_nullfield_not_null = this.NullField !== null;
  104. this.is_notnullfield_not_null = this.NotNullField !== null;
  105. this.has_emptyfield_not_null = this.EmptyField !== null;
  106. ";
  107. var wrapperScript = $@"function ExecutePatchScript(docInner){{ (function(doc){{ {script} }}).apply(docInner); }};";
  108. engine.Execute(wrapperScript, "main.js");
  109. engine.Invoke("ExecutePatchScript", jsObject);
  110. Assert.False(jsObject.Get("is_nullfield_not_null").AsBoolean());
  111. Assert.True(jsObject.Get("is_notnullfield_not_null").AsBoolean());
  112. Assert.True(jsObject.Get("has_emptyfield_not_null").AsBoolean());
  113. }
  114. }
  115. }