mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 18:30:54 +01:00
Merge branch 'master' into merge
This commit is contained in:
commit
9ae5f377a4
5 changed files with 413 additions and 161 deletions
|
@ -33,6 +33,7 @@ LUAU_FASTINT(LuauCheckRecursionLimit)
|
||||||
LUAU_FASTFLAG(DebugLuauLogSolverToJson)
|
LUAU_FASTFLAG(DebugLuauLogSolverToJson)
|
||||||
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
||||||
LUAU_FASTFLAG(LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
LUAU_FASTFLAG(LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauPropagateExpectedTypesForCalls)
|
||||||
LUAU_FASTFLAG(DebugLuauGreedyGeneralization)
|
LUAU_FASTFLAG(DebugLuauGreedyGeneralization)
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTypesOnScope)
|
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
@ -2148,13 +2149,23 @@ InferencePack ConstraintGenerator::checkPack(const ScopePtr& scope, AstExprCall*
|
||||||
}
|
}
|
||||||
else if (i < exprArgs.size() - 1 || !(arg->is<AstExprCall>() || arg->is<AstExprVarargs>()))
|
else if (i < exprArgs.size() - 1 || !(arg->is<AstExprCall>() || arg->is<AstExprVarargs>()))
|
||||||
{
|
{
|
||||||
auto [ty, refinement] = check(scope, arg, /*expectedType*/ std::nullopt, /*forceSingleton*/ false, /*generalize*/ false);
|
std::optional<TypeId> expectedType = std::nullopt;
|
||||||
|
if (FFlag::LuauPropagateExpectedTypesForCalls && i < expectedTypesForCall.size())
|
||||||
|
{
|
||||||
|
expectedType = expectedTypesForCall[i];
|
||||||
|
}
|
||||||
|
auto [ty, refinement] = check(scope, arg, expectedType, /*forceSingleton*/ false, /*generalize*/ false);
|
||||||
args.push_back(ty);
|
args.push_back(ty);
|
||||||
argumentRefinements.push_back(refinement);
|
argumentRefinements.push_back(refinement);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
auto [tp, refis] = checkPack(scope, arg, {});
|
std::vector<std::optional<Luau::TypeId>> expectedTypes = {};
|
||||||
|
if (FFlag::LuauPropagateExpectedTypesForCalls && i < expectedTypesForCall.size())
|
||||||
|
{
|
||||||
|
expectedTypes.insert(expectedTypes.end(), expectedTypesForCall.begin() + i, expectedTypesForCall.end());
|
||||||
|
}
|
||||||
|
auto [tp, refis] = checkPack(scope, arg, expectedTypes);
|
||||||
argTail = tp;
|
argTail = tp;
|
||||||
argumentRefinements.insert(argumentRefinements.end(), refis.begin(), refis.end());
|
argumentRefinements.insert(argumentRefinements.end(), refis.begin(), refis.end());
|
||||||
}
|
}
|
||||||
|
|
|
@ -228,9 +228,9 @@ private:
|
||||||
Position colonPosition;
|
Position colonPosition;
|
||||||
};
|
};
|
||||||
|
|
||||||
TableIndexerResult parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation);
|
TableIndexerResult parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin);
|
||||||
// Remove with FFlagLuauStoreCSTData
|
// Remove with FFlagLuauStoreCSTData
|
||||||
AstTableIndexer* parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation);
|
AstTableIndexer* parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin);
|
||||||
|
|
||||||
AstTypeOrPack parseFunctionType(bool allowPack, const AstArray<AstAttr*>& attributes);
|
AstTypeOrPack parseFunctionType(bool allowPack, const AstArray<AstAttr*>& attributes);
|
||||||
AstType* parseFunctionTypeTail(
|
AstType* parseFunctionTypeTail(
|
||||||
|
|
|
@ -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,75 +1261,158 @@ 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 == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
|
|
||||||
{
|
|
||||||
const Lexeme begin = lexer.current();
|
|
||||||
nextLexeme(); // [
|
|
||||||
|
|
||||||
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{
|
props.push_back(parseDeclaredClassMethod());
|
||||||
AstName(chars->data), Location(nameBegin, nameEnd), type, false, Location(begin.location, lexer.previousLocation())
|
}
|
||||||
});
|
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
|
else
|
||||||
{
|
{
|
||||||
report(begin.location, "String literal contains malformed escape sequence or \\0");
|
Location propStart = lexer.current().location;
|
||||||
}
|
std::optional<Name> propName = parseNameOpt("property name");
|
||||||
}
|
|
||||||
else if (lexer.current().type == '[')
|
|
||||||
{
|
|
||||||
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).node;
|
|
||||||
else
|
|
||||||
badIndexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt);
|
|
||||||
|
|
||||||
// we lose all additional indexer expressions from the AST after error recovery here
|
if (!propName)
|
||||||
report(badIndexer->location, "Cannot have more than one class indexer");
|
break;
|
||||||
}
|
|
||||||
else
|
expectAndConsume(':', "property type annotation");
|
||||||
{
|
AstType* propType = parseType();
|
||||||
if (FFlag::LuauStoreCSTData)
|
props.push_back(
|
||||||
indexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt).node;
|
AstDeclaredClassProp{propName->name, propName->location, propType, false, Location(propStart, lexer.previousLocation())}
|
||||||
else
|
);
|
||||||
indexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Location propStart = lexer.current().location;
|
// There are two possibilities: Either it's a property or a function.
|
||||||
std::optional<Name> propName = parseNameOpt("property name");
|
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(); // [
|
||||||
|
|
||||||
if (!propName)
|
const Location nameBegin = lexer.current().location;
|
||||||
break;
|
std::optional<AstArray<char>> chars = parseCharArray();
|
||||||
|
|
||||||
expectAndConsume(':', "property type annotation");
|
const Location nameEnd = lexer.previousLocation();
|
||||||
AstType* propType = parseType();
|
|
||||||
props.push_back(
|
expectMatchAndConsume(']', begin);
|
||||||
AstDeclaredClassProp{propName->name, propName->location, propType, false, Location(propStart, lexer.previousLocation())}
|
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 (lexer.current().type == '[')
|
||||||
|
{
|
||||||
|
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)
|
||||||
|
// the last param in the parseTableIndexer is ignored since FFlagLuauParseStringIndexer is false here
|
||||||
|
badIndexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, lexer.current()).node;
|
||||||
|
else
|
||||||
|
// the last param in the parseTableIndexer is ignored since FFlagLuauParseStringIndexer is false here
|
||||||
|
badIndexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, lexer.current());
|
||||||
|
|
||||||
|
// 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)
|
||||||
|
// the last param in the parseTableIndexer is ignored since FFlagLuauParseStringIndexer is false here
|
||||||
|
indexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt, lexer.current()).node;
|
||||||
|
else
|
||||||
|
// the last param in the parseTableIndexer is ignored since FFlagLuauParseStringIndexer is false here
|
||||||
|
indexer = parseTableIndexer_DEPRECATED(AstTableAccess::ReadWrite, std::nullopt, lexer.current());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
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())}
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1861,10 +1945,13 @@ std::pair<CstExprConstantString::QuoteStyle, unsigned int> Parser::extractString
|
||||||
}
|
}
|
||||||
|
|
||||||
// TableIndexer ::= `[' Type `]' `:' Type
|
// TableIndexer ::= `[' Type `]' `:' Type
|
||||||
Parser::TableIndexerResult Parser::parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation)
|
Parser::TableIndexerResult Parser::parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin)
|
||||||
{
|
{
|
||||||
const Lexeme begin = lexer.current();
|
if (!FFlag::LuauParseStringIndexer)
|
||||||
nextLexeme(); // [
|
{
|
||||||
|
begin = lexer.current();
|
||||||
|
nextLexeme(); // [
|
||||||
|
}
|
||||||
|
|
||||||
AstType* index = parseType();
|
AstType* index = parseType();
|
||||||
|
|
||||||
|
@ -1885,10 +1972,13 @@ Parser::TableIndexerResult Parser::parseTableIndexer(AstTableAccess access, std:
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove with FFlagLuauStoreCSTData
|
// Remove with FFlagLuauStoreCSTData
|
||||||
AstTableIndexer* Parser::parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation)
|
AstTableIndexer* Parser::parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation, Lexeme begin)
|
||||||
{
|
{
|
||||||
const Lexeme begin = lexer.current();
|
if (!FFlag::LuauParseStringIndexer)
|
||||||
nextLexeme(); // [
|
{
|
||||||
|
begin = lexer.current();
|
||||||
|
nextLexeme(); // [
|
||||||
|
}
|
||||||
|
|
||||||
AstType* index = parseType();
|
AstType* index = parseType();
|
||||||
|
|
||||||
|
@ -1941,116 +2031,240 @@ AstType* Parser::parseTableType(bool inDeclarationContext)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lexer.current().type == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
|
if (FFlag::LuauParseStringIndexer)
|
||||||
{
|
{
|
||||||
const Lexeme begin = lexer.current();
|
if (lexer.current().type == '[')
|
||||||
nextLexeme(); // [
|
|
||||||
|
|
||||||
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});
|
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)
|
if (FFlag::LuauStoreCSTData && options.storeCstData)
|
||||||
cstItems.push_back(CstTypeTable::Item{
|
cstItems.push_back(CstTypeTable::Item{
|
||||||
CstTypeTable::Item::Kind::StringProperty,
|
CstTypeTable::Item::Kind::Property,
|
||||||
begin.location.begin,
|
Position{0, 0},
|
||||||
indexerClosePosition,
|
Position{0, 0},
|
||||||
colonPosition,
|
colonPosition,
|
||||||
tableSeparator(),
|
tableSeparator(),
|
||||||
lexer.current().location.begin,
|
lexer.current().location.begin
|
||||||
allocator.alloc<CstExprConstantString>(sourceString, style, blockDepth)
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else
|
|
||||||
report(begin.location, "String literal contains malformed escape sequence or \\0");
|
|
||||||
}
|
|
||||||
else if (lexer.current().type == '[')
|
|
||||||
{
|
|
||||||
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).node;
|
|
||||||
else
|
|
||||||
badIndexer = parseTableIndexer_DEPRECATED(access, accessLocation);
|
|
||||||
|
|
||||||
// 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);
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
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
|
else
|
||||||
{
|
{
|
||||||
std::optional<Name> name = parseNameOpt("table field");
|
if (lexer.current().type == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
|
||||||
|
{
|
||||||
|
const Lexeme begin = lexer.current();
|
||||||
|
nextLexeme(); // [
|
||||||
|
|
||||||
|
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 (lexer.current().type == '[')
|
||||||
|
{
|
||||||
|
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)
|
||||||
|
// the last param in the parseTableIndexer is ignored
|
||||||
|
badIndexer = parseTableIndexer(access, accessLocation, lexer.current()).node;
|
||||||
|
else
|
||||||
|
// 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
|
||||||
|
report(badIndexer->location, "Cannot have more than one table indexer");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (FFlag::LuauStoreCSTData)
|
||||||
|
{
|
||||||
|
// the last param in the parseTableIndexer is ignored
|
||||||
|
auto tableIndexerResult = parseTableIndexer(access, accessLocation, lexer.current());
|
||||||
|
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
|
||||||
|
{
|
||||||
|
// 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 == ':'))
|
||||||
|
{
|
||||||
|
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});
|
||||||
|
|
||||||
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 == ';')
|
||||||
|
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
@ -3938,5 +3939,10 @@ TEST_CASE_FIXTURE(Fixture, "parsing_type_suffix_for_return_type_with_variadic")
|
||||||
CHECK(result.errors.size() == 0);
|
CHECK(result.errors.size() == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(Fixture, "parsing_string_union_indexers")
|
||||||
|
{
|
||||||
|
ScopedFastFlag _{FFlag::LuauParseStringIndexer, true};
|
||||||
|
parse(R"(type foo = { ["bar" | "baz"]: number })");
|
||||||
|
}
|
||||||
|
|
||||||
TEST_SUITE_END();
|
TEST_SUITE_END();
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
using namespace Luau;
|
using namespace Luau;
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
|
LUAU_FASTFLAG(LuauPropagateExpectedTypesForCalls)
|
||||||
LUAU_FASTFLAG(LuauImproveTypePathsInErrors)
|
LUAU_FASTFLAG(LuauImproveTypePathsInErrors)
|
||||||
|
|
||||||
TEST_SUITE_BEGIN("TypeSingletons");
|
TEST_SUITE_BEGIN("TypeSingletons");
|
||||||
|
@ -152,6 +153,26 @@ TEST_CASE_FIXTURE(Fixture, "overloaded_function_call_with_singletons")
|
||||||
LUAU_REQUIRE_NO_ERRORS(result);
|
LUAU_REQUIRE_NO_ERRORS(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(Fixture, "overloaded_function_resolution_singleton_parameters")
|
||||||
|
{
|
||||||
|
ScopedFastFlag sff{FFlag::LuauPropagateExpectedTypesForCalls, true};
|
||||||
|
|
||||||
|
CheckResult result = check(R"(
|
||||||
|
type A = ("A") -> string
|
||||||
|
type B = ("B") -> number
|
||||||
|
|
||||||
|
local function foo(f: A & B)
|
||||||
|
return f("A"), f("B")
|
||||||
|
end
|
||||||
|
)");
|
||||||
|
LUAU_REQUIRE_NO_ERRORS(result);
|
||||||
|
TypeId t = requireType("foo");
|
||||||
|
const FunctionType* fooType = get<FunctionType>(requireType("foo"));
|
||||||
|
REQUIRE(fooType != nullptr);
|
||||||
|
|
||||||
|
CHECK(toString(t) == "(((\"A\") -> string) & ((\"B\") -> number)) -> (string, number)");
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(Fixture, "overloaded_function_call_with_singletons_mismatch")
|
TEST_CASE_FIXTURE(Fixture, "overloaded_function_call_with_singletons_mismatch")
|
||||||
{
|
{
|
||||||
DOES_NOT_PASS_NEW_SOLVER_GUARD();
|
DOES_NOT_PASS_NEW_SOLVER_GUARD();
|
||||||
|
|
Loading…
Add table
Reference in a new issue