luau/tests/TypePack.test.cpp
Alexander Youngblood 0bd9321957 Sync to upstream/release/653
## What's Changed?

* Optimized the vector dot product by up to 24%
* Allow for x/y/z/X/Y/Z vector field access by registering a `vector` metatable
with an `__index` method
* Fixed a bug preventing consistent recovery from parse errors in table types.
* Optimized `k*n` and `k+n` when types are known
* Allow fragment autocomplete to handle cases like the automatic insertion of
parens, keywords, strings, etc., while maintaining a correct relative positioning

 ### New Solver

* Added support for 'thread' and 'buffer' primitive types in Luau user-defined
type functions
* Allow for `nil` assignment to tables and classes with indexers

---------

Co-authored-by: Aaron Weiss <aaronweiss@roblox.com>
Co-authored-by: Andy Friesen <afriesen@roblox.com>
Co-authored-by: Aviral Goel <agoel@roblox.com>
Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com>
Co-authored-by: Varun Saini <vsaini@roblox.com>
Co-authored-by: Vighnesh Vijay <vvijay@roblox.com>
Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
2024-11-22 12:37:17 -08:00

214 lines
5.3 KiB
C++

// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#include "Luau/TypeInfer.h"
#include "Luau/Type.h"
#include "Fixture.h"
#include "doctest.h"
using namespace Luau;
namespace
{
struct TypePackFixture
{
TypePackFixture()
{
typeVars.emplace_back(new Type(PrimitiveType(PrimitiveType::NilType)));
typeVars.emplace_back(new Type(PrimitiveType(PrimitiveType::Boolean)));
typeVars.emplace_back(new Type(PrimitiveType(PrimitiveType::Number)));
typeVars.emplace_back(new Type(PrimitiveType(PrimitiveType::String)));
for (const auto& ptr : typeVars)
types.push_back(ptr.get());
}
TypePackId freshTypePack()
{
typePacks.emplace_back(new TypePackVar{FreeTypePack{TypeLevel{}}});
return typePacks.back().get();
}
TypePackId newTypePack(std::initializer_list<TypeId> types, std::optional<TypePackId> tail)
{
typePacks.emplace_back(new TypePackVar{TypePack{types, tail}});
return typePacks.back().get();
}
std::vector<std::unique_ptr<TypePackVar>> typePacks;
std::vector<std::unique_ptr<Type>> typeVars;
std::vector<TypeId> types;
};
} // namespace
TEST_SUITE_BEGIN("TypePackTests");
TEST_CASE_FIXTURE(TypePackFixture, "type_pack_hello")
{
auto tp = TypePackVar{TypePack{{types[0], types[1]}, std::nullopt}};
CHECK(tp == tp);
}
TEST_CASE_FIXTURE(TypePackFixture, "first_chases_Bound_TypePackVars")
{
Type nilType{PrimitiveType{PrimitiveType::NilType}};
auto tp1 = TypePackVar{TypePack{{&nilType}, std::nullopt}};
auto tp2 = TypePackVar{BoundTypePack{&tp1}};
auto tp3 = TypePackVar{TypePack{{}, &tp2}};
CHECK_EQ(first(&tp3), &nilType);
}
TEST_CASE_FIXTURE(TypePackFixture, "iterate_over_TypePack")
{
TypePackId typePack = newTypePack({types[0], types[1]}, std::nullopt);
std::vector<TypeId> res;
for (TypeId t : typePack)
res.push_back(t);
REQUIRE_EQ(2, res.size());
}
TEST_CASE_FIXTURE(TypePackFixture, "iterate_over_TypePack_with_2_links")
{
auto typePack1 = newTypePack({types[0], types[1]}, std::nullopt);
auto typePack2 = newTypePack({types[0], types[3]}, typePack1);
std::vector<TypeId> result;
for (TypeId ty : typePack2)
result.push_back(ty);
REQUIRE_EQ(4, result.size());
CHECK_EQ(types[0], result[0]);
CHECK_EQ(types[3], result[1]);
CHECK_EQ(types[0], result[2]);
CHECK_EQ(types[1], result[3]);
}
TEST_CASE_FIXTURE(TypePackFixture, "get_the_tail")
{
TypePackId freeTail = freshTypePack();
TypePackId typePack = newTypePack({types[0]}, freeTail);
auto it = begin(typePack);
auto endIt = end(typePack);
int count = 0;
while (it != endIt)
++count, ++it;
REQUIRE_EQ(1, count);
CHECK(it == end(typePack));
REQUIRE_EQ(it.tail(), freeTail);
}
TEST_CASE_FIXTURE(TypePackFixture, "tail_can_be_nullopt")
{
TypePackId typePack = newTypePack({types[0], types[0]}, std::nullopt);
auto it = end(typePack);
REQUIRE_EQ(std::nullopt, it.tail());
}
TEST_CASE_FIXTURE(TypePackFixture, "tail_is_end_for_free_TypePack")
{
TypePackId typePack = freshTypePack();
auto it = begin(typePack);
auto endIt = end(typePack);
while (it != endIt)
++it;
REQUIRE_EQ(typePack, it.tail());
}
TEST_CASE_FIXTURE(TypePackFixture, "skip_over_empty_head_typepack_with_tail")
{
TypePackId tailTP = newTypePack({types[2], types[3]}, std::nullopt);
TypePackId headTP = newTypePack({}, tailTP);
int count = 0;
for (TypeId ty : headTP)
{
(void)ty;
++count;
}
CHECK_EQ(2, count);
}
TEST_CASE_FIXTURE(TypePackFixture, "skip_over_empty_middle_link")
{
TypePackId tailTP = newTypePack({types[2], types[3]}, std::nullopt);
TypePackId middleTP = newTypePack({}, tailTP);
TypePackId headTP = newTypePack({types[0], types[1]}, middleTP);
int count = 0;
for (TypeId ty : headTP)
{
(void)ty;
++count;
}
CHECK_EQ(4, count);
}
TEST_CASE_FIXTURE(TypePackFixture, "follows_Bound_TypePacks")
{
TypePackId tailTP = newTypePack({types[2], types[3]}, std::nullopt);
TypePackId middleTP = freshTypePack();
*asMutable(middleTP) = Unifiable::Bound(tailTP);
TypePackId headTP = newTypePack({}, middleTP);
int count = 0;
for (TypeId ty : headTP)
{
(void)ty;
++count;
}
CHECK_EQ(2, count);
}
TEST_CASE_FIXTURE(TypePackFixture, "post_and_pre_increment")
{
TypePackId typePack = newTypePack({types[0], types[1], types[2], types[3]}, std::nullopt);
auto it1 = begin(typePack);
auto it2 = it1++;
auto it3 = ++it2;
CHECK_EQ(*it2, *it3);
}
TEST_CASE_FIXTURE(TypePackFixture, "std_distance")
{
TypePackId typePack = newTypePack({types[0], types[1], types[2], types[3]}, std::nullopt);
auto b = begin(typePack);
auto e = end(typePack);
CHECK_EQ(4, std::distance(b, e));
}
TEST_CASE("content_reassignment")
{
TypePackVar myError{ErrorTypePack{}, /*presistent*/ true};
TypeArena arena;
TypePackId futureError = arena.addTypePack(TypePackVar{FreeTypePack{TypeLevel{}}});
asMutable(futureError)->reassign(myError);
CHECK(get<ErrorTypePack>(futureError) != nullptr);
CHECK(!futureError->persistent);
CHECK(futureError->owningArena == &arena);
}
TEST_SUITE_END();