flag the changes

This commit is contained in:
jackdotink 2025-03-25 16:42:01 -05:00
parent d0d6d47924
commit 229a91e8a3
No known key found for this signature in database
2 changed files with 293 additions and 75 deletions

View file

@ -28,6 +28,7 @@ LUAU_FASTFLAGVARIABLE(LuauAstTypeGroup3)
LUAU_FASTFLAGVARIABLE(ParserNoErrorLimit) LUAU_FASTFLAGVARIABLE(ParserNoErrorLimit)
LUAU_FASTFLAGVARIABLE(LuauFixDoBlockEndLocation) LUAU_FASTFLAGVARIABLE(LuauFixDoBlockEndLocation)
LUAU_FASTFLAGVARIABLE(LuauParseOptionalAsNode) LUAU_FASTFLAGVARIABLE(LuauParseOptionalAsNode)
LUAU_FASTFLAGVARIABLE(LuauParseStringIndexer)
namespace Luau namespace Luau
{ {
@ -1260,30 +1261,106 @@ AstStat* Parser::parseDeclaration(const Location& start, const AstArray<AstAttr*
while (lexer.current().type != Lexeme::ReservedEnd) while (lexer.current().type != Lexeme::ReservedEnd)
{ {
// There are two possibilities: Either it's a property or a function. if (FFlag::LuauParseStringIndexer)
if (lexer.current().type == Lexeme::ReservedFunction)
{ {
props.push_back(parseDeclaredClassMethod()); // There are two possibilities: Either it's a property or a function.
} if (lexer.current().type == Lexeme::ReservedFunction)
else if (lexer.current().type == '[')
{
const Lexeme begin = lexer.current();
nextLexeme(); // [
if ((lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString) && lexer.lookahead().type == ']')
{ {
props.push_back(parseDeclaredClassMethod());
}
else if (lexer.current().type == '[')
{
const Lexeme begin = lexer.current();
nextLexeme(); // [
if ((lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString) && lexer.lookahead().type == ']')
{
const Location nameBegin = lexer.current().location;
std::optional<AstArray<char>> chars = parseCharArray();
const Location nameEnd = lexer.previousLocation();
expectMatchAndConsume(']', begin);
expectAndConsume(':', "property type annotation");
AstType* type = parseType();
// since AstName contains a char*, it can't contain null
bool containsNull = chars && (memchr(chars->data, 0, chars->size) != nullptr);
if (chars && !containsNull)
{
props.push_back(AstDeclaredClassProp{
AstName(chars->data), Location(nameBegin, nameEnd), type, false, Location(begin.location, lexer.previousLocation())
});
}
else
{
report(begin.location, "String literal contains malformed escape sequence or \\0");
}
}
else
{
if (indexer)
{
// maybe we don't need to parse the entire badIndexer...
// however, we either have { or [ to lint, not the entire table type or the bad indexer.
AstTableIndexer* badIndexer;
if (FFlag::LuauStoreCSTData)
badIndexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, begin).node;
else
badIndexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, begin);
// we lose all additional indexer expressions from the AST after error recovery here
report(badIndexer->location, "Cannot have more than one class indexer");
}
else
{
if (FFlag::LuauStoreCSTData)
indexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, begin).node;
else
indexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, begin);
}
}
}
else
{
Location propStart = lexer.current().location;
std::optional<Name> propName = parseNameOpt("property name");
if (!propName)
break;
expectAndConsume(':', "property type annotation");
AstType* propType = parseType();
props.push_back(
AstDeclaredClassProp{propName->name, propName->location, propType, false, Location(propStart, lexer.previousLocation())}
);
}
}
else
{
// There are two possibilities: Either it's a property or a function.
if (lexer.current().type == Lexeme::ReservedFunction)
{
props.push_back(parseDeclaredClassMethod());
}
else if (lexer.current().type == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
{
const Lexeme begin = lexer.current();
nextLexeme(); // [
const Location nameBegin = lexer.current().location; const Location nameBegin = lexer.current().location;
std::optional<AstArray<char>> chars = parseCharArray(); std::optional<AstArray<char>> chars = parseCharArray();
const Location nameEnd = lexer.previousLocation(); const Location nameEnd = lexer.previousLocation();
expectMatchAndConsume(']', begin); expectMatchAndConsume(']', begin);
expectAndConsume(':', "property type annotation"); expectAndConsume(':', "property type annotation");
AstType* type = parseType(); AstType* type = parseType();
// since AstName contains a char*, it can't contain null // since AstName contains a char*, it can't contain null
bool containsNull = chars && (memchr(chars->data, 0, chars->size) != nullptr); bool containsNull = chars && (memchr(chars->data, 0, chars->size) != nullptr);
if (chars && !containsNull) if (chars && !containsNull)
{ {
props.push_back(AstDeclaredClassProp{ props.push_back(AstDeclaredClassProp{
@ -1295,7 +1372,7 @@ AstStat* Parser::parseDeclaration(const Location& start, const AstArray<AstAttr*
report(begin.location, "String literal contains malformed escape sequence or \\0"); report(begin.location, "String literal contains malformed escape sequence or \\0");
} }
} }
else else if (lexer.current().type == '[')
{ {
if (indexer) if (indexer)
{ {
@ -1303,35 +1380,39 @@ AstStat* Parser::parseDeclaration(const Location& start, const AstArray<AstAttr*
// however, we either have { or [ to lint, not the entire table type or the bad indexer. // however, we either have { or [ to lint, not the entire table type or the bad indexer.
AstTableIndexer* badIndexer; AstTableIndexer* badIndexer;
if (FFlag::LuauStoreCSTData) if (FFlag::LuauStoreCSTData)
badIndexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, begin).node; // the last param in the parseTableIndexer is ignored
badIndexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, lexer.current()).node;
else else
badIndexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, begin); // the last param in the parseTableIndexer is ignored
badIndexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, lexer.current());
// we lose all additional indexer expressions from the AST after error recovery here // we lose all additional indexer expressions from the AST after error recovery here
report(badIndexer->location, "Cannot have more than one class indexer"); report(badIndexer->location, "Cannot have more than one class indexer");
} }
else else
{ {
if (FFlag::LuauStoreCSTData) if (FFlag::LuauStoreCSTData)
indexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, begin).node; // the last param in the parseTableIndexer is ignored
indexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, lexer.current()).node;
else else
indexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, begin); // the last param in the parseTableIndexer is ignored
indexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, lexer.current());
} }
} }
} else
else {
{ Location propStart = lexer.current().location;
Location propStart = lexer.current().location; std::optional<Name> propName = parseNameOpt("property name");
std::optional<Name> propName = parseNameOpt("property name");
if (!propName) if (!propName)
break; break;
expectAndConsume(':', "property type annotation"); expectAndConsume(':', "property type annotation");
AstType* propType = parseType(); AstType* propType = parseType();
props.push_back( props.push_back(
AstDeclaredClassProp{propName->name, propName->location, propType, false, Location(propStart, lexer.previousLocation())} AstDeclaredClassProp{propName->name, propName->location, propType, false, Location(propStart, lexer.previousLocation())}
); );
}
} }
} }
@ -1866,6 +1947,12 @@ std::pair<CstExprConstantString::QuoteStyle, unsigned int> Parser::extractString
// TableIndexer ::= `[' Type `]' `:' Type // TableIndexer ::= `[' Type `]' `:' Type
Parser::TableIndexerResult Parser::parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin) Parser::TableIndexerResult Parser::parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin)
{ {
if (!FFlag::LuauParseStringIndexer)
{
begin = lexer.current();
nextLexeme(); // [
}
AstType* index = parseType(); AstType* index = parseType();
Position indexerClosePosition = lexer.current().location.begin; Position indexerClosePosition = lexer.current().location.begin;
@ -1887,6 +1974,12 @@ Parser::TableIndexerResult Parser::parseTableIndexer(AstTableAccess access, std:
// Remove with FFlagLuauStoreCSTData // Remove with FFlagLuauStoreCSTData
AstTableIndexer* Parser::parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin) AstTableIndexer* Parser::parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin)
{ {
if (!FFlag::LuauParseStringIndexer)
{
begin = lexer.current();
nextLexeme(); // [
}
AstType* index = parseType(); AstType* index = parseType();
expectMatchAndConsume(']', begin); expectMatchAndConsume(']', begin);
@ -1938,29 +2031,148 @@ AstType* Parser::parseTableType(bool inDeclarationContext)
} }
} }
if (lexer.current().type == '[') { if (FFlag::LuauParseStringIndexer)
const Lexeme begin = lexer.current(); {
nextLexeme(); // [ if (lexer.current().type == '[')
{
const Lexeme begin = lexer.current();
nextLexeme(); // [
if ((lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString) && lexer.lookahead().type == ']')
{
CstExprConstantString::QuoteStyle style;
unsigned int blockDepth = 0;
if (FFlag::LuauStoreCSTData && options.storeCstData)
std::tie(style, blockDepth) = extractStringDetails();
AstArray<char> sourceString;
std::optional<AstArray<char>> chars = parseCharArray(options.storeCstData ? &sourceString : nullptr);
Position indexerClosePosition = lexer.current().location.begin;
expectMatchAndConsume(']', begin);
Position colonPosition = lexer.current().location.begin;
expectAndConsume(':', "table field");
AstType* type = parseType();
// since AstName contains a char*, it can't contain null
bool containsNull = chars && (memchr(chars->data, 0, chars->size) != nullptr);
if (chars && !containsNull)
{
props.push_back(AstTableProp{AstName(chars->data), begin.location, type, access, accessLocation});
if (FFlag::LuauStoreCSTData && options.storeCstData)
cstItems.push_back(CstTypeTable::Item{
CstTypeTable::Item::Kind::StringProperty,
begin.location.begin,
indexerClosePosition,
colonPosition,
tableSeparator(),
lexer.current().location.begin,
allocator.alloc<CstExprConstantString>(sourceString, style, blockDepth)
});
}
else
report(begin.location, "String literal contains malformed escape sequence or \\0");
}
else
{
if (indexer)
{
// maybe we don't need to parse the entire badIndexer...
// however, we either have { or [ to lint, not the entire table type or the bad indexer.
AstTableIndexer* badIndexer;
if (FFlag::LuauStoreCSTData)
badIndexer = parseTableIndexer(access, accessLocation, begin).node;
else
badIndexer = parseTableIndexer_DEPRECATED(access, accessLocation, begin);
// we lose all additional indexer expressions from the AST after error recovery here
report(badIndexer->location, "Cannot have more than one table indexer");
}
else
{
if (FFlag::LuauStoreCSTData)
{
auto tableIndexerResult = parseTableIndexer(access, accessLocation, begin);
indexer = tableIndexerResult.node;
if (options.storeCstData)
cstItems.push_back(CstTypeTable::Item{
CstTypeTable::Item::Kind::Indexer,
tableIndexerResult.indexerOpenPosition,
tableIndexerResult.indexerClosePosition,
tableIndexerResult.colonPosition,
tableSeparator(),
lexer.current().location.begin,
});
}
else
{
indexer = parseTableIndexer_DEPRECATED(access, accessLocation, begin);
}
}
}
}
else if (props.empty() && !indexer && !(lexer.current().type == Lexeme::Name && lexer.lookahead().type == ':'))
{
AstType* type = parseType();
// array-like table type: {T} desugars into {[number]: T}
isArray = true;
AstType* index = allocator.alloc<AstTypeReference>(type->location, std::nullopt, nameNumber, std::nullopt, type->location);
indexer = allocator.alloc<AstTableIndexer>(AstTableIndexer{index, type, type->location, access, accessLocation});
break;
}
else
{
std::optional<Name> name = parseNameOpt("table field");
if (!name)
break;
Position colonPosition = lexer.current().location.begin;
expectAndConsume(':', "table field");
AstType* type = parseType(inDeclarationContext);
props.push_back(AstTableProp{name->name, name->location, type, access, accessLocation});
if (FFlag::LuauStoreCSTData && options.storeCstData)
cstItems.push_back(CstTypeTable::Item{
CstTypeTable::Item::Kind::Property,
Position{0, 0},
Position{0, 0},
colonPosition,
tableSeparator(),
lexer.current().location.begin
});
}
}
else
{
if (lexer.current().type == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
{
const Lexeme begin = lexer.current();
nextLexeme(); // [
if ((lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString) && lexer.lookahead().type == ']') {
CstExprConstantString::QuoteStyle style; CstExprConstantString::QuoteStyle style;
unsigned int blockDepth = 0; unsigned int blockDepth = 0;
if (FFlag::LuauStoreCSTData && options.storeCstData) if (FFlag::LuauStoreCSTData && options.storeCstData)
std::tie(style, blockDepth) = extractStringDetails(); std::tie(style, blockDepth) = extractStringDetails();
AstArray<char> sourceString; AstArray<char> sourceString;
std::optional<AstArray<char>> chars = parseCharArray(options.storeCstData ? &sourceString : nullptr); std::optional<AstArray<char>> chars = parseCharArray(options.storeCstData ? &sourceString : nullptr);
Position indexerClosePosition = lexer.current().location.begin; Position indexerClosePosition = lexer.current().location.begin;
expectMatchAndConsume(']', begin); expectMatchAndConsume(']', begin);
Position colonPosition = lexer.current().location.begin; Position colonPosition = lexer.current().location.begin;
expectAndConsume(':', "table field"); expectAndConsume(':', "table field");
AstType* type = parseType(); AstType* type = parseType();
// since AstName contains a char*, it can't contain null // since AstName contains a char*, it can't contain null
bool containsNull = chars && (memchr(chars->data, 0, chars->size) != nullptr); bool containsNull = chars && (memchr(chars->data, 0, chars->size) != nullptr);
if (chars && !containsNull) if (chars && !containsNull)
{ {
props.push_back(AstTableProp{AstName(chars->data), begin.location, type, access, accessLocation}); props.push_back(AstTableProp{AstName(chars->data), begin.location, type, access, accessLocation});
@ -1978,7 +2190,7 @@ AstType* Parser::parseTableType(bool inDeclarationContext)
else else
report(begin.location, "String literal contains malformed escape sequence or \\0"); report(begin.location, "String literal contains malformed escape sequence or \\0");
} }
else else if (lexer.current().type == '[')
{ {
if (indexer) if (indexer)
{ {
@ -1986,10 +2198,12 @@ AstType* Parser::parseTableType(bool inDeclarationContext)
// however, we either have { or [ to lint, not the entire table type or the bad indexer. // however, we either have { or [ to lint, not the entire table type or the bad indexer.
AstTableIndexer* badIndexer; AstTableIndexer* badIndexer;
if (FFlag::LuauStoreCSTData) if (FFlag::LuauStoreCSTData)
badIndexer = parseTableIndexer(access, accessLocation, begin).node; // the last param in the parseTableIndexer is ignored
badIndexer = parseTableIndexer(access, accessLocation, lexer.current()).node;
else else
badIndexer = parseTableIndexer_DEPRECATED(access, accessLocation, begin); // the last param in the parseTableIndexer is ignored
badIndexer = parseTableIndexer_DEPRECATED(access, accessLocation, lexer.current());
// we lose all additional indexer expressions from the AST after error recovery here // we lose all additional indexer expressions from the AST after error recovery here
report(badIndexer->location, "Cannot have more than one table indexer"); report(badIndexer->location, "Cannot have more than one table indexer");
} }
@ -1997,7 +2211,8 @@ AstType* Parser::parseTableType(bool inDeclarationContext)
{ {
if (FFlag::LuauStoreCSTData) if (FFlag::LuauStoreCSTData)
{ {
auto tableIndexerResult = parseTableIndexer(access, accessLocation, begin); // the last param in the parseTableIndexer is ignored
auto tableIndexerResult = parseTableIndexer(access, accessLocation, lexer.current());
indexer = tableIndexerResult.node; indexer = tableIndexerResult.node;
if (options.storeCstData) if (options.storeCstData)
cstItems.push_back(CstTypeTable::Item{ cstItems.push_back(CstTypeTable::Item{
@ -2011,44 +2226,45 @@ AstType* Parser::parseTableType(bool inDeclarationContext)
} }
else else
{ {
indexer = parseTableIndexer_DEPRECATED(access, accessLocation, begin); // the last param in the parseTableIndexer is ignored
indexer = parseTableIndexer_DEPRECATED(access, accessLocation, lexer.current());
} }
} }
} }
} else if (props.empty() && !indexer && !(lexer.current().type == Lexeme::Name && lexer.lookahead().type == ':'))
else if (props.empty() && !indexer && !(lexer.current().type == Lexeme::Name && lexer.lookahead().type == ':')) {
{ AstType* type = parseType();
AstType* type = parseType();
// array-like table type: {T} desugars into {[number]: T} // array-like table type: {T} desugars into {[number]: T}
isArray = true; isArray = true;
AstType* index = allocator.alloc<AstTypeReference>(type->location, std::nullopt, nameNumber, std::nullopt, type->location); AstType* index = allocator.alloc<AstTypeReference>(type->location, std::nullopt, nameNumber, std::nullopt, type->location);
indexer = allocator.alloc<AstTableIndexer>(AstTableIndexer{index, type, type->location, access, accessLocation}); indexer = allocator.alloc<AstTableIndexer>(AstTableIndexer{index, type, type->location, access, accessLocation});
break;
}
else
{
std::optional<Name> name = parseNameOpt("table field");
if (!name)
break; break;
}
else
{
std::optional<Name> name = parseNameOpt("table field");
Position colonPosition = lexer.current().location.begin; if (!name)
expectAndConsume(':', "table field"); break;
AstType* type = parseType(inDeclarationContext); Position colonPosition = lexer.current().location.begin;
expectAndConsume(':', "table field");
props.push_back(AstTableProp{name->name, name->location, type, access, accessLocation}); AstType* type = parseType(inDeclarationContext);
if (FFlag::LuauStoreCSTData && options.storeCstData)
cstItems.push_back(CstTypeTable::Item{ props.push_back(AstTableProp{name->name, name->location, type, access, accessLocation});
CstTypeTable::Item::Kind::Property, if (FFlag::LuauStoreCSTData && options.storeCstData)
Position{0, 0}, cstItems.push_back(CstTypeTable::Item{
Position{0, 0}, CstTypeTable::Item::Kind::Property,
colonPosition, Position{0, 0},
tableSeparator(), Position{0, 0},
lexer.current().location.begin colonPosition,
}); tableSeparator(),
lexer.current().location.begin
});
}
} }
if (lexer.current().type == ',' || lexer.current().type == ';') if (lexer.current().type == ',' || lexer.current().type == ';')

View file

@ -24,6 +24,7 @@ LUAU_FASTFLAG(LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
LUAU_FASTFLAG(LuauAstTypeGroup3) LUAU_FASTFLAG(LuauAstTypeGroup3)
LUAU_FASTFLAG(LuauFixDoBlockEndLocation) LUAU_FASTFLAG(LuauFixDoBlockEndLocation)
LUAU_FASTFLAG(LuauParseOptionalAsNode) LUAU_FASTFLAG(LuauParseOptionalAsNode)
LUAU_FASTFLAG(LuauParseStringIndexer)
namespace namespace
{ {
@ -3940,6 +3941,7 @@ TEST_CASE_FIXTURE(Fixture, "parsing_type_suffix_for_return_type_with_variadic")
TEST_CASE_FIXTURE(Fixture, "parsing_string_union_indexers") TEST_CASE_FIXTURE(Fixture, "parsing_string_union_indexers")
{ {
ScopedFastFlag _{FFlag::LuauParseStringIndexer, true};
parse(R"(type foo = { ["bar" | "baz"]: number })"); parse(R"(type foo = { ["bar" | "baz"]: number })");
} }