2022-04-29 02:04:52 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
|
|
|
|
/* Tests in this source file are meant to be a bellwether to verify that the numeric limits we've set are sufficient for
|
|
|
|
* most real-world scripts.
|
|
|
|
*
|
|
|
|
* If a change breaks a test in this source file, please don't adjust the flag values set in the fixture. Instead,
|
|
|
|
* consider it a latent performance problem by default.
|
|
|
|
*
|
|
|
|
* We should periodically revisit this to retest the limits.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Fixture.h"
|
|
|
|
|
|
|
|
#include "doctest.h"
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2022-04-29 02:04:52 +01:00
|
|
|
using namespace Luau;
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
|
|
|
LUAU_FASTFLAG(DebugLuauDeferredConstraintResolution)
|
|
|
|
|
2022-05-13 20:16:50 +01:00
|
|
|
struct LimitFixture : BuiltinsFixture
|
2022-04-29 02:04:52 +01:00
|
|
|
{
|
2022-05-06 00:52:48 +01:00
|
|
|
#if defined(_NOOPT) || defined(_DEBUG)
|
2023-12-02 02:04:44 +00:00
|
|
|
ScopedFastInt LuauTypeInferRecursionLimit{FInt::LuauTypeInferRecursionLimit, 100};
|
2022-04-29 02:04:52 +01:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2022-05-26 21:33:48 +01:00
|
|
|
template<typename T>
|
2022-04-29 02:04:52 +01:00
|
|
|
bool hasError(const CheckResult& result, T* = nullptr)
|
|
|
|
{
|
2024-08-02 00:25:12 +01:00
|
|
|
auto it = std::find_if(
|
|
|
|
result.errors.begin(),
|
|
|
|
result.errors.end(),
|
|
|
|
[](const TypeError& a)
|
|
|
|
{
|
|
|
|
return nullptr != get<T>(a);
|
|
|
|
}
|
|
|
|
);
|
2022-04-29 02:04:52 +01:00
|
|
|
return it != result.errors.end();
|
|
|
|
}
|
|
|
|
|
2022-05-06 00:52:48 +01:00
|
|
|
TEST_SUITE_BEGIN("RuntimeLimits");
|
2022-04-29 02:04:52 +01:00
|
|
|
|
2022-05-06 00:52:48 +01:00
|
|
|
TEST_CASE_FIXTURE(LimitFixture, "typescript_port_of_Result_type")
|
2022-04-29 02:04:52 +01:00
|
|
|
{
|
2023-08-04 18:01:35 +01:00
|
|
|
ScopedFastFlag sff[] = {
|
2023-12-02 02:04:44 +00:00
|
|
|
{FFlag::DebugLuauDeferredConstraintResolution, false},
|
2023-08-04 18:01:35 +01:00
|
|
|
};
|
|
|
|
|
2022-04-29 02:04:52 +01:00
|
|
|
constexpr const char* src = R"LUA(
|
|
|
|
--!strict
|
2022-05-06 00:52:48 +01:00
|
|
|
|
|
|
|
-- Big thanks to Dionysusnu by letting us use this code as part of our test suite!
|
|
|
|
-- https://github.com/Dionysusnu/rbxts-rust-classes
|
|
|
|
-- Licensed under the MPL 2.0: https://raw.githubusercontent.com/Dionysusnu/rbxts-rust-classes/master/LICENSE
|
|
|
|
|
2022-04-29 02:04:52 +01:00
|
|
|
local TS = _G[script]
|
|
|
|
local lazyGet = TS.import(script, script.Parent.Parent, "util", "lazyLoad").lazyGet
|
|
|
|
local unit = TS.import(script, script.Parent.Parent, "util", "Unit").unit
|
|
|
|
local Iterator
|
|
|
|
lazyGet("Iterator", function(c)
|
|
|
|
Iterator = c
|
|
|
|
end)
|
|
|
|
local Option
|
|
|
|
lazyGet("Option", function(c)
|
|
|
|
Option = c
|
|
|
|
end)
|
|
|
|
local Vec
|
|
|
|
lazyGet("Vec", function(c)
|
|
|
|
Vec = c
|
|
|
|
end)
|
|
|
|
local Result
|
|
|
|
do
|
|
|
|
Result = setmetatable({}, {
|
|
|
|
__tostring = function()
|
|
|
|
return "Result"
|
|
|
|
end,
|
|
|
|
})
|
|
|
|
Result.__index = Result
|
|
|
|
function Result.new(...)
|
|
|
|
local self = setmetatable({}, Result)
|
|
|
|
self:constructor(...)
|
|
|
|
return self
|
|
|
|
end
|
|
|
|
function Result:constructor(okValue, errValue)
|
|
|
|
self.okValue = okValue
|
|
|
|
self.errValue = errValue
|
|
|
|
end
|
|
|
|
function Result:ok(val)
|
|
|
|
return Result.new(val, nil)
|
|
|
|
end
|
|
|
|
function Result:err(val)
|
|
|
|
return Result.new(nil, val)
|
|
|
|
end
|
|
|
|
function Result:fromCallback(c)
|
|
|
|
local _0 = c
|
|
|
|
local _1, _2 = pcall(_0)
|
|
|
|
local result = _1 and {
|
|
|
|
success = true,
|
|
|
|
value = _2,
|
|
|
|
} or {
|
|
|
|
success = false,
|
|
|
|
error = _2,
|
|
|
|
}
|
|
|
|
return result.success and Result:ok(result.value) or Result:err(Option:wrap(result.error))
|
|
|
|
end
|
|
|
|
function Result:fromVoidCallback(c)
|
|
|
|
local _0 = c
|
|
|
|
local _1, _2 = pcall(_0)
|
|
|
|
local result = _1 and {
|
|
|
|
success = true,
|
|
|
|
value = _2,
|
|
|
|
} or {
|
|
|
|
success = false,
|
|
|
|
error = _2,
|
|
|
|
}
|
|
|
|
return result.success and Result:ok(unit()) or Result:err(Option:wrap(result.error))
|
|
|
|
end
|
|
|
|
Result.fromPromise = TS.async(function(self, p)
|
|
|
|
local _0, _1 = TS.try(function()
|
|
|
|
return TS.TRY_RETURN, { Result:ok(TS.await(p)) }
|
|
|
|
end, function(e)
|
|
|
|
return TS.TRY_RETURN, { Result:err(Option:wrap(e)) }
|
|
|
|
end)
|
|
|
|
if _0 then
|
|
|
|
return unpack(_1)
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
Result.fromVoidPromise = TS.async(function(self, p)
|
|
|
|
local _0, _1 = TS.try(function()
|
|
|
|
TS.await(p)
|
|
|
|
return TS.TRY_RETURN, { Result:ok(unit()) }
|
|
|
|
end, function(e)
|
|
|
|
return TS.TRY_RETURN, { Result:err(Option:wrap(e)) }
|
|
|
|
end)
|
|
|
|
if _0 then
|
|
|
|
return unpack(_1)
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
function Result:isOk()
|
|
|
|
return self.okValue ~= nil
|
|
|
|
end
|
|
|
|
function Result:isErr()
|
|
|
|
return self.errValue ~= nil
|
|
|
|
end
|
|
|
|
function Result:contains(x)
|
|
|
|
return self.okValue == x
|
|
|
|
end
|
|
|
|
function Result:containsErr(x)
|
|
|
|
return self.errValue == x
|
|
|
|
end
|
|
|
|
function Result:okOption()
|
|
|
|
return Option:wrap(self.okValue)
|
|
|
|
end
|
|
|
|
function Result:errOption()
|
|
|
|
return Option:wrap(self.errValue)
|
|
|
|
end
|
|
|
|
function Result:map(func)
|
|
|
|
return self:isOk() and Result:ok(func(self.okValue)) or Result:err(self.errValue)
|
|
|
|
end
|
|
|
|
function Result:mapOr(def, func)
|
|
|
|
local _0
|
|
|
|
if self:isOk() then
|
|
|
|
_0 = func(self.okValue)
|
|
|
|
else
|
|
|
|
_0 = def
|
|
|
|
end
|
|
|
|
return _0
|
|
|
|
end
|
|
|
|
function Result:mapOrElse(def, func)
|
|
|
|
local _0
|
|
|
|
if self:isOk() then
|
|
|
|
_0 = func(self.okValue)
|
|
|
|
else
|
|
|
|
_0 = def(self.errValue)
|
|
|
|
end
|
|
|
|
return _0
|
|
|
|
end
|
|
|
|
function Result:mapErr(func)
|
|
|
|
return self:isErr() and Result:err(func(self.errValue)) or Result:ok(self.okValue)
|
|
|
|
end
|
|
|
|
Result["and"] = function(self, other)
|
|
|
|
return self:isErr() and Result:err(self.errValue) or other
|
|
|
|
end
|
|
|
|
function Result:andThen(func)
|
|
|
|
return self:isErr() and Result:err(self.errValue) or func(self.okValue)
|
|
|
|
end
|
|
|
|
Result["or"] = function(self, other)
|
|
|
|
return self:isOk() and Result:ok(self.okValue) or other
|
|
|
|
end
|
|
|
|
function Result:orElse(other)
|
|
|
|
return self:isOk() and Result:ok(self.okValue) or other(self.errValue)
|
|
|
|
end
|
|
|
|
function Result:expect(msg)
|
|
|
|
if self:isOk() then
|
|
|
|
return self.okValue
|
|
|
|
else
|
|
|
|
error(msg)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
function Result:unwrap()
|
|
|
|
return self:expect("called `Result.unwrap()` on an `Err` value: " .. tostring(self.errValue))
|
|
|
|
end
|
|
|
|
function Result:unwrapOr(def)
|
|
|
|
local _0
|
|
|
|
if self:isOk() then
|
|
|
|
_0 = self.okValue
|
|
|
|
else
|
|
|
|
_0 = def
|
|
|
|
end
|
|
|
|
return _0
|
|
|
|
end
|
|
|
|
function Result:unwrapOrElse(gen)
|
|
|
|
local _0
|
|
|
|
if self:isOk() then
|
|
|
|
_0 = self.okValue
|
|
|
|
else
|
|
|
|
_0 = gen(self.errValue)
|
|
|
|
end
|
|
|
|
return _0
|
|
|
|
end
|
|
|
|
function Result:expectErr(msg)
|
|
|
|
if self:isErr() then
|
|
|
|
return self.errValue
|
|
|
|
else
|
|
|
|
error(msg)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
function Result:unwrapErr()
|
|
|
|
return self:expectErr("called `Result.unwrapErr()` on an `Ok` value: " .. tostring(self.okValue))
|
|
|
|
end
|
|
|
|
function Result:transpose()
|
|
|
|
return self:isOk() and self.okValue:map(function(some)
|
|
|
|
return Result:ok(some)
|
|
|
|
end) or Option:some(Result:err(self.errValue))
|
|
|
|
end
|
|
|
|
function Result:flatten()
|
|
|
|
return self:isOk() and Result.new(self.okValue.okValue, self.okValue.errValue) or Result:err(self.errValue)
|
|
|
|
end
|
|
|
|
function Result:match(ifOk, ifErr)
|
|
|
|
local _0
|
|
|
|
if self:isOk() then
|
|
|
|
_0 = ifOk(self.okValue)
|
|
|
|
else
|
|
|
|
_0 = ifErr(self.errValue)
|
|
|
|
end
|
|
|
|
return _0
|
|
|
|
end
|
|
|
|
function Result:asPtr()
|
|
|
|
local _0 = (self.okValue)
|
|
|
|
if _0 == nil then
|
|
|
|
_0 = (self.errValue)
|
|
|
|
end
|
|
|
|
return _0
|
|
|
|
end
|
|
|
|
end
|
|
|
|
local resultMeta = Result
|
|
|
|
resultMeta.__eq = function(a, b)
|
|
|
|
return b:match(function(ok)
|
|
|
|
return a:contains(ok)
|
|
|
|
end, function(err)
|
|
|
|
return a:containsErr(err)
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
resultMeta.__tostring = function(result)
|
|
|
|
return result:match(function(ok)
|
|
|
|
return "Result.ok(" .. tostring(ok) .. ")"
|
|
|
|
end, function(err)
|
|
|
|
return "Result.err(" .. tostring(err) .. ")"
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
return {
|
|
|
|
Result = Result,
|
|
|
|
}
|
|
|
|
)LUA";
|
|
|
|
|
2022-06-24 02:44:07 +01:00
|
|
|
CheckResult result = check(src);
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
CHECK(hasError<CodeTooComplex>(result));
|
2022-04-29 02:04:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_SUITE_END();
|