NullPropagation.cs 4.0 KB

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