1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586 |
- using System.Numerics;
- using System.Text.RegularExpressions;
- using Esprima;
- using Esprima.Ast;
- using Jint.Native;
- namespace Jint.Runtime.Interpreter.Expressions
- {
- internal sealed class JintLiteralExpression : JintExpression
- {
- private static readonly object _nullMarker = new();
- private JintLiteralExpression(Literal expression) : base(expression)
- {
- }
- internal static JintExpression Build(Literal expression)
- {
- var value = expression.AssociatedData ??= ConvertToJsValue(expression) ?? _nullMarker;
- if (value is JsValue constant)
- {
- return new JintConstantExpression(expression, constant);
- }
- return new JintLiteralExpression(expression);
- }
- internal static JsValue? ConvertToJsValue(Literal literal)
- {
- switch (literal.TokenType)
- {
- case TokenType.BooleanLiteral:
- return literal.BooleanValue!.Value ? JsBoolean.True : JsBoolean.False;
- case TokenType.NullLiteral:
- return JsValue.Null;
- case TokenType.NumericLiteral:
- {
- // unbox only once
- var numericValue = (double) literal.Value!;
- var intValue = (int) numericValue;
- return numericValue == intValue
- && (intValue != 0 || BitConverter.DoubleToInt64Bits(numericValue) != JsNumber.NegativeZeroBits)
- ? JsNumber.Create(intValue)
- : JsNumber.Create(numericValue);
- }
- case TokenType.StringLiteral:
- return JsString.Create((string) literal.Value!);
- case TokenType.BigIntLiteral:
- return JsBigInt.Create((BigInteger) literal.Value!);
- case TokenType.RegularExpression:
- break;
- }
- return null;
- }
- public override JsValue GetValue(EvaluationContext context)
- {
- // need to notify correct node when taking shortcut
- context.LastSyntaxElement = _expression;
- return ResolveValue(context);
- }
- protected override object EvaluateInternal(EvaluationContext context) => ResolveValue(context);
- private JsValue ResolveValue(EvaluationContext context)
- {
- var expression = (Literal) _expression;
- if (expression.TokenType == TokenType.RegularExpression)
- {
- var regExpLiteral = (RegExpLiteral) expression;
- var regExpParseResult = regExpLiteral.ParseResult;
- if (regExpParseResult.Success)
- {
- var regex = regExpLiteral.AssociatedData as Regex ?? regExpParseResult.Regex!;
- return context.Engine.Realm.Intrinsics.RegExp.Construct(regex, regExpLiteral.Regex.Pattern, regExpLiteral.Regex.Flags, regExpParseResult);
- }
- ExceptionHelper.ThrowSyntaxError(context.Engine.Realm, $"Unsupported regular expression. {regExpParseResult.ConversionError!.Description}");
- }
- return JsValue.FromObject(context.Engine, expression.Value);
- }
- }
- }
|