mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-04 10:50:54 +01:00
TyperChecker2 fix which ended up as UnionType Table Indexer Support
This commit is contained in:
parent
e8a7acb802
commit
9f94c49674
2 changed files with 133 additions and 0 deletions
|
@ -3015,6 +3015,60 @@ PropertyType TypeChecker2::hasIndexTypeFromType(
|
||||||
if (isPrim(indexType, PrimitiveType::String))
|
if (isPrim(indexType, PrimitiveType::String))
|
||||||
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
||||||
// If the indexer looks like { [any] : _} - the prop lookup should be allowed!
|
// If the indexer looks like { [any] : _} - the prop lookup should be allowed!
|
||||||
|
else if (auto indexTy = get<UnionType>(indexType))
|
||||||
|
{
|
||||||
|
// If the Indexer is a UnionType
|
||||||
|
// TODO: This is looping everytime, on every request, through everything?
|
||||||
|
// That doesn't sound good. But I am not sure if the other parts do that as well.
|
||||||
|
auto visitOption = [&](TypeId& option, const std::string& prop) -> PropertyType
|
||||||
|
{
|
||||||
|
if (auto tySingleton = get<SingletonType>(option))
|
||||||
|
if (auto ty = get<StringSingleton>(tySingleton))
|
||||||
|
if (ty->value == prop)
|
||||||
|
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
||||||
|
|
||||||
|
// If all fails
|
||||||
|
return {NormalizationResult::False, {}};
|
||||||
|
};
|
||||||
|
|
||||||
|
auto visitUnionOptions = [&](UnionType unionTy, const std::string& prop) -> PropertyType
|
||||||
|
{
|
||||||
|
for (TypeId option : unionTy.options)
|
||||||
|
{
|
||||||
|
option = follow(option);
|
||||||
|
|
||||||
|
// No recurse
|
||||||
|
if (get<UnionType>(option))
|
||||||
|
return {NormalizationResult::False, {}};
|
||||||
|
|
||||||
|
PropertyType result = visitOption(option, prop);
|
||||||
|
if (result.present == NormalizationResult::False)
|
||||||
|
continue; // Continue searching if it failed
|
||||||
|
|
||||||
|
// Otherwise return
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If all fails
|
||||||
|
return {NormalizationResult::False, {}};
|
||||||
|
};
|
||||||
|
|
||||||
|
for (TypeId option : indexTy->options)
|
||||||
|
{
|
||||||
|
option = follow(option);
|
||||||
|
|
||||||
|
if (auto unionTy = get<UnionType>(option))
|
||||||
|
{
|
||||||
|
PropertyType result = visitUnionOptions(*unionTy, prop);
|
||||||
|
if (result.present == NormalizationResult::False)
|
||||||
|
continue; // Continue searching if it failed
|
||||||
|
else
|
||||||
|
return result; // Otherwise return
|
||||||
|
}
|
||||||
|
|
||||||
|
return visitOption(option, prop);
|
||||||
|
}
|
||||||
|
}
|
||||||
else if (get<AnyType>(indexType) || get<UnknownType>(indexType))
|
else if (get<AnyType>(indexType) || get<UnknownType>(indexType))
|
||||||
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
||||||
}
|
}
|
||||||
|
|
|
@ -690,6 +690,85 @@ TEST_CASE_FIXTURE(ClassFixture, "vector2_multiply_is_overloaded")
|
||||||
CHECK("mul<Vector2, string>" == toString(requireType("v4")));
|
CHECK("mul<Vector2, string>" == toString(requireType("v4")));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(BuiltinsFixture, "union_used_as_table_indexer_assignProp_no_error")
|
||||||
|
{
|
||||||
|
if (!FFlag::LuauSolverV2)
|
||||||
|
return;
|
||||||
|
|
||||||
|
CheckResult check1 = check(R"(
|
||||||
|
--!strict
|
||||||
|
|
||||||
|
type NumberTable = {
|
||||||
|
One: number,
|
||||||
|
Two: number,
|
||||||
|
Three: number
|
||||||
|
}
|
||||||
|
|
||||||
|
type Indexables = "One" | "Two" | "Three" | typeof(5)
|
||||||
|
|
||||||
|
type Test = {
|
||||||
|
TestKey: number,
|
||||||
|
[Indexables | "Four"]: number
|
||||||
|
}
|
||||||
|
|
||||||
|
local tbl = {} :: Test
|
||||||
|
|
||||||
|
-- This was just used to compare it with index
|
||||||
|
-- type TestIndex = index<typeof(tbl), "Three">
|
||||||
|
|
||||||
|
tbl["Three"] = 3;
|
||||||
|
tbl.Three = 3;
|
||||||
|
tbl.Four = 4;
|
||||||
|
tbl.TestKey = 5;
|
||||||
|
)");
|
||||||
|
|
||||||
|
// It should be possible to do "tbl.Three" without erroring.
|
||||||
|
LUAU_REQUIRE_NO_ERRORS(check1);
|
||||||
|
|
||||||
|
// auto test1 = requireTypeAlias("TestIndex");
|
||||||
|
// auto test2 = requireType("tbl");
|
||||||
|
|
||||||
|
// TODO: This doesn't include the autocomplete fix
|
||||||
|
// That has to be still made.
|
||||||
|
// It is supposed to include the indexes from the UnionType
|
||||||
|
// but it is possible to have more than just that
|
||||||
|
// It's also to note that strings may not be the only indexable things
|
||||||
|
// auto ac = autocomplete('1');
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(BuiltinsFixture, "keyof_used_as_table_indexer_assignProp_no_error")
|
||||||
|
{
|
||||||
|
if (!FFlag::LuauSolverV2)
|
||||||
|
return;
|
||||||
|
|
||||||
|
CheckResult check1 = check(R"(
|
||||||
|
--!strict
|
||||||
|
|
||||||
|
type NumberTable = {
|
||||||
|
One: number,
|
||||||
|
Two: number,
|
||||||
|
Three: number
|
||||||
|
}
|
||||||
|
|
||||||
|
type Indexables = keyof<NumberTable>
|
||||||
|
|
||||||
|
type Test = {
|
||||||
|
TestKey: number,
|
||||||
|
[Indexables | "Four"]: number
|
||||||
|
}
|
||||||
|
|
||||||
|
local tbl = {} :: Test
|
||||||
|
|
||||||
|
tbl["Three"] = 3;
|
||||||
|
tbl.Three = 3;
|
||||||
|
tbl.Four = 4;
|
||||||
|
tbl.TestKey = 5;
|
||||||
|
)");
|
||||||
|
|
||||||
|
// It should be possible to do "tbl.Three" without erroring.
|
||||||
|
LUAU_REQUIRE_NO_ERRORS(check1);
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(BuiltinsFixture, "keyof_rfc_example")
|
TEST_CASE_FIXTURE(BuiltinsFixture, "keyof_rfc_example")
|
||||||
{
|
{
|
||||||
if (!FFlag::LuauSolverV2)
|
if (!FFlag::LuauSolverV2)
|
||||||
|
|
Loading…
Add table
Reference in a new issue