luau/tests/TypeInfer.anyerror.test.cpp

447 lines
9.5 KiB
C++
Raw Normal View History

2022-03-18 00:06:25 +00:00
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#include "Luau/AstQuery.h"
#include "Luau/BuiltinDefinitions.h"
#include "Luau/Scope.h"
#include "Luau/TypeInfer.h"
2023-01-03 17:33:19 +00:00
#include "Luau/Type.h"
#include "Luau/VisitType.h"
2022-03-18 00:06:25 +00:00
#include "Fixture.h"
#include "doctest.h"
using namespace Luau;
2024-08-30 20:28:44 +01:00
LUAU_FASTFLAG(LuauSolverV2);
2023-07-28 12:37:00 +01:00
2022-03-18 00:06:25 +00:00
TEST_SUITE_BEGIN("TypeInferAnyError");
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_returns_any")
{
CheckResult result = check(R"(
function bar(): any
return true
end
local a
for b in bar do
a = b
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-07-19 18:21:40 +01:00
CHECK("any?" == toString(requireType("a")));
else
CHECK(builtinTypes->anyType == requireType("a"));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_returns_any2")
{
CheckResult result = check(R"(
function bar(): any
return true
end
local a
for b in bar() do
a = b
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-07-19 18:21:40 +01:00
CHECK("any?" == toString(requireType("a")));
else
CHECK("any" == toString(requireType("a")));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_is_any")
{
CheckResult result = check(R"(
2024-03-01 13:58:44 +00:00
local bar = nil :: any
2022-03-18 00:06:25 +00:00
local a
for b in bar do
a = b
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-07-19 18:21:40 +01:00
CHECK("any?" == toString(requireType("a")));
else
CHECK("any" == toString(requireType("a")));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_is_any2")
{
CheckResult result = check(R"(
2024-03-01 13:58:44 +00:00
local bar = nil :: any
2022-03-18 00:06:25 +00:00
local a
for b in bar() do
a = b
end
)");
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-07-19 18:21:40 +01:00
CHECK("any?" == toString(requireType("a")));
else
CHECK("any" == toString(requireType("a")));
2024-03-01 13:58:44 +00:00
}
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_is_any_pack")
{
CheckResult result = check(R"(
function bar(): ...any end
local a
for b in bar() do
a = b
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-07-19 18:21:40 +01:00
CHECK("any?" == toString(requireType("a")));
else
CHECK("any" == toString(requireType("a")));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_is_error")
{
CheckResult result = check(R"(
local a
for b in bar do
a = b
end
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
2024-03-01 13:58:44 +00:00
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-03-01 13:58:44 +00:00
{
// Bug: We do not simplify at the right time
CHECK_EQ("*error-type*?", toString(requireType("a")));
}
else
{
CHECK_EQ("*error-type*", toString(requireType("a")));
}
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "for_in_loop_iterator_is_error2")
{
CheckResult result = check(R"(
function bar(c) return c end
local a
for b in bar() do
a = b
end
)");
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-03-01 13:58:44 +00:00
{
// CLI-97375(awe): `bar()` is returning `nil` here, which isn't wrong necessarily,
// but then we're signaling an additional error for the access on `nil`.
LUAU_REQUIRE_ERROR_COUNT(2, result);
// Bug: We do not simplify at the right time
CHECK_EQ("*error-type*?", toString(requireType("a")));
}
else
{
LUAU_REQUIRE_ERROR_COUNT(1, result);
2022-03-18 00:06:25 +00:00
2024-03-01 13:58:44 +00:00
CHECK_EQ("*error-type*", toString(requireType("a")));
}
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "length_of_error_type_does_not_produce_an_error")
{
CheckResult result = check(R"(
local l = #this_is_not_defined
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
}
TEST_CASE_FIXTURE(Fixture, "indexing_error_type_does_not_produce_an_error")
{
CheckResult result = check(R"(
local originalReward = unknown.Parent.Reward:GetChildren()[1]
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
}
TEST_CASE_FIXTURE(Fixture, "dot_on_error_type_does_not_produce_an_error")
{
CheckResult result = check(R"(
local foo = (true).x
foo.x = foo.y
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
}
TEST_CASE_FIXTURE(Fixture, "any_type_propagates")
{
CheckResult result = check(R"(
local foo: any
local bar = foo:method("argument")
)");
LUAU_REQUIRE_NO_ERRORS(result);
CHECK_EQ("any", toString(requireType("bar")));
}
TEST_CASE_FIXTURE(Fixture, "can_subscript_any")
{
CheckResult result = check(R"(
local foo: any
local bar = foo[5]
)");
LUAU_REQUIRE_NO_ERRORS(result);
CHECK_EQ("any", toString(requireType("bar")));
}
// Not strictly correct: metatables permit overriding this
TEST_CASE_FIXTURE(Fixture, "can_get_length_of_any")
{
CheckResult result = check(R"(
2023-10-20 21:36:26 +01:00
local foo = ({} :: any)
2022-03-18 00:06:25 +00:00
local bar = #foo
)");
LUAU_REQUIRE_NO_ERRORS(result);
2023-01-03 17:33:19 +00:00
CHECK_EQ(PrimitiveType::Number, getPrimitiveType(requireType("bar")));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "assign_prop_to_table_by_calling_any_yields_any")
{
CheckResult result = check(R"(
local f: any
local T = {}
T.prop = f()
return T
)");
LUAU_REQUIRE_NO_ERRORS(result);
2023-01-03 17:33:19 +00:00
TableType* ttv = getMutable<TableType>(requireType("T"));
2022-03-18 00:06:25 +00:00
REQUIRE(ttv);
REQUIRE(ttv->props.count("prop"));
2023-04-28 12:55:55 +01:00
REQUIRE_EQ("any", toString(ttv->props["prop"].type()));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "quantify_any_does_not_bind_to_itself")
{
CheckResult result = check(R"(
local A : any
function A.B() end
A:C()
)");
LUAU_REQUIRE_NO_ERRORS(result);
TypeId aType = requireType("A");
2023-03-10 19:20:04 +00:00
CHECK_EQ(aType, builtinTypes->anyType);
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "calling_error_type_yields_error")
{
CheckResult result = check(R"(
local a = unknown.Parent.Reward.GetChildren()
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
UnknownSymbol* err = get<UnknownSymbol>(result.errors[0]);
REQUIRE(err != nullptr);
CHECK_EQ("unknown", err->name);
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2023-03-17 14:59:30 +00:00
CHECK_EQ("any", toString(requireType("a")));
else
CHECK_EQ("*error-type*", toString(requireType("a")));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "chain_calling_error_type_yields_error")
{
CheckResult result = check(R"(
local a = Utility.Create "Foo" {}
)");
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2023-03-17 14:59:30 +00:00
CHECK_EQ("any", toString(requireType("a")));
else
CHECK_EQ("*error-type*", toString(requireType("a")));
2022-03-18 00:06:25 +00:00
}
2022-05-13 20:16:50 +01:00
TEST_CASE_FIXTURE(BuiltinsFixture, "replace_every_free_type_when_unifying_a_complex_function_with_any")
2022-03-18 00:06:25 +00:00
{
CheckResult result = check(R"(
local a: any
local b
for _, i in pairs(a) do
b = i
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
2024-07-19 18:21:40 +01:00
2024-08-30 20:28:44 +01:00
if (FFlag::LuauSolverV2)
2024-07-19 18:21:40 +01:00
CHECK_EQ("any?", toString(requireType("b")));
else
CHECK_EQ("any", toString(requireType("b")));
2022-03-18 00:06:25 +00:00
}
TEST_CASE_FIXTURE(Fixture, "call_to_any_yields_any")
{
CheckResult result = check(R"(
local a: any
local b = a()
)");
REQUIRE_EQ("any", toString(requireType("b")));
}
TEST_CASE_FIXTURE(Fixture, "CheckMethodsOfAny")
{
CheckResult result = check(R"(
local x: any = {}
function x:y(z: number)
local s: string = z
end
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
}
TEST_CASE_FIXTURE(Fixture, "CheckMethodsOfError")
{
CheckResult result = check(R"(
local x = (true).foo
function x:y(z: number)
local s: string = z
end
)");
LUAU_REQUIRE_ERRORS(result);
}
2022-05-13 20:16:50 +01:00
TEST_CASE_FIXTURE(BuiltinsFixture, "metatable_of_any_can_be_a_table")
2022-03-18 00:06:25 +00:00
{
CheckResult result = check(R"(
--!strict
local T: any
T = {}
T.__index = T
function T.new(...)
local self = {}
setmetatable(self, T)
self:construct(...)
return self
end
function T:construct(index)
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
}
TEST_CASE_FIXTURE(Fixture, "type_error_addition")
{
CheckResult result = check(R"(
--!strict
local foo = makesandwich()
local bar = foo.nutrition + 100
)");
LUAU_REQUIRE_ERROR_COUNT(1, result);
// We should definitely get this error
CHECK_EQ("Unknown global 'makesandwich'", toString(result.errors[0]));
// We get this error if makesandwich() returns a free type
// CHECK_EQ("Unknown type used in + operation; consider adding a type annotation to 'foo'", toString(result.errors[1]));
}
TEST_CASE_FIXTURE(Fixture, "prop_access_on_any_with_other_options")
{
CheckResult result = check(R"(
local function f(thing: any | string)
local foo = thing.SomeRandomKey
end
)");
LUAU_REQUIRE_NO_ERRORS(result);
}
2022-09-02 00:00:14 +01:00
TEST_CASE_FIXTURE(BuiltinsFixture, "union_of_types_regression_test")
{
CheckResult result = check(R"(
--!strict
local stat
stat = stat and tonumber(stat) or stat
)");
LUAU_REQUIRE_NO_ERRORS(result);
}
2024-07-11 23:13:45 +01:00
TEST_CASE_FIXTURE(BuiltinsFixture, "table_of_any_calls")
{
CheckResult result = check(R"(
local function testFunc(input: {any})
end
local v = {true}
testFunc(v)
)");
LUAU_REQUIRE_NO_ERRORS(result);
}
2023-03-17 14:59:30 +00:00
TEST_CASE_FIXTURE(Fixture, "intersection_of_any_can_have_props")
{
// *blocked-130* ~ hasProp any & ~(false?), "_status"
CheckResult result = check(R"(
function foo(x: any, y)
if x then
return x._status
end
return y
end
)");
CHECK("(any, any) -> any" == toString(requireType("foo")));
}
2024-06-29 01:07:35 +01:00
TEST_CASE_FIXTURE(Fixture, "cast_to_table_of_any")
{
CheckResult result = check(R"(
local v = {true} :: {any}
)");
LUAU_REQUIRE_NO_ERRORS(result);
}
2022-03-18 00:06:25 +00:00
TEST_SUITE_END();