RegExpConstructor.cs 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. using System.Text.RegularExpressions;
  2. using Jint.Collections;
  3. using Jint.Native.Function;
  4. using Jint.Native.Object;
  5. using Jint.Native.Symbol;
  6. using Jint.Runtime;
  7. using Jint.Runtime.Descriptors;
  8. using Jint.Runtime.Interop;
  9. namespace Jint.Native.RegExp
  10. {
  11. public sealed class RegExpConstructor : Constructor
  12. {
  13. private static readonly JsString _functionName = new JsString("RegExp");
  14. internal RegExpConstructor(
  15. Engine engine,
  16. Realm realm,
  17. FunctionPrototype functionPrototype,
  18. ObjectPrototype objectPrototype)
  19. : base(engine, realm, _functionName)
  20. {
  21. _prototype = functionPrototype;
  22. PrototypeObject = new RegExpPrototype(engine, realm, this, objectPrototype);
  23. _length = new PropertyDescriptor(2, PropertyFlag.Configurable);
  24. _prototypeDescriptor = new PropertyDescriptor(PrototypeObject, PropertyFlag.AllForbidden);
  25. }
  26. internal RegExpPrototype PrototypeObject { get; }
  27. protected override void Initialize()
  28. {
  29. var symbols = new SymbolDictionary(1)
  30. {
  31. [GlobalSymbolRegistry.Species] = new GetSetPropertyDescriptor(get: new ClrFunction(_engine, "get [Symbol.species]", (thisObj, _) => thisObj, 0, PropertyFlag.Configurable), set: Undefined, PropertyFlag.Configurable)
  32. };
  33. SetSymbols(symbols);
  34. }
  35. protected internal override JsValue Call(JsValue thisObject, JsValue[] arguments)
  36. {
  37. return Construct(arguments, thisObject);
  38. }
  39. public ObjectInstance Construct(JsValue[] arguments)
  40. {
  41. return Construct(arguments, this);
  42. }
  43. /// <summary>
  44. /// https://tc39.es/ecma262/#sec-regexp-pattern-flags
  45. /// </summary>
  46. public override ObjectInstance Construct(JsValue[] arguments, JsValue newTarget)
  47. {
  48. var pattern = arguments.At(0);
  49. var flags = arguments.At(1);
  50. var patternIsRegExp = pattern.IsRegExp();
  51. if (newTarget.IsUndefined())
  52. {
  53. newTarget = this;
  54. if (patternIsRegExp && flags.IsUndefined())
  55. {
  56. var patternConstructor = pattern.Get(CommonProperties.Constructor);
  57. if (ReferenceEquals(newTarget, patternConstructor))
  58. {
  59. return (ObjectInstance) pattern;
  60. }
  61. }
  62. }
  63. JsValue p;
  64. JsValue f;
  65. if (pattern is JsRegExp regExpInstance)
  66. {
  67. p = regExpInstance.Source;
  68. f = flags.IsUndefined() ? regExpInstance.Flags : flags;
  69. }
  70. else if (patternIsRegExp)
  71. {
  72. p = pattern.Get(RegExpPrototype.PropertySource);
  73. f = flags.IsUndefined() ? pattern.Get(RegExpPrototype.PropertyFlags) : flags;
  74. }
  75. else
  76. {
  77. p = pattern;
  78. f = flags;
  79. }
  80. var r = RegExpAlloc(newTarget);
  81. return RegExpInitialize(r, p, f);
  82. }
  83. private JsRegExp RegExpInitialize(JsRegExp r, JsValue pattern, JsValue flags)
  84. {
  85. var p = pattern.IsUndefined() ? "" : TypeConverter.ToString(pattern);
  86. if (string.IsNullOrEmpty(p))
  87. {
  88. p = "(?:)";
  89. }
  90. var f = flags.IsUndefined() ? "" : TypeConverter.ToString(flags);
  91. var parserOptions = _engine.GetActiveParserOptions();
  92. try
  93. {
  94. var regExpParseResult = Tokenizer.AdaptRegExp(p, f, compiled: false, parserOptions.RegexTimeout,
  95. ecmaVersion: parserOptions.EcmaVersion,
  96. experimentalESFeatures: parserOptions.ExperimentalESFeatures);
  97. if (!regExpParseResult.Success)
  98. {
  99. ExceptionHelper.ThrowSyntaxError(_realm, $"Unsupported regular expression. {regExpParseResult.ConversionError!.Description}");
  100. }
  101. r.Value = regExpParseResult.Regex!;
  102. r.ParseResult = regExpParseResult;
  103. }
  104. catch (Exception ex)
  105. {
  106. ExceptionHelper.ThrowSyntaxError(_realm, ex.Message);
  107. }
  108. r.Flags = f;
  109. r.Source = p;
  110. RegExpInitialize(r);
  111. return r;
  112. }
  113. private JsRegExp RegExpAlloc(JsValue newTarget)
  114. {
  115. var r = OrdinaryCreateFromConstructor(
  116. newTarget,
  117. static intrinsics => intrinsics.RegExp.PrototypeObject,
  118. static (Engine engine, Realm _, object? _) => new JsRegExp(engine));
  119. return r;
  120. }
  121. public JsRegExp Construct(Regex regExp, string source, string flags, RegExpParseResult regExpParseResult = default)
  122. {
  123. var r = RegExpAlloc(this);
  124. r.Value = regExp;
  125. r.Source = source;
  126. r.Flags = flags;
  127. r.ParseResult = regExpParseResult;
  128. RegExpInitialize(r);
  129. return r;
  130. }
  131. private static void RegExpInitialize(JsRegExp r)
  132. {
  133. r.SetOwnProperty(JsRegExp.PropertyLastIndex, new PropertyDescriptor(0, PropertyFlag.OnlyWritable));
  134. }
  135. }
  136. }