Skip to content

Commit

Permalink
Clean up User-defined type related parser flag (#2697)
Browse files Browse the repository at this point in the history
* Remove `AllowParseAsTypeLiteral, AllowTypeLiteral` parser flag.
* AsType, IsType and ParseJSON overloads are part of Builtin Functions
* Add constant for Type Keyword.
* Add type arg functions to Built-in
* Refactor test
  • Loading branch information
adithyaselv authored Oct 22, 2024
1 parent 2434fd3 commit f473c48
Show file tree
Hide file tree
Showing 23 changed files with 113 additions and 144 deletions.
5 changes: 5 additions & 0 deletions releasenotes/releasenotes-1.3.0-rc.md
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,11 @@
`Patch(DS, table_of_rows_with_updates)`\
`Patch(Record, Updates1, Updates2,…)`

- ParseJSON, IsType, AsType (https://github.com/microsoft/Power-Fx/pull/2569): ParseJSON, IsType, AsType functions now supports in-lined and user-defined types as argument.\
`ParseJSON(Text, Type)`\
`IsType(UntypedObject, Type)`\
`AsType(UntypedObject, Type)`

## Other:
- Untyped object
- Read a field from an untyped object by index (https://github.com/microsoft/Power-Fx/pull/2555):
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ internal static string ConvertExpression(string expressionText, RecordType param

var formula = new Formula(expressionText, toDisplay ? CultureInfo.InvariantCulture : options?.Culture ?? CultureInfo.InvariantCulture);

formula.EnsureParsed(options.GetParserFlags());
formula.EnsureParsed(options.GetParserFlags(), flags);

var binding = TexlBinding.Run(
binderGlue,
Expand Down
6 changes: 0 additions & 6 deletions src/libraries/Microsoft.PowerFx.Core/Parser/ParserOptions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -47,11 +47,6 @@ public class ParserOptions
/// </summary>
public int MaxExpressionLength { get; set; }

/// <summary>
/// Flag for parse type literals.
/// </summary>
internal bool AllowParseAsTypeLiteral { get; set; }

/// <summary>
/// Allow parsing of attributes on user definitions
/// This is an early prototype, and so is internal.
Expand Down Expand Up @@ -88,7 +83,6 @@ internal ParseResult Parse(string script, Features features)
(NumberIsFloat ? TexlParser.Flags.NumberIsFloat : 0) |
(DisableReservedKeywords ? TexlParser.Flags.DisableReservedKeywords : 0) |
(TextFirst ? TexlParser.Flags.TextFirst : 0) |
(AllowParseAsTypeLiteral ? TexlParser.Flags.AllowTypeLiteral : 0) |
(features.PowerFxV1CompatibilityRules ? TexlParser.Flags.PFxV1 : 0);

var result = TexlParser.ParseScript(script, features, Culture, flags);
Expand Down
22 changes: 11 additions & 11 deletions src/libraries/Microsoft.PowerFx.Core/Parser/TexlParser.cs
Original file line number Diff line number Diff line change
Expand Up @@ -34,9 +34,6 @@ public enum Flags
// When specified, allows reserved keywords to be used as identifiers.
DisableReservedKeywords = 1 << 3,

// When specified, allows type literals to be parsed.
AllowTypeLiteral = 1 << 4,

// Text first. Implemented entirely in the Lexer.
TextFirst = 1 << 5,

Expand Down Expand Up @@ -90,16 +87,16 @@ private TexlParser(IReadOnlyList<Token> tokens, Flags flags, Features features =
/// </summary>
/// <param name="script">Script to be parsed.</param>
/// <param name="parserOptions">Options for parsing an expression.</param>
/// <param name="features">Power Fx feature flags.</param>
/// <returns><see cref="ParseUserDefinitionResult"/>.</returns>
public static ParseUserDefinitionResult ParseUserDefinitionScript(string script, ParserOptions parserOptions)
public static ParseUserDefinitionResult ParseUserDefinitionScript(string script, ParserOptions parserOptions, Features features = null)
{
Contracts.AssertValue(parserOptions);
var flags = Flags.NamedFormulas |
(parserOptions.NumberIsFloat ? Flags.NumberIsFloat : 0) |
(parserOptions.AllowParseAsTypeLiteral ? Flags.AllowTypeLiteral : 0) |
(parserOptions.AllowAttributes ? Flags.AllowAttributes : 0);
var formulaTokens = TokenizeScript(script, parserOptions.Culture, flags);
var parser = new TexlParser(formulaTokens, flags);
var parser = new TexlParser(formulaTokens, flags, features);

return parser.ParseUDFsAndNamedFormulas(script, parserOptions);
}
Expand Down Expand Up @@ -300,7 +297,7 @@ private ParseUserDefinitionResult ParseUDFsAndNamedFormulas(string script, Parse
continue;
}

if (_curs.TidCur == TokKind.ColonEqual && _flagsMode.Peek().HasFlag(Flags.AllowTypeLiteral))
if (_curs.TidCur == TokKind.ColonEqual && _features.IsUserDefinedTypesEnabled)
{
var declaration = script.Substring(declarationStart, _curs.TokCur.Span.Min - declarationStart);
_curs.TokMove();
Expand Down Expand Up @@ -1268,7 +1265,7 @@ private TexlNode ParseOperand()

if (AfterSpaceTokenId() == TokKind.ParenOpen)
{
if (ident.Token.As<IdentToken>().Name.Value == "Type" && _flagsMode.Peek().HasFlag(Flags.AllowTypeLiteral))
if (ident.Token.As<IdentToken>().Name.Value == LanguageConstants.TypeLiteralInvariantName && _features.IsUserDefinedTypesEnabled)
{
var typeLiteralNode = ParseTypeLiteral();

Expand Down Expand Up @@ -2048,11 +2045,13 @@ internal static string GetTokString(TokKind kind)
/// </summary>
/// <param name="text">Expression text to format.</param>
/// <param name="flags">Optional flags to customize the behavior of underlying lexer and parser. By default, expression chaining is enabled.</param>
/// <param name="features">Power Fx features.</param>
/// <returns>Formatted expression text.</returns>
public static string Format(string text, Flags flags = Flags.EnableExpressionChaining)
public static string Format(string text, Flags flags = Flags.EnableExpressionChaining, Features features = null)
{
var result = ParseScript(
text,
features ?? Features.None,
flags: flags);

// Can't pretty print a script with errors.
Expand All @@ -2064,11 +2063,12 @@ public static string Format(string text, Flags flags = Flags.EnableExpressionCha
return PrettyPrintVisitor.Format(result.Root, result.Before, result.After, text);
}

public static string FormatUserDefinitions(string text, ParserOptions options)
public static string FormatUserDefinitions(string text, ParserOptions options, Features features = null)
{
var result = ParseUserDefinitionScript(
text,
options);
options,
features ?? Features.None);

// Can't pretty print a script with errors.
if (result.HasErrors)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,8 @@ public class DefinitionsCheckResult : IOperationStatus

private ParseUserDefinitionResult _parse;

private readonly Features _features;

// Local symboltable to store new symbols in a given script and use in binding.
private readonly SymbolTable _localSymbolTable;

Expand All @@ -49,9 +51,15 @@ public class DefinitionsCheckResult : IOperationStatus
// All errors accumulated.
private readonly List<ExpressionError> _errors = new List<ExpressionError>();

public DefinitionsCheckResult()
public DefinitionsCheckResult()
: this(Features.PowerFxV1)
{
}

public DefinitionsCheckResult(Features features = null)
{
_localSymbolTable = new SymbolTable { DebugName = "LocalUserDefinitions" };
_features = features ?? Features.PowerFxV1;
}

internal DefinitionsCheckResult SetBindingInfo(ReadOnlySymbolTable symbols)
Expand Down Expand Up @@ -93,7 +101,7 @@ internal ParseUserDefinitionResult ApplyParse()

if (_parse == null)
{
_parse = UserDefinitions.Parse(_definitions, _parserOptions);
_parse = UserDefinitions.Parse(_definitions, _parserOptions, _features);

if (_parse.HasErrors)
{
Expand Down
4 changes: 2 additions & 2 deletions src/libraries/Microsoft.PowerFx.Core/Syntax/Formula.cs
Original file line number Diff line number Diff line change
Expand Up @@ -70,13 +70,13 @@ private void AssertValid()
// True if the formula has already been parsed.
public bool IsParsed => ParseTree != null;

public bool EnsureParsed(TexlParser.Flags flags)
public bool EnsureParsed(TexlParser.Flags flags, Features features = null)
{
AssertValid();

if (ParseTree == null)
{
var result = TexlParser.ParseScript(Script, loc: Loc, flags: flags);
var result = TexlParser.ParseScript(Script, features ?? Features.None, culture: Loc, flags: flags);
ApplyParse(result);
}

Expand Down
8 changes: 3 additions & 5 deletions src/libraries/Microsoft.PowerFx.Core/Syntax/TexlPretty.cs
Original file line number Diff line number Diff line change
Expand Up @@ -412,7 +412,7 @@ public override LazyList<string> Visit(TypeLiteralNode node, Precedence parentPr

result = result
.With(
"Type",
LanguageConstants.TypeLiteralInvariantName,
TexlLexer.PunctuatorParenOpen)
.With(node.TypeRoot.Accept(this, Precedence.Atomic))
.With(TexlLexer.PunctuatorParenClose);
Expand Down Expand Up @@ -466,9 +466,7 @@ private string SpacedOper(string op)

internal sealed class PrettyPrintVisitor : TexlFunctionalVisitor<LazyList<string>, PrettyPrintVisitor.Context>
{
private readonly string _script;

public const string TypeInvariantFunctionName = "Type";
private readonly string _script;

private static readonly Dictionary<BinaryOp, Precedence> BinaryPrecedence =
new Dictionary<BinaryOp, Precedence>()
Expand Down Expand Up @@ -542,7 +540,7 @@ public static string FormatUserDefinitions(ParseUserDefinitionResult result, str
case UserDefinitionType.DefinedType:
var type = result.DefinedTypes.First(type => type.Ident == name);

definitions.Add(declaration + $" = {TypeInvariantFunctionName}(" + string.Concat(visitor.CommentsOf(before).With(type.Type.Accept(visitor, new Context(0)).With(visitor.CommentsOf(after)))) + ")");
definitions.Add(declaration + $" = {LanguageConstants.TypeLiteralInvariantName}(" + string.Concat(visitor.CommentsOf(before).With(type.Type.Accept(visitor, new Context(0)).With(visitor.CommentsOf(after)))) + ")");
break;
default:
continue;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,14 +44,15 @@ private UserDefinitions(string script, ParserOptions parserOptions, Features fea
/// </summary>
/// <param name="script">Script with named formulas, user-defined functions and user-defined types.</param>
/// <param name="parserOptions">Options for parsing an expression.</param>
/// <param name="features">Power Fx feature flags.</param>
/// <returns><see cref="ParseUserDefinitionResult"/>.</returns>
public static ParseUserDefinitionResult Parse(string script, ParserOptions parserOptions)
public static ParseUserDefinitionResult Parse(string script, ParserOptions parserOptions, Features features = null)
{
var parseResult = TexlParser.ParseUserDefinitionScript(script, parserOptions);
var parseResult = TexlParser.ParseUserDefinitionScript(script, parserOptions, features);

if (parserOptions.AllowAttributes)
{
var userDefinitions = new UserDefinitions(script, parserOptions);
var userDefinitions = new UserDefinitions(script, parserOptions, features);
parseResult = userDefinitions.ProcessPartialAttributes(parseResult);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,8 @@ internal class BuiltinFunctionsCore
public static readonly TexlFunction And = _library.Add(new VariadicLogicalFunction(isAnd: true));
public static readonly TexlFunction Asin = _library.Add(new AsinFunction());
public static readonly TexlFunction AsinT = _library.Add(new AsinTableFunction());
public static readonly TexlFunction AsType = _library.Add(new AsTypeFunction());
public static readonly TexlFunction AsType = _library.Add(new AsTypeFunction());
public static readonly TexlFunction AsType_UO = _library.Add(new AsTypeFunction_UO());
public static readonly TexlFunction Atan = _library.Add(new AtanFunction());
public static readonly TexlFunction Atan2 = _library.Add(new Atan2Function());
public static readonly TexlFunction AtanT = _library.Add(new AtanTableFunction());
Expand Down Expand Up @@ -132,7 +133,8 @@ internal class BuiltinFunctionsCore
public static readonly TexlFunction IsError = _library.Add(new IsErrorFunction());
public static readonly TexlFunction IsNumeric = _library.Add(new IsNumericFunction());
public static readonly TexlFunction ISOWeekNum = _library.Add(new ISOWeekNumFunction());
public static readonly TexlFunction IsToday = _library.Add(new IsTodayFunction());
public static readonly TexlFunction IsToday = _library.Add(new IsTodayFunction());
public static readonly TexlFunction IsType_UO = _library.Add(new IsTypeFunction_UO());
public static readonly TexlFunction Language = _library.Add(new LanguageFunction());
public static readonly TexlFunction Last = _library.Add(new FirstLastFunction(isFirst: false));
public static readonly TexlFunction Last_UO = _library.Add(new FirstLastFunction_UO(isFirst: false));
Expand Down Expand Up @@ -231,6 +233,7 @@ internal class BuiltinFunctionsCore
public static readonly TexlFunction TrimT = _library.Add(new TrimTFunction());
public static readonly TexlFunction Trunc = _library.Add(new TruncFunction());
public static readonly TexlFunction TruncT = _library.Add(new TruncTableFunction());
public static readonly TexlFunction TypedParseJSON = _library.Add(new TypedParseJSONFunction());
public static readonly TexlFunction UniChar = _library.Add(new UniCharFunction());
public static readonly TexlFunction UniCharT = _library.Add(new UniCharTFunction());
public static readonly TexlFunction Upper = _library.Add(new LowerUpperFunction(isLower: false));
Expand Down Expand Up @@ -260,10 +263,7 @@ internal class BuiltinFunctionsCore
public static readonly TexlFunction UTCToday = _featureGateFunctions.Add(new UTCTodayFunction());
public static readonly TexlFunction BooleanL = _featureGateFunctions.Add(new BooleanLFunction());
public static readonly TexlFunction BooleanL_T = _featureGateFunctions.Add(new BooleanLFunction_T());
public static readonly TexlFunction Summarize = _featureGateFunctions.Add(new SummarizeFunction());
public static readonly TexlFunction AsType_UO = _featureGateFunctions.Add(new AsTypeFunction_UO());
public static readonly TexlFunction IsType_UO = _featureGateFunctions.Add(new IsTypeFunction_UO());
public static readonly TexlFunction TypedParseJSON = _featureGateFunctions.Add(new TypedParseJSONFunction());
public static readonly TexlFunction Summarize = _featureGateFunctions.Add(new SummarizeFunction());

// Slow API, only use for backward compatibility
#pragma warning disable CS0618 // Type or member is obsolete
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,5 +75,10 @@ internal class LanguageConstants
/// The string value representing the JSON format.
/// </summary>
internal const string JSONFormatEnumString = "JSONFormat";

/// <summary>
/// The string value representing the Type literal keyword.
/// </summary>
public const string TypeLiteralInvariantName = "Type";
}
}
3 changes: 1 addition & 2 deletions src/libraries/Microsoft.PowerFx.Interpreter/RecalcEngine.cs
Original file line number Diff line number Diff line change
Expand Up @@ -397,13 +397,12 @@ public void AddUserDefinitions(string script, CultureInfo parseCulture = null, A
var options = new ParserOptions()
{
AllowsSideEffects = false,
AllowParseAsTypeLiteral = true,
Culture = parseCulture ?? CultureInfo.InvariantCulture
};

var sb = new StringBuilder();

var checkResult = new DefinitionsCheckResult()
var checkResult = new DefinitionsCheckResult(this.Config.Features)
.SetText(script, options);

var parseResult = checkResult.ApplyParse();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -142,11 +142,10 @@ private bool ResolveBody(ModulePoco poco, SymbolTable moduleExports, ReadOnlySym
bool allowSideEffects = true;
var options = new ParserOptions
{
AllowParseAsTypeLiteral = true,
AllowsSideEffects = true
};

var parseResult = UserDefinitions.Parse(str, options);
var parseResult = UserDefinitions.Parse(str, options, Features.PowerFxV1);

var fragmentLocation = poco.Formulas.Location;

Expand Down
2 changes: 1 addition & 1 deletion src/libraries/Microsoft.PowerFx.Repl/Repl.cs
Original file line number Diff line number Diff line change
Expand Up @@ -503,7 +503,7 @@ await this.Output.WriteLineAsync($"Error: Can't set '{name}' to a Void value.",
var errors = check.ApplyErrors();
if (!check.IsSuccess)
{
var definitionsCheckResult = new DefinitionsCheckResult();
var definitionsCheckResult = new DefinitionsCheckResult(this.Engine.Config.Features);

definitionsCheckResult.SetText(expression, this.ParserOptions)
.ApplyParseErrors();
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
#SETUP: AllowTypeLiteral, TimeZoneInfo("Pacific Standard Time")

// Primitives
>> AsType(ParseJSON("987654321"), Number)
987654321
Expand All @@ -25,27 +23,18 @@ true
>> AsType(ParseJSON("""1984-01-01"""), Date)
Date(1984,1,1)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999Z"""), DateTime)
DateTime(1900,12,31,15,59,59,999)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999"""), DateTime)
DateTime(1900,12,31,23,59,59,999)
>> AsType(ParseJSON("""1900-12-31T23:59:59.999"""), Date)
Date(1900,12,31)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999+00:00"""), DateTime)
DateTime(1900,12,31,15,59,59,999)
>> AsType(ParseJSON("""1900-12-31T00:00:00.000Z"""), Date)
Date(1900,12,31)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999-08:00"""), DateTime)
>> AsType(ParseJSON("""1900-12-31T23:59:59.999"""), DateTime)
DateTime(1900,12,31,23,59,59,999)

>> AsType(ParseJSON("""1900-12-31"""), DateTime)
DateTime(1900,12,31,0,0,0,0)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999"""), Date)
Date(1900,12,31)

>> AsType(ParseJSON("""1900-12-31T00:00:00.000Z"""), Date)
Date(1900,12,31)

>> AsType(ParseJSON("""11:59:59.999"""), Time)
Time(11,59,59,999)

Expand All @@ -64,12 +53,6 @@ DateTime(1900,12,31,0,0,0,0)
>> AsType(ParseJSON("""1900-12-31T00:00:00.000-08:00"""), DateTimeTZInd)
DateTime(1900,12,31,8,0,0,0)

>> DateTimeValue(AsType(ParseJSON("""1900-12-31T00:00:00.000Z"""), UntypedObject))
DateTime(1900,12,30,16,0,0,0)

>> DateValue(AsType(ParseJSON("""1900-12-31T00:00:00.000Z"""), UntypedObject))
Date(1900,12,30)

>> Value(AsType(ParseJSON("42"), UntypedObject))
42

Expand Down Expand Up @@ -111,9 +94,6 @@ Error({Kind:ErrorKind.InvalidArgument})
>> AsType(ParseJSON("""42"""), Number)
Error({Kind:ErrorKind.InvalidArgument})

>> AsType(ParseJSON("""1900-12-31T24:59:59.1002Z"""), DateTime)
Error({Kind:ErrorKind.InvalidArgument})

>> AsType(ParseJSON("""24:59:59.12345678"""), Time)
Error({Kind:ErrorKind.InvalidArgument})

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
#SETUP: TimeZoneInfo("Pacific Standard Time")

>> AsType(ParseJSON("""1900-12-31T23:59:59.999Z"""), DateTime)
DateTime(1900,12,31,15,59,59,999)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999+00:00"""), DateTime)
DateTime(1900,12,31,15,59,59,999)

>> AsType(ParseJSON("""1900-12-31T23:59:59.999-08:00"""), DateTime)
DateTime(1900,12,31,23,59,59,999)

>> DateTimeValue(AsType(ParseJSON("""1900-12-31T00:00:00.000Z"""), UntypedObject))
DateTime(1900,12,30,16,0,0,0)

>> DateValue(AsType(ParseJSON("""1900-12-31T00:00:00.000Z"""), UntypedObject))
Date(1900,12,30)

>> AsType(ParseJSON("""1900-12-31T24:59:59.1002Z"""), DateTime)
Error({Kind:ErrorKind.InvalidArgument})
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
#SETUP: AllowTypeLiteral, TimeZoneInfo("Pacific Standard Time")

// Primitives
>> IsType(ParseJSON("987654321"), Number)
true
Expand Down
Loading

0 comments on commit f473c48

Please sign in to comment.