mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-05 11:20:54 +01:00
## New Type Solver 1. Update resolved types for singleton unions and intersections to avoid crashing when type checking type assertions. 2. Generalize free return type pack of a function type inferred at call site to ensure that the free type does not leak to another module. 3. Fix crash from cyclic indexers by reducing if possible or producing an error otherwise. 4. Fix handling of irreducible type functions to prevent type inference from failing. 5. Fix handling of recursive metatables to avoid infinite recursion. ## New and Old Type Solver Fix accidental capture of all exceptions in multi-threaded typechecking by converting all typechecking exceptions to `InternalCompilerError` and only capturing those. ## Fragment Autocomplete 1. Add a block based diff algorithm based on class index and span for re-typechecking. This reduces the granularity of fragment autocomplete to avoid flakiness when the fragment does not have enough type information. 2. Fix bugs arising from incorrect scope selection for autocompletion. ## Roundtrippable AST Store type alias location in `TypeFun` class to ensure it is accessible for exported types as part of the public interface. ## Build System 1. Bump minimum supported CMake version to 3.10 since GitHub is phasing out the currently supported minimum version 3.0, released 11 years ago. 2. Fix compilation when `HARDSTACKTESTS` is enabled. ## Miscellaneous Flag removals and cleanup of unused code. ## Internal Contributors Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Talha Pathan <tpathan@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> ## External Contributors Thanks to [@grh-official](https://github.com/grh-official) for PR #1759 **Full Changelog**: https://github.com/luau-lang/luau/compare/0.667...0.668 --------- Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Varun Saini <61795485+vrn-sn@users.noreply.github.com> Co-authored-by: Alexander Youngblood <ayoungblood@roblox.com> Co-authored-by: Menarul Alam <malam@roblox.com> Co-authored-by: Vighnesh <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com>
3600 lines
97 KiB
C++
3600 lines
97 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/FragmentAutocomplete.h"
|
|
#include "Fixture.h"
|
|
#include "Luau/Ast.h"
|
|
#include "Luau/AstQuery.h"
|
|
#include "Luau/Autocomplete.h"
|
|
#include "Luau/BuiltinDefinitions.h"
|
|
#include "Luau/Common.h"
|
|
#include "Luau/Frontend.h"
|
|
#include "Luau/AutocompleteTypes.h"
|
|
#include "Luau/ToString.h"
|
|
#include "Luau/Type.h"
|
|
#include "ScopedFlags.h"
|
|
|
|
#include <algorithm>
|
|
#include <chrono>
|
|
#include <ctime>
|
|
#include <iomanip>
|
|
#include <iostream>
|
|
#include <memory>
|
|
#include <optional>
|
|
|
|
|
|
using namespace Luau;
|
|
|
|
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
|
LUAU_FASTINT(LuauParseErrorLimit)
|
|
LUAU_FASTFLAG(LuauCloneIncrementalModule)
|
|
|
|
LUAU_FASTFLAG(LuauMixedModeDefFinderTraversesTypeOf)
|
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
|
|
|
LUAU_FASTFLAG(LuauBetterReverseDependencyTracking)
|
|
LUAU_FASTFLAG(LuauAutocompleteUsesModuleForTypeCompatibility)
|
|
LUAU_FASTFLAG(LuauBetterCursorInCommentDetection)
|
|
LUAU_FASTFLAG(LuauAllFreeTypesHaveScopes)
|
|
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope)
|
|
LUAU_FASTFLAG(LuauClonedTableAndFunctionTypesMustHaveScopes)
|
|
LUAU_FASTFLAG(LuauDisableNewSolverAssertsInMixedMode)
|
|
LUAU_FASTFLAG(LuauCloneTypeAliasBindings)
|
|
LUAU_FASTFLAG(LuauDoNotClonePersistentBindings)
|
|
LUAU_FASTFLAG(LuauCloneReturnTypePack)
|
|
LUAU_FASTFLAG(LuauIncrementalAutocompleteDemandBasedCloning)
|
|
LUAU_FASTFLAG(LuauUserTypeFunTypecheck)
|
|
LUAU_FASTFLAG(LuauBetterScopeSelection)
|
|
LUAU_FASTFLAG(LuauBlockDiffFragmentSelection)
|
|
|
|
static std::optional<AutocompleteEntryMap> nullCallback(std::string tag, std::optional<const ClassType*> ptr, std::optional<std::string> contents)
|
|
{
|
|
return std::nullopt;
|
|
}
|
|
|
|
static FrontendOptions getOptions()
|
|
{
|
|
FrontendOptions options;
|
|
options.retainFullTypeGraphs = true;
|
|
|
|
if (!FFlag::LuauSolverV2)
|
|
options.forAutocomplete = true;
|
|
|
|
options.runLintChecks = false;
|
|
|
|
return options;
|
|
}
|
|
|
|
static ModuleResolver& getModuleResolver(Luau::Frontend& frontend)
|
|
{
|
|
return FFlag::LuauSolverV2 ? frontend.moduleResolver : frontend.moduleResolverForAutocomplete;
|
|
}
|
|
|
|
template<class BaseType>
|
|
struct FragmentAutocompleteFixtureImpl : BaseType
|
|
{
|
|
static_assert(std::is_base_of_v<Fixture, BaseType>, "BaseType must be a descendant of Fixture");
|
|
|
|
ScopedFastFlag luauAutocompleteRefactorsForIncrementalAutocomplete{FFlag::LuauAutocompleteRefactorsForIncrementalAutocomplete, true};
|
|
ScopedFastFlag luauFreeTypesMustHaveBounds{FFlag::LuauFreeTypesMustHaveBounds, true};
|
|
ScopedFastFlag luauCloneIncrementalModule{FFlag::LuauCloneIncrementalModule, true};
|
|
ScopedFastFlag luauAllFreeTypesHaveScopes{FFlag::LuauAllFreeTypesHaveScopes, true};
|
|
ScopedFastFlag luauClonedTableAndFunctionTypesMustHaveScopes{FFlag::LuauClonedTableAndFunctionTypesMustHaveScopes, true};
|
|
ScopedFastFlag luauDisableNewSolverAssertsInMixedMode{FFlag::LuauDisableNewSolverAssertsInMixedMode, true};
|
|
ScopedFastFlag luauCloneTypeAliasBindings{FFlag::LuauCloneTypeAliasBindings, true};
|
|
ScopedFastFlag luauDoNotClonePersistentBindings{FFlag::LuauDoNotClonePersistentBindings, true};
|
|
ScopedFastFlag luauCloneReturnTypePack{FFlag::LuauCloneReturnTypePack, true};
|
|
ScopedFastFlag luauIncrementalAutocompleteDemandBasedCloning{FFlag::LuauIncrementalAutocompleteDemandBasedCloning, true};
|
|
ScopedFastFlag luauBetterScopeSelection{FFlag::LuauBetterScopeSelection, true};
|
|
ScopedFastFlag luauBlockDiffFragmentSelection{FFlag::LuauBlockDiffFragmentSelection, true};
|
|
ScopedFastFlag luauAutocompleteUsesModuleForTypeCompatibility{FFlag::LuauAutocompleteUsesModuleForTypeCompatibility, true};
|
|
|
|
FragmentAutocompleteFixtureImpl()
|
|
: BaseType(true)
|
|
{
|
|
}
|
|
|
|
CheckResult checkWithOptions(const std::string& source)
|
|
{
|
|
return this->check(source, getOptions());
|
|
}
|
|
|
|
ParseResult parseHelper_(SourceModule& source, std::string document)
|
|
{
|
|
ParseOptions parseOptions;
|
|
parseOptions.captureComments = true;
|
|
ParseResult parseResult = Parser::parse(document.c_str(), document.length(), *source.names, *source.allocator, parseOptions);
|
|
return parseResult;
|
|
}
|
|
|
|
ParseResult parseHelper(std::string document)
|
|
{
|
|
SourceModule& source = getSource();
|
|
return parseHelper_(source, document);
|
|
}
|
|
|
|
FragmentAutocompleteAncestryResult runAutocompleteVisitor(const std::string& source, const Position& cursorPos)
|
|
{
|
|
ParseResult p = this->tryParse(source); // We don't care about parsing incomplete asts
|
|
REQUIRE(p.root);
|
|
return findAncestryForFragmentParse(p.root, cursorPos, p.root);
|
|
}
|
|
|
|
FragmentRegion getAutocompleteRegion(const std::string source, const Position& cursorPos)
|
|
{
|
|
ParseResult p = parseHelper(source);
|
|
return Luau::getFragmentRegion(p.root, cursorPos);
|
|
}
|
|
|
|
std::optional<FragmentParseResult> parseFragment(
|
|
const std::string& document,
|
|
const Position& cursorPos,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt
|
|
)
|
|
{
|
|
ParseResult p = parseHelper(document);
|
|
ModulePtr module = this->getMainModule(getOptions().forAutocomplete);
|
|
std::string_view srcString = document;
|
|
return Luau::parseFragment(module->root, p.root, module->names.get(), srcString, cursorPos, fragmentEndPosition);
|
|
}
|
|
|
|
CheckResult checkOldSolver(const std::string& source)
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, false};
|
|
return this->check(Mode::Strict, source, getOptions());
|
|
}
|
|
|
|
FragmentTypeCheckResult checkFragment(
|
|
const std::string& document,
|
|
const Position& cursorPos,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt
|
|
)
|
|
{
|
|
ParseResult p = parseHelper(document);
|
|
auto [_, result] = Luau::typecheckFragment(this->frontend, "MainModule", cursorPos, getOptions(), document, fragmentEndPosition, p.root);
|
|
return result;
|
|
}
|
|
|
|
FragmentAutocompleteStatusResult autocompleteFragment(
|
|
const std::string& document,
|
|
Position cursorPos,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt
|
|
)
|
|
{
|
|
ParseOptions parseOptions;
|
|
parseOptions.captureComments = true;
|
|
ParseResult parseResult = parseHelper(document);
|
|
FrontendOptions options = getOptions();
|
|
FragmentContext context{document, parseResult, options, fragmentEndPosition};
|
|
return Luau::tryFragmentAutocomplete(this->frontend, "MainModule", cursorPos, context, nullCallback);
|
|
}
|
|
|
|
|
|
void autocompleteFragmentInBothSolvers(
|
|
const std::string& document,
|
|
const std::string& updated,
|
|
Position cursorPos,
|
|
std::function<void(FragmentAutocompleteStatusResult& result)> assertions,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt
|
|
)
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
this->check(document, getOptions());
|
|
|
|
FragmentAutocompleteStatusResult result = autocompleteFragment(updated, cursorPos, fragmentEndPosition);
|
|
CHECK(result.status != FragmentAutocompleteStatus::InternalIce);
|
|
assertions(result);
|
|
|
|
ScopedFastFlag _{FFlag::LuauSolverV2, false};
|
|
this->check(document, getOptions());
|
|
|
|
result = autocompleteFragment(updated, cursorPos, fragmentEndPosition);
|
|
CHECK(result.status != FragmentAutocompleteStatus::InternalIce);
|
|
assertions(result);
|
|
}
|
|
|
|
std::pair<FragmentTypeCheckStatus, FragmentTypeCheckResult> typecheckFragmentForModule(
|
|
const ModuleName& module,
|
|
const std::string& document,
|
|
Position cursorPos,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt
|
|
)
|
|
{
|
|
ParseResult pr = parseHelper(document);
|
|
return Luau::typecheckFragment(this->frontend, module, cursorPos, getOptions(), document, fragmentEndPosition, pr.root);
|
|
}
|
|
|
|
FragmentAutocompleteStatusResult autocompleteFragmentForModule(
|
|
const ModuleName& module,
|
|
const std::string& document,
|
|
Position cursorPos,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt
|
|
)
|
|
{
|
|
ParseOptions parseOptions;
|
|
parseOptions.captureComments = true;
|
|
ParseResult parseResult = parseHelper(document);
|
|
FrontendOptions options;
|
|
FragmentContext context{document, parseResult, options, fragmentEndPosition};
|
|
return Luau::tryFragmentAutocomplete(this->frontend, module, cursorPos, context, nullCallback);
|
|
}
|
|
|
|
SourceModule& getSource()
|
|
{
|
|
source = std::make_unique<SourceModule>();
|
|
return *source;
|
|
}
|
|
|
|
private:
|
|
std::unique_ptr<SourceModule> source = std::make_unique<SourceModule>();
|
|
};
|
|
|
|
struct FragmentAutocompleteFixture : FragmentAutocompleteFixtureImpl<Fixture>
|
|
{
|
|
FragmentAutocompleteFixture()
|
|
: FragmentAutocompleteFixtureImpl<Fixture>()
|
|
{
|
|
addGlobalBinding(frontend.globals, "table", Binding{builtinTypes->anyType});
|
|
addGlobalBinding(frontend.globals, "math", Binding{builtinTypes->anyType});
|
|
addGlobalBinding(frontend.globalsForAutocomplete, "table", Binding{builtinTypes->anyType});
|
|
addGlobalBinding(frontend.globalsForAutocomplete, "math", Binding{builtinTypes->anyType});
|
|
}
|
|
};
|
|
|
|
struct FragmentAutocompleteBuiltinsFixture : FragmentAutocompleteFixtureImpl<BuiltinsFixture>
|
|
{
|
|
FragmentAutocompleteBuiltinsFixture()
|
|
: FragmentAutocompleteFixtureImpl<BuiltinsFixture>()
|
|
{
|
|
const std::string fakeVecDecl = R"(
|
|
declare class FakeVec
|
|
function dot(self, x: FakeVec) : FakeVec
|
|
zero : FakeVec
|
|
end
|
|
)";
|
|
// The old solver always performs a strict mode check and populates the module resolver and globals
|
|
// for autocomplete.
|
|
// The new solver just populates the globals and the moduleResolver.
|
|
// Because these tests run in both the old solver and the new solver, and the test suite
|
|
// now picks the module resolver as appropriate in order to better mimic the studio code path,
|
|
// we have to load the definition file into both the 'globals'/'resolver' and the equivalent
|
|
// 'for autocomplete'.
|
|
loadDefinition(fakeVecDecl);
|
|
loadDefinition(fakeVecDecl, /* For Autocomplete Module */ true);
|
|
|
|
addGlobalBinding(frontend.globals, "game", Binding{builtinTypes->anyType});
|
|
addGlobalBinding(frontend.globalsForAutocomplete, "game", Binding{builtinTypes->anyType});
|
|
}
|
|
};
|
|
|
|
TEST_SUITE_BEGIN("FragmentSelectionSpecTests");
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "just_two_locals")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)",
|
|
{2, 11}
|
|
);
|
|
|
|
CHECK_EQ(Location{{2, 0}, {2, 11}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
REQUIRE(region.nearestStatement);
|
|
CHECK(region.nearestStatement->as<AstStatLocal>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "singleline_call")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
abc("foo")
|
|
)",
|
|
{1, 10}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 10}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
REQUIRE(region.nearestStatement);
|
|
CHECK(region.nearestStatement->as<AstStatExpr>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "midway_multiline_call")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
abc(
|
|
"foo"
|
|
)
|
|
)",
|
|
{2, 4}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {2, 4}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
REQUIRE(region.nearestStatement);
|
|
CHECK(region.nearestStatement->as<AstStatExpr>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "end_multiline_call")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
abc(
|
|
"foo"
|
|
)
|
|
)",
|
|
{3, 1}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {3, 1}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
REQUIRE(region.nearestStatement);
|
|
CHECK(region.nearestStatement->as<AstStatExpr>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "midway_through_call")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
abc("foo")
|
|
)",
|
|
{1, 6}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 6}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
REQUIRE(region.nearestStatement);
|
|
CHECK(region.nearestStatement->as<AstStatExpr>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "inside_incomplete_do")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local x = 4
|
|
do
|
|
)",
|
|
{2, 2}
|
|
);
|
|
|
|
CHECK_EQ(Location{{2, 2}, {2, 2}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatBlock>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "end_of_do")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local x = 4
|
|
do
|
|
end
|
|
)",
|
|
{3, 3}
|
|
);
|
|
|
|
CHECK_EQ(Location{{3, 3}, {3, 3}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatBlock>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "inside_do")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local x = 4
|
|
do
|
|
|
|
end
|
|
)",
|
|
{3, 3}
|
|
);
|
|
|
|
CHECK_EQ(Location{{3, 3}, {3, 3}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatBlock>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_statement_inside_do")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local x = 4
|
|
do
|
|
local x =
|
|
end
|
|
)",
|
|
{3, 13}
|
|
);
|
|
|
|
CHECK_EQ(Location{{3, 4}, {3, 13}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocal>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_statement_after_do")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local x = 4
|
|
do
|
|
|
|
end
|
|
local x =
|
|
)",
|
|
{5, 9}
|
|
);
|
|
|
|
CHECK_EQ(Location{{5, 0}, {5, 9}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocal>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "before_func")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f()
|
|
end
|
|
)",
|
|
{1, 0}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 0}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "after_func_same_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f()
|
|
end
|
|
)",
|
|
{2, 3}
|
|
);
|
|
CHECK_EQ(Location{{2, 3}, {2, 3}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "after_func_new_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f()
|
|
end
|
|
|
|
)",
|
|
{3, 0}
|
|
);
|
|
CHECK_EQ(Location{{3, 0}, {3, 0}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "while_writing_func")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f(arg1,
|
|
)",
|
|
{1, 17}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 17}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "writing_func_annotation")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f(arg1 : T
|
|
)",
|
|
{1, 19}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 19}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "writing_func_return")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f(arg1 : T) :
|
|
)",
|
|
{1, 22}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 22}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "writing_func_return_pack")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
function f(arg1 : T) : T...
|
|
)",
|
|
{1, 27}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 27}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "before_local_func")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f()
|
|
end
|
|
)",
|
|
{1, 0}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 0}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "after_local_func_same_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f()
|
|
end
|
|
)",
|
|
{2, 3}
|
|
);
|
|
CHECK_EQ(Location{{2, 3}, {2, 3}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "after_local_func_new_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f()
|
|
end
|
|
|
|
)",
|
|
{3, 0}
|
|
);
|
|
CHECK_EQ(Location{{3, 0}, {3, 0}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "while_writing_local_func")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f(arg1,
|
|
)",
|
|
{1, 22}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 22}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "writing_local_func_annotation")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f(arg1 : T
|
|
)",
|
|
{1, 25}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 25}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "writing_local_func_return")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f(arg1 : T) :
|
|
)",
|
|
{1, 28}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 28}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "writing_local_func_return_pack")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
local function f(arg1 : T) : T...
|
|
)",
|
|
{1, 33}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 33}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocalFunction>());
|
|
}
|
|
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "single_line_local_and_annot")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
type Part = {x : number}
|
|
local part : Part = {x = 3}; pa
|
|
)",
|
|
{2, 32}
|
|
);
|
|
CHECK_EQ(Location{{2, 29}, {2, 32}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatError>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_while_in_condition")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
while t
|
|
)",
|
|
Position{1, 7}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 7}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatWhile>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "while_inside_condition_same_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
while true do
|
|
end
|
|
)",
|
|
Position{1, 13}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 13}, {1, 13}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatWhile>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_for_numeric_in_condition")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
for c = 1,3
|
|
)",
|
|
Position{1, 11}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 11}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFor>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_for_numeric_in_body")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
for c = 1,3 do
|
|
)",
|
|
Position{1, 14}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 14}, {1, 14}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatFor>());
|
|
}
|
|
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_for_in_in_condition_1")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
for i,v in {1,2,3}
|
|
)",
|
|
Position{1, 18}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 18}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatForIn>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_for_in_in_condition_2")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
for i,v in
|
|
)",
|
|
Position{1, 10}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 10}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatForIn>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_for_in_in_condition_3")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
for i,
|
|
)",
|
|
Position{1, 6}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 0}, {1, 6}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatForIn>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "partial_for_in_in_body")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
for i,v in {1,2,3} do
|
|
)",
|
|
Position{1, 21}
|
|
);
|
|
|
|
CHECK_EQ(Location{{1, 21}, {1, 21}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatForIn>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_partial")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if
|
|
)",
|
|
Position{1, 3}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 3}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_partial_in_condition_at")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true
|
|
)",
|
|
Position{1, 7}
|
|
);
|
|
CHECK_EQ(Location{{1, 0}, {1, 7}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_partial_in_condition_after")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true
|
|
)",
|
|
Position{1, 8}
|
|
);
|
|
CHECK_EQ(Location{{1, 8}, {1, 8}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_partial_after_condition")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
)",
|
|
Position{1, 12}
|
|
);
|
|
CHECK_EQ(Location{{1, 12}, {1, 12}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_partial_new_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
|
|
)",
|
|
Position{2, 0}
|
|
);
|
|
CHECK_EQ(Location{{2, 0}, {2, 0}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_complete_inside_scope_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
local x =
|
|
end
|
|
|
|
)",
|
|
Position{2, 13}
|
|
);
|
|
CHECK_EQ(Location{{2, 4}, {2, 13}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatLocal>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_else_if")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
elseif
|
|
end
|
|
|
|
)",
|
|
Position{2, 8}
|
|
);
|
|
CHECK_EQ(Location{{2, 8}, {2, 8}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_else_if_no_end")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
elseif
|
|
)",
|
|
Position{2, 8}
|
|
);
|
|
CHECK_EQ(Location{{2, 0}, {2, 8}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_else_if_after_then")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
elseif false then
|
|
end
|
|
|
|
)",
|
|
Position{2, 17}
|
|
);
|
|
CHECK_EQ(Location{{2, 17}, {2, 17}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "if_else_if_after_then_new_line")
|
|
{
|
|
auto region = getAutocompleteRegion(
|
|
R"(
|
|
if true then
|
|
elseif false then
|
|
|
|
end
|
|
|
|
)",
|
|
Position{3, 0}
|
|
);
|
|
CHECK_EQ(Location{{3, 0}, {3, 0}}, region.fragmentLocation);
|
|
REQUIRE(region.parentBlock);
|
|
CHECK(region.nearestStatement->as<AstStatIf>());
|
|
}
|
|
|
|
|
|
TEST_SUITE_END();
|
|
|
|
// NOLINTBEGIN(bugprone-unchecked-optional-access)
|
|
TEST_SUITE_BEGIN("FragmentAutocompleteTraversalTests");
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "just_two_locals")
|
|
{
|
|
auto result = runAutocompleteVisitor(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)",
|
|
{2, 11}
|
|
);
|
|
|
|
CHECK_EQ(3, result.ancestry.size());
|
|
CHECK_EQ(1, result.localStack.size());
|
|
CHECK_EQ(result.localMap.size(), result.localStack.size());
|
|
REQUIRE(result.nearestStatement);
|
|
|
|
AstStatLocal* local = result.nearestStatement->as<AstStatLocal>();
|
|
REQUIRE(local);
|
|
CHECK(1 == local->vars.size);
|
|
CHECK_EQ("y", std::string(local->vars.data[0]->name.value));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "cursor_within_scope_tracks_locals_from_previous_scope")
|
|
{
|
|
auto result = runAutocompleteVisitor(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
if x == 4 then
|
|
local e = y
|
|
end
|
|
)",
|
|
{4, 15}
|
|
);
|
|
|
|
CHECK_EQ(5, result.ancestry.size());
|
|
CHECK_EQ(2, result.localStack.size());
|
|
CHECK_EQ(result.localMap.size(), result.localStack.size());
|
|
REQUIRE(result.nearestStatement);
|
|
CHECK_EQ("y", std::string(result.localStack.back()->name.value));
|
|
|
|
AstStatLocal* local = result.nearestStatement->as<AstStatLocal>();
|
|
REQUIRE(local);
|
|
CHECK(1 == local->vars.size);
|
|
CHECK_EQ("e", std::string(local->vars.data[0]->name.value));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "cursor_that_comes_later_shouldnt_capture_locals_in_unavailable_scope")
|
|
{
|
|
auto result = runAutocompleteVisitor(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
if x == 4 then
|
|
local e = y
|
|
end
|
|
local z = x + x
|
|
if y == 5 then
|
|
local q = x + y + z
|
|
end
|
|
)",
|
|
{8, 23}
|
|
);
|
|
|
|
CHECK_EQ(6, result.ancestry.size());
|
|
CHECK_EQ(3, result.localStack.size());
|
|
CHECK_EQ(result.localMap.size(), result.localStack.size());
|
|
REQUIRE(result.nearestStatement);
|
|
CHECK_EQ("z", std::string(result.localStack.back()->name.value));
|
|
|
|
AstStatLocal* local = result.nearestStatement->as<AstStatLocal>();
|
|
REQUIRE(local);
|
|
CHECK(1 == local->vars.size);
|
|
CHECK_EQ("q", std::string(local->vars.data[0]->name.value));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "nearest_enclosing_statement_can_be_non_local")
|
|
{
|
|
auto result = runAutocompleteVisitor(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
if x == 4 then
|
|
)",
|
|
{3, 4}
|
|
);
|
|
|
|
CHECK_EQ(4, result.ancestry.size());
|
|
CHECK_EQ(2, result.localStack.size());
|
|
CHECK_EQ(result.localMap.size(), result.localStack.size());
|
|
REQUIRE(result.nearestStatement);
|
|
CHECK_EQ("y", std::string(result.localStack.back()->name.value));
|
|
|
|
AstStatIf* ifS = result.nearestStatement->as<AstStatIf>();
|
|
CHECK(ifS != nullptr);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "local_funcs_show_up_in_local_stack")
|
|
{
|
|
auto result = runAutocompleteVisitor(
|
|
R"(
|
|
local function foo() return 4 end
|
|
local x = foo()
|
|
local function bar() return x + foo() end
|
|
)",
|
|
{3, 32}
|
|
);
|
|
|
|
CHECK_EQ(8, result.ancestry.size());
|
|
CHECK_EQ(3, result.localStack.size());
|
|
CHECK_EQ(result.localMap.size(), result.localStack.size());
|
|
CHECK_EQ("bar", std::string(result.localStack.back()->name.value));
|
|
auto returnSt = result.nearestStatement->as<AstStatReturn>();
|
|
CHECK(returnSt != nullptr);
|
|
}
|
|
|
|
TEST_SUITE_END();
|
|
|
|
|
|
TEST_SUITE_BEGIN("FragmentAutocompleteParserTests");
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "empty_program_1")
|
|
{
|
|
checkWithOptions("");
|
|
ScopedFastInt sfi{FInt::LuauParseErrorLimit, 1};
|
|
auto fragment = parseFragment("", Position(0, 39));
|
|
REQUIRE(fragment);
|
|
CHECK(fragment->fragmentToParse == "");
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "empty_program_2")
|
|
{
|
|
const std::string source = R"(
|
|
|
|
)";
|
|
checkWithOptions(source);
|
|
ScopedFastInt sfi{FInt::LuauParseErrorLimit, 1};
|
|
auto fragment = parseFragment(source, Position(1, 39));
|
|
REQUIRE(fragment);
|
|
CHECK(fragment->fragmentToParse == "");
|
|
}
|
|
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "thrown_parse_error_leads_to_null_root")
|
|
{
|
|
checkWithOptions("type A = ");
|
|
ScopedFastInt sfi{FInt::LuauParseErrorLimit, 1};
|
|
auto fragment = parseFragment("type A = <>function<> more garbage here", Position(0, 39));
|
|
CHECK(fragment == std::nullopt);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "local_initializer")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
checkWithOptions("local a =");
|
|
auto fragment = parseFragment("local a =", Position(0, 9));
|
|
|
|
REQUIRE(fragment.has_value());
|
|
CHECK_EQ("local a =", fragment->fragmentToParse);
|
|
CHECK_EQ(Location{Position{0, 0}, 9}, fragment->root->location);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "statement_in_empty_fragment_is_non_null")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
auto res = checkWithOptions(R"(
|
|
|
|
)");
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = parseFragment(
|
|
R"(
|
|
|
|
)",
|
|
Position(1, 0)
|
|
);
|
|
REQUIRE(fragment.has_value());
|
|
CHECK_EQ("", fragment->fragmentToParse);
|
|
CHECK_EQ(2, fragment->ancestry.size());
|
|
REQUIRE(fragment->root);
|
|
CHECK_EQ(0, fragment->root->body.size);
|
|
auto statBody = fragment->root->as<AstStatBlock>();
|
|
CHECK(statBody != nullptr);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_parse_complete_fragments")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
auto res = checkWithOptions(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)"
|
|
);
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = parseFragment(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
local z = x + y
|
|
)",
|
|
Position{3, 15}
|
|
);
|
|
|
|
REQUIRE(fragment.has_value());
|
|
|
|
CHECK_EQ(Location{Position{3, 0}, Position{3, 15}}, fragment->root->location);
|
|
|
|
CHECK_EQ("local z = x + y", fragment->fragmentToParse);
|
|
CHECK_EQ(5, fragment->ancestry.size());
|
|
REQUIRE(fragment->root);
|
|
CHECK_EQ(1, fragment->root->body.size);
|
|
auto stat = fragment->root->body.data[0]->as<AstStatLocal>();
|
|
REQUIRE(stat);
|
|
CHECK_EQ(1, stat->vars.size);
|
|
CHECK_EQ(1, stat->values.size);
|
|
CHECK_EQ("z", std::string(stat->vars.data[0]->name.value));
|
|
|
|
auto bin = stat->values.data[0]->as<AstExprBinary>();
|
|
REQUIRE(bin);
|
|
CHECK_EQ(AstExprBinary::Op::Add, bin->op);
|
|
|
|
auto lhs = bin->left->as<AstExprLocal>();
|
|
auto rhs = bin->right->as<AstExprLocal>();
|
|
REQUIRE(lhs);
|
|
REQUIRE(rhs);
|
|
CHECK_EQ("x", std::string(lhs->local->name.value));
|
|
CHECK_EQ("y", std::string(rhs->local->name.value));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_parse_fragments_in_line")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
auto res = checkWithOptions(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)"
|
|
);
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = parseFragment(
|
|
R"(
|
|
local x = 4
|
|
local z = x + y
|
|
local y = 5
|
|
)",
|
|
Position{2, 15}
|
|
);
|
|
|
|
REQUIRE(fragment.has_value());
|
|
|
|
CHECK_EQ("local z = x + y", fragment->fragmentToParse);
|
|
CHECK_EQ(5, fragment->ancestry.size());
|
|
REQUIRE(fragment->root);
|
|
CHECK_EQ(Location{Position{2, 0}, Position{2, 15}}, fragment->root->location);
|
|
CHECK_EQ(1, fragment->root->body.size);
|
|
auto stat = fragment->root->body.data[0]->as<AstStatLocal>();
|
|
REQUIRE(stat);
|
|
CHECK_EQ(1, stat->vars.size);
|
|
CHECK_EQ(1, stat->values.size);
|
|
CHECK_EQ("z", std::string(stat->vars.data[0]->name.value));
|
|
|
|
auto bin = stat->values.data[0]->as<AstExprBinary>();
|
|
REQUIRE(bin);
|
|
CHECK_EQ(AstExprBinary::Op::Add, bin->op);
|
|
|
|
auto lhs = bin->left->as<AstExprLocal>();
|
|
auto rhs = bin->right->as<AstExprGlobal>();
|
|
REQUIRE(lhs);
|
|
REQUIRE(rhs);
|
|
CHECK_EQ("x", std::string(lhs->local->name.value));
|
|
CHECK_EQ("y", std::string(rhs->name.value));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_parse_in_correct_scope")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
checkWithOptions(R"(
|
|
local myLocal = 4
|
|
function abc()
|
|
local myInnerLocal = 1
|
|
|
|
end
|
|
)");
|
|
|
|
auto fragment = parseFragment(
|
|
R"(
|
|
local myLocal = 4
|
|
function abc()
|
|
local myInnerLocal = 1
|
|
|
|
end
|
|
)",
|
|
Position{6, 0}
|
|
);
|
|
|
|
REQUIRE(fragment.has_value());
|
|
|
|
CHECK_EQ("", fragment->fragmentToParse);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_parse_single_line_fragment_override")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
auto res = checkWithOptions("function abc(foo: string) end");
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto callFragment = parseFragment(
|
|
R"(function abc(foo: string) end
|
|
abc("foo")
|
|
abc("bar")
|
|
)",
|
|
Position{1, 6},
|
|
Position{1, 10}
|
|
);
|
|
|
|
REQUIRE(callFragment.has_value());
|
|
|
|
CHECK_EQ("abc(\"foo\")", callFragment->fragmentToParse);
|
|
CHECK(callFragment->nearestStatement);
|
|
CHECK(callFragment->nearestStatement->is<AstStatExpr>());
|
|
|
|
CHECK_GE(callFragment->ancestry.size(), 2);
|
|
|
|
AstNode* back = callFragment->ancestry.back();
|
|
CHECK(back->is<AstExprConstantString>());
|
|
CHECK_EQ(Position{1, 4}, back->location.begin);
|
|
CHECK_EQ(Position{1, 9}, back->location.end);
|
|
|
|
AstNode* parent = callFragment->ancestry.rbegin()[1];
|
|
CHECK(parent->is<AstExprCall>());
|
|
CHECK_EQ(Position{1, 0}, parent->location.begin);
|
|
CHECK_EQ(Position{1, 10}, parent->location.end);
|
|
|
|
|
|
auto stringFragment = parseFragment(
|
|
R"(function abc(foo: string) end
|
|
abc("foo")
|
|
abc("bar")
|
|
)",
|
|
Position{1, 6},
|
|
Position{1, 9}
|
|
);
|
|
|
|
REQUIRE(stringFragment.has_value());
|
|
|
|
CHECK_EQ("abc(\"foo\")", stringFragment->fragmentToParse);
|
|
CHECK(stringFragment->nearestStatement);
|
|
CHECK(stringFragment->nearestStatement->is<AstStatExpr>());
|
|
|
|
CHECK_GE(stringFragment->ancestry.size(), 1);
|
|
|
|
back = stringFragment->ancestry.back();
|
|
|
|
auto asString = back->as<AstExprConstantString>();
|
|
CHECK(asString);
|
|
|
|
CHECK_EQ(Position{1, 4}, asString->location.begin);
|
|
CHECK_EQ(Position{1, 9}, asString->location.end);
|
|
CHECK_EQ("foo", std::string{asString->value.data});
|
|
CHECK_EQ(AstExprConstantString::QuotedSimple, asString->quoteStyle);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_parse_multi_line_fragment_override")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
|
|
auto res = checkWithOptions("function abc(foo: string) end");
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = parseFragment(
|
|
R"(function abc(foo: string) end
|
|
abc(
|
|
"foo"
|
|
)
|
|
abc("bar")
|
|
)",
|
|
Position{2, 5},
|
|
Position{3, 1}
|
|
);
|
|
|
|
REQUIRE(fragment.has_value());
|
|
|
|
CHECK_EQ("abc(\n\"foo\"\n)", fragment->fragmentToParse);
|
|
CHECK(fragment->nearestStatement);
|
|
CHECK(fragment->nearestStatement->is<AstStatExpr>());
|
|
|
|
CHECK_GE(fragment->ancestry.size(), 2);
|
|
|
|
AstNode* back = fragment->ancestry.back();
|
|
CHECK(back->is<AstExprConstantString>());
|
|
CHECK_EQ(Position{2, 0}, back->location.begin);
|
|
CHECK_EQ(Position{2, 5}, back->location.end);
|
|
|
|
AstNode* parent = fragment->ancestry.rbegin()[1];
|
|
CHECK(parent->is<AstExprCall>());
|
|
CHECK_EQ(Position{1, 0}, parent->location.begin);
|
|
CHECK_EQ(Position{3, 1}, parent->location.end);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "respects_frontend_options")
|
|
{
|
|
DOES_NOT_PASS_NEW_SOLVER_GUARD();
|
|
|
|
std::string source = R"(
|
|
local tbl = { abc = 1234}
|
|
t
|
|
)";
|
|
fileResolver.source["game/A"] = source;
|
|
|
|
FrontendOptions opts;
|
|
opts.forAutocomplete = true;
|
|
|
|
frontend.check("game/A", opts);
|
|
CHECK_NE(frontend.moduleResolverForAutocomplete.getModule("game/A"), nullptr);
|
|
CHECK_EQ(frontend.moduleResolver.getModule("game/A"), nullptr);
|
|
ParseOptions parseOptions;
|
|
parseOptions.captureComments = true;
|
|
SourceModule sourceMod;
|
|
ParseResult parseResult = Parser::parse(source.c_str(), source.length(), *sourceMod.names, *sourceMod.allocator, parseOptions);
|
|
FragmentContext context{source, parseResult, opts, std::nullopt};
|
|
|
|
FragmentAutocompleteStatusResult frag = Luau::tryFragmentAutocomplete(frontend, "game/A", Position{2, 1}, context, nullCallback);
|
|
REQUIRE(frag.result);
|
|
CHECK_EQ("game/A", frag.result->incrementalModule->name);
|
|
CHECK_NE(frontend.moduleResolverForAutocomplete.getModule("game/A"), nullptr);
|
|
CHECK_EQ(frontend.moduleResolver.getModule("game/A"), nullptr);
|
|
}
|
|
|
|
TEST_SUITE_END();
|
|
|
|
|
|
TEST_SUITE_BEGIN("FragmentAutocompleteTypeCheckerTests");
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_typecheck_simple_fragment")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
auto res = checkWithOptions(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)"
|
|
);
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = checkFragment(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
local z = x + y
|
|
)",
|
|
Position{3, 15}
|
|
);
|
|
|
|
auto opt = linearSearchForBinding(fragment.freshScope.get(), "z");
|
|
REQUIRE(opt);
|
|
CHECK_EQ("number", toString(*opt));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_typecheck_fragment_inserted_inline")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
auto res = checkWithOptions(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)"
|
|
);
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
auto fragment = checkFragment(
|
|
R"(
|
|
local x = 4
|
|
local z = x
|
|
local y = 5
|
|
)",
|
|
Position{2, 11}
|
|
);
|
|
|
|
auto correct = linearSearchForBinding(fragment.freshScope.get(), "z");
|
|
REQUIRE(correct);
|
|
CHECK_EQ("number", toString(*correct));
|
|
}
|
|
|
|
TEST_SUITE_END();
|
|
|
|
|
|
TEST_SUITE_BEGIN("MixedModeTests");
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "mixed_mode_basic_example_append")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, false};
|
|
auto res = checkOldSolver(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)"
|
|
);
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = checkFragment(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
local z = x + y
|
|
)",
|
|
Position{3, 15}
|
|
);
|
|
|
|
auto opt = linearSearchForBinding(fragment.freshScope.get(), "z");
|
|
REQUIRE(opt);
|
|
CHECK_EQ("number", toString(*opt));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "mixed_mode_basic_example_inlined")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, false};
|
|
auto res = checkOldSolver(
|
|
R"(
|
|
local x = 4
|
|
local y = 5
|
|
)"
|
|
);
|
|
|
|
auto fragment = checkFragment(
|
|
R"(
|
|
local x = 4
|
|
local z = x
|
|
local y = 5
|
|
)",
|
|
Position{2, 11}
|
|
);
|
|
|
|
auto correct = linearSearchForBinding(fragment.freshScope.get(), "z");
|
|
REQUIRE(correct);
|
|
CHECK_EQ("number", toString(*correct));
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "mixed_mode_can_autocomplete_simple_property_access")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, false};
|
|
auto res = checkOldSolver(
|
|
R"(
|
|
local tbl = { abc = 1234}
|
|
)"
|
|
);
|
|
|
|
LUAU_REQUIRE_NO_ERRORS(res);
|
|
|
|
auto fragment = autocompleteFragment(
|
|
R"(
|
|
local tbl = { abc = 1234}
|
|
tbl.
|
|
)",
|
|
Position{2, 5}
|
|
);
|
|
REQUIRE(fragment.result);
|
|
LUAU_ASSERT(fragment.result->freshScope);
|
|
|
|
CHECK_EQ(1, fragment.result->acResults.entryMap.size());
|
|
CHECK(fragment.result->acResults.entryMap.count("abc"));
|
|
CHECK_EQ(AutocompleteContext::Property, fragment.result->acResults.context);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "typecheck_fragment_handles_unusable_module")
|
|
{
|
|
const std::string sourceA = "MainModule";
|
|
fileResolver.source[sourceA] = R"(
|
|
local Modules = game:GetService('Gui').Modules
|
|
local B = require(Modules.B)
|
|
return { hello = B }
|
|
)";
|
|
|
|
const std::string sourceB = "game/Gui/Modules/B";
|
|
fileResolver.source[sourceB] = R"(return {hello = "hello"})";
|
|
|
|
CheckResult result = frontend.check(sourceA, getOptions());
|
|
CHECK(!frontend.isDirty(sourceA, getOptions().forAutocomplete));
|
|
|
|
std::weak_ptr<Module> weakModule = getModuleResolver(frontend).getModule(sourceB);
|
|
REQUIRE(!weakModule.expired());
|
|
|
|
frontend.markDirty(sourceB);
|
|
CHECK(frontend.isDirty(sourceA, getOptions().forAutocomplete));
|
|
|
|
frontend.check(sourceB, getOptions());
|
|
CHECK(weakModule.expired());
|
|
|
|
auto [status, _] = typecheckFragmentForModule(sourceA, fileResolver.source[sourceA], Luau::Position(0, 0));
|
|
CHECK_EQ(status, FragmentTypeCheckStatus::SkipAutocomplete);
|
|
|
|
auto [status2, _2] = typecheckFragmentForModule(sourceB, fileResolver.source[sourceB], Luau::Position(3, 20));
|
|
CHECK_EQ(status2, FragmentTypeCheckStatus::Success);
|
|
}
|
|
|
|
TEST_SUITE_END();
|
|
|
|
TEST_SUITE_BEGIN("FragmentAutocompleteTests");
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "multiple_fragment_autocomplete")
|
|
{
|
|
ToStringOptions opt;
|
|
opt.exhaustive = true;
|
|
opt.exhaustive = true;
|
|
opt.functionTypeArguments = true;
|
|
opt.maxTableLength = 0;
|
|
opt.maxTypeLength = 0;
|
|
|
|
auto checkAndExamine = [&](const std::string& src, const std::string& idName, const std::string& idString)
|
|
{
|
|
checkWithOptions(src);
|
|
auto id = getType(idName, true);
|
|
LUAU_ASSERT(id);
|
|
CHECK_EQ(Luau::toString(*id, opt), idString);
|
|
};
|
|
|
|
auto getTypeFromModule = [](ModulePtr module, const std::string& name) -> std::optional<TypeId>
|
|
{
|
|
if (!module->hasModuleScope())
|
|
return std::nullopt;
|
|
return lookupName(module->getModuleScope(), name);
|
|
};
|
|
|
|
auto fragmentACAndCheck = [&](const std::string& updated,
|
|
const Position& pos,
|
|
const std::string& idName,
|
|
const std::string& srcIdString,
|
|
const std::string& fragIdString)
|
|
{
|
|
FragmentAutocompleteStatusResult frag = autocompleteFragment(updated, pos, std::nullopt);
|
|
REQUIRE(frag.result);
|
|
auto fragId = getTypeFromModule(frag.result->incrementalModule, idName);
|
|
LUAU_ASSERT(fragId);
|
|
CHECK_EQ(Luau::toString(*fragId, opt), fragIdString);
|
|
|
|
auto srcId = getType(idName, true);
|
|
LUAU_ASSERT(srcId);
|
|
CHECK_EQ(Luau::toString(*srcId, opt), srcIdString);
|
|
};
|
|
|
|
const std::string source = R"(local module = {}
|
|
f
|
|
return module)";
|
|
|
|
const std::string updated1 = R"(local module = {}
|
|
function module.a
|
|
return module)";
|
|
|
|
const std::string updated2 = R"(local module = {}
|
|
function module.ab
|
|
return module)";
|
|
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, false};
|
|
ScopedFastFlag sff2{FFlag::LuauCloneIncrementalModule, true};
|
|
ScopedFastFlag sff3{FFlag::LuauFreeTypesMustHaveBounds, true};
|
|
checkAndExamine(source, "module", "{ }");
|
|
fragmentACAndCheck(updated1, Position{1, 17}, "module", "{ }", "{ a: (%error-id%: unknown) -> () }");
|
|
fragmentACAndCheck(updated2, Position{1, 18}, "module", "{ }", "{ ab: (%error-id%: unknown) -> () }");
|
|
}
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
checkAndExamine(source, "module", "{ }");
|
|
// [TODO] CLI-140762 Fragment autocomplete still doesn't return correct result when LuauSolverV2 is on
|
|
return;
|
|
fragmentACAndCheck(updated1, Position{1, 17}, "module", "{ }", "{ a: (%error-id%: unknown) -> () }");
|
|
fragmentACAndCheck(updated2, Position{1, 18}, "module", "{ }", "{ ab: (%error-id%: unknown) -> () }");
|
|
}
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_autocomplete_simple_property_access")
|
|
{
|
|
|
|
const std::string source = R"(
|
|
local tbl = { abc = 1234}
|
|
)";
|
|
const std::string updated = R"(
|
|
local tbl = { abc = 1234}
|
|
tbl.
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{2, 5},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto acResults = fragment.result->acResults;
|
|
|
|
CHECK_EQ(1, acResults.entryMap.size());
|
|
CHECK(acResults.entryMap.count("abc"));
|
|
CHECK_EQ(AutocompleteContext::Property, acResults.context);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "can_autocomplete_nested_property_access")
|
|
{
|
|
const std::string source = R"(
|
|
local tbl = { abc = { def = 1234, egh = false } }
|
|
)";
|
|
const std::string updated = R"(
|
|
local tbl = { abc = { def = 1234, egh = false } }
|
|
tbl.abc.
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{2, 8},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
LUAU_ASSERT(fragment.result->freshScope);
|
|
|
|
CHECK_EQ(2, fragment.result->acResults.entryMap.size());
|
|
CHECK(fragment.result->acResults.entryMap.count("def"));
|
|
CHECK(fragment.result->acResults.entryMap.count("egh"));
|
|
CHECK_EQ(fragment.result->acResults.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "multiple_functions_complex")
|
|
{
|
|
const std::string text = R"( local function f1(a1)
|
|
local l1 = 1;
|
|
g1 = 1;
|
|
end
|
|
|
|
local function f2(a2)
|
|
local l2 = 1;
|
|
g2 = 1;
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{0, 0},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") == 0);
|
|
CHECK(strings.count("a1") == 0);
|
|
CHECK(strings.count("l1") == 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") == 0);
|
|
CHECK(strings.count("a2") == 0);
|
|
CHECK(strings.count("l2") == 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{0, 22},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") != 0);
|
|
CHECK(strings.count("a1") != 0);
|
|
CHECK(strings.count("l1") == 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") == 0);
|
|
CHECK(strings.count("a2") == 0);
|
|
CHECK(strings.count("l2") == 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{1, 17},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") != 0);
|
|
CHECK(strings.count("a1") != 0);
|
|
CHECK(strings.count("l1") != 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") == 0);
|
|
CHECK(strings.count("a2") == 0);
|
|
CHECK(strings.count("l2") == 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{2, 11},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") != 0);
|
|
CHECK(strings.count("a1") != 0);
|
|
CHECK(strings.count("l1") != 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") == 0);
|
|
CHECK(strings.count("a2") == 0);
|
|
CHECK(strings.count("l2") == 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{4, 0},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") != 0);
|
|
CHECK(strings.count("a1") == 0);
|
|
CHECK(strings.count("l1") == 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") == 0);
|
|
CHECK(strings.count("a2") == 0);
|
|
CHECK(strings.count("l2") == 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{6, 17},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") != 0);
|
|
CHECK(strings.count("a1") == 0);
|
|
CHECK(strings.count("l1") == 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") != 0);
|
|
CHECK(strings.count("a2") != 0);
|
|
CHECK(strings.count("l2") != 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
text,
|
|
text,
|
|
Position{8, 4},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto strings = fragment.result->acResults.entryMap;
|
|
CHECK(strings.count("f1") != 0);
|
|
CHECK(strings.count("a1") == 0);
|
|
CHECK(strings.count("l1") == 0);
|
|
CHECK(strings.count("g1") != 0);
|
|
CHECK(strings.count("f2") != 0);
|
|
CHECK(strings.count("a2") == 0);
|
|
CHECK(strings.count("l2") == 0);
|
|
CHECK(strings.count("g2") != 0);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "inline_autocomplete_picks_the_right_scope_1")
|
|
{
|
|
const std::string source = R"(
|
|
type Table = { a: number, b: number }
|
|
do
|
|
type Table = { x: string, y: string }
|
|
end
|
|
)";
|
|
|
|
const std::string updated = R"(
|
|
type Table = { a: number, b: number }
|
|
do
|
|
type Table = { x: string, y: string }
|
|
local a : T
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{4, 15},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
LUAU_ASSERT(fragment.result->freshScope);
|
|
REQUIRE(fragment.result->acResults.entryMap.count("Table"));
|
|
REQUIRE(fragment.result->acResults.entryMap["Table"].type);
|
|
const TableType* tv = get<TableType>(follow(*fragment.result->acResults.entryMap["Table"].type));
|
|
REQUIRE(tv);
|
|
CHECK(tv->props.count("x"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "inline_autocomplete_picks_the_right_scope_2")
|
|
{
|
|
const std::string source = R"(
|
|
type Table = { a: number, b: number }
|
|
do
|
|
type Table = { x: string, y: string }
|
|
end
|
|
)";
|
|
|
|
const std::string updated = R"(
|
|
type Table = { a: number, b: number }
|
|
do
|
|
type Table = { x: string, y: string }
|
|
end
|
|
local a : T
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{5, 11},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
LUAU_ASSERT(fragment.result->freshScope);
|
|
REQUIRE(fragment.result->acResults.entryMap.count("Table"));
|
|
REQUIRE(fragment.result->acResults.entryMap["Table"].type);
|
|
const TableType* tv = get<TableType>(follow(*fragment.result->acResults.entryMap["Table"].type));
|
|
REQUIRE(tv);
|
|
CHECK(tv->props.count("a"));
|
|
CHECK(tv->props.count("b"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "nested_recursive_function")
|
|
{
|
|
const std::string source = R"(
|
|
function foo()
|
|
end
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{2, 0},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
CHECK(fragment.result->acResults.entryMap.count("foo"));
|
|
CHECK_EQ(AutocompleteContext::Statement, fragment.result->acResults.context);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "string_literal_with_override")
|
|
{
|
|
const std::string source = R"(
|
|
function foo(bar: string) end
|
|
foo("abc")
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{2, 6},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
CHECK(fragment.result->acResults.entryMap.empty());
|
|
CHECK_EQ(AutocompleteContext::String, fragment.result->acResults.context);
|
|
},
|
|
Position{2, 9}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "empty_program")
|
|
{
|
|
autocompleteFragmentInBothSolvers(
|
|
"",
|
|
"",
|
|
Position{0, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.count("table"));
|
|
CHECK(ac.entryMap.count("math"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Statement);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "local_initializer")
|
|
{
|
|
const std::string source = "local a =";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{0, 9},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
|
|
CHECK(ac.entryMap.count("table"));
|
|
CHECK(ac.entryMap.count("math"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Expression);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "leave_numbers_alone")
|
|
{
|
|
const std::string source = "local a = 3.";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{0, 12},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.empty());
|
|
CHECK_EQ(ac.context, AutocompleteContext::Unknown);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "user_defined_globals")
|
|
{
|
|
const std::string source = "local myLocal = 4; ";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{0, 18},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
|
|
CHECK(ac.entryMap.count("myLocal"));
|
|
CHECK(ac.entryMap.count("table"));
|
|
CHECK(ac.entryMap.count("math"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Statement);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "dont_suggest_local_before_its_definition")
|
|
{
|
|
const std::string source = R"(
|
|
local myLocal = 4
|
|
function abc()
|
|
local myInnerLocal = 1
|
|
|
|
end
|
|
)";
|
|
|
|
// autocomplete after abc but before myInnerLocal
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 0},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto ac = fragment.result->acResults;
|
|
CHECK(ac.entryMap.count("myLocal"));
|
|
LUAU_CHECK_HAS_NO_KEY(ac.entryMap, "myInnerLocal");
|
|
}
|
|
);
|
|
// autocomplete after my inner local
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 0},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto ac = fragment.result->acResults;
|
|
CHECK(ac.entryMap.count("myLocal"));
|
|
CHECK(ac.entryMap.count("myInnerLocal"));
|
|
}
|
|
);
|
|
|
|
// autocomplete after abc, but don't include myInnerLocal(in the hidden scope)
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{6, 0},
|
|
[](FragmentAutocompleteStatusResult& fragment)
|
|
{
|
|
REQUIRE(fragment.result);
|
|
auto ac = fragment.result->acResults;
|
|
CHECK(ac.entryMap.count("myLocal"));
|
|
LUAU_CHECK_HAS_NO_KEY(ac.entryMap, "myInnerLocal");
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "nested_recursive_function")
|
|
{
|
|
const std::string source = R"(
|
|
local function outer()
|
|
local function inner()
|
|
end
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.count("inner"));
|
|
CHECK(ac.entryMap.count("outer"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "user_defined_local_functions_in_own_definition")
|
|
{
|
|
const std::string source = R"(
|
|
local function abc()
|
|
|
|
end
|
|
)";
|
|
// Autocomplete inside of abc
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{2, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.count("abc"));
|
|
CHECK(ac.entryMap.count("table"));
|
|
CHECK(ac.entryMap.count("math"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "global_functions_are_not_scoped_lexically")
|
|
{
|
|
const std::string source = R"(
|
|
if true then
|
|
function abc()
|
|
|
|
end
|
|
end
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{6, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(!ac.entryMap.empty());
|
|
CHECK(ac.entryMap.count("abc"));
|
|
CHECK(ac.entryMap.count("table"));
|
|
CHECK(ac.entryMap.count("math"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "local_functions_fall_out_of_scope")
|
|
{
|
|
const std::string source = R"(
|
|
if true then
|
|
local function abc()
|
|
|
|
end
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{6, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK_NE(0, ac.entryMap.size());
|
|
LUAU_CHECK_HAS_NO_KEY(ac.entryMap, "abc");
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "function_parameters")
|
|
{
|
|
const std::string source = R"(
|
|
function abc(test)
|
|
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.count("test"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "unsealed_table")
|
|
{
|
|
const std::string source = R"(
|
|
local tbl = {}
|
|
tbl.prop = 5
|
|
tbl.
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 12},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK_EQ(1, ac.entryMap.size());
|
|
CHECK(ac.entryMap.count("prop"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "unsealed_table_2")
|
|
{
|
|
const std::string source = R"(
|
|
local tbl = {}
|
|
local inner = { prop = 5 }
|
|
tbl.inner = inner
|
|
tbl.inner.
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 18},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK_EQ(1, ac.entryMap.size());
|
|
CHECK(ac.entryMap.count("prop"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "cyclic_table")
|
|
{
|
|
const std::string source = R"(
|
|
local abc = {}
|
|
local def = { abc = abc }
|
|
abc.def = def
|
|
abc.def.
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 16},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.count("abc"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "table_union")
|
|
{
|
|
const std::string source = R"(
|
|
type t1 = { a1 : string, b2 : number }
|
|
type t2 = { b2 : string, c3 : string }
|
|
function func(abc : t1 | t2)
|
|
|
|
end
|
|
)";
|
|
const std::string updated = R"(
|
|
type t1 = { a1 : string, b2 : number }
|
|
type t2 = { b2 : string, c3 : string }
|
|
function func(abc : t1 | t2)
|
|
abc.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{4, 16},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK_EQ(1, ac.entryMap.size());
|
|
CHECK(ac.entryMap.count("b2"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "table_intersection")
|
|
{
|
|
const std::string source = R"(
|
|
type t1 = { a1 : string, b2 : number }
|
|
type t2 = { b2 : number, c3 : string }
|
|
function func(abc : t1 & t2)
|
|
|
|
end
|
|
)";
|
|
const std::string updated = R"(
|
|
type t1 = { a1 : string, b2 : number }
|
|
type t2 = { b2 : number, c3 : string }
|
|
function func(abc : t1 & t2)
|
|
abc.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{4, 16},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK_EQ(3, ac.entryMap.size());
|
|
CHECK(ac.entryMap.count("a1"));
|
|
CHECK(ac.entryMap.count("b2"));
|
|
CHECK(ac.entryMap.count("c3"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "get_suggestions_for_the_very_start_of_the_script")
|
|
{
|
|
const std::string source = R"(
|
|
|
|
function aaa() end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{0, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK(ac.entryMap.count("table"));
|
|
CHECK_EQ(ac.context, AutocompleteContext::Statement);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "studio_ice_1")
|
|
{
|
|
const std::string source = R"(
|
|
--Woop
|
|
@native
|
|
local function test()
|
|
|
|
end
|
|
)";
|
|
|
|
const std::string updated = R"(
|
|
--Woop
|
|
@native
|
|
local function test()
|
|
|
|
end
|
|
function a
|
|
)";
|
|
autocompleteFragmentInBothSolvers(source, updated, Position{6, 10}, [](FragmentAutocompleteStatusResult& result) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "method_call_inside_function_body")
|
|
{
|
|
const std::string source = R"(
|
|
local game = { GetService=function(s) return 'hello' end }
|
|
|
|
function a()
|
|
|
|
end
|
|
)";
|
|
|
|
const std::string updated = R"(
|
|
local game = { GetService=function(s) return 'hello' end }
|
|
|
|
function a()
|
|
game:
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{4, 17},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto ac = frag.result->acResults;
|
|
CHECK_NE(0, ac.entryMap.size());
|
|
|
|
LUAU_CHECK_HAS_NO_KEY(ac.entryMap, "math");
|
|
CHECK_EQ(ac.context, AutocompleteContext::Property);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "tbl_function_parameter")
|
|
{
|
|
const std::string source = R"(
|
|
--!strict
|
|
type Foo = {x : number, y : number}
|
|
local function func(abc : Foo)
|
|
abc.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 7},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK_EQ(2, frag.result->acResults.entryMap.size());
|
|
CHECK(frag.result->acResults.entryMap.count("x"));
|
|
CHECK(frag.result->acResults.entryMap.count("y"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "tbl_local_function_parameter")
|
|
{
|
|
const std::string source = R"(
|
|
--!strict
|
|
type Foo = {x : number, y : number}
|
|
local function func(abc : Foo)
|
|
abc.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 7},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK_EQ(2, frag.result->acResults.entryMap.size());
|
|
CHECK(frag.result->acResults.entryMap.count("x"));
|
|
CHECK(frag.result->acResults.entryMap.count("y"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "vec3_function_parameter")
|
|
{
|
|
const std::string source = R"(
|
|
--!strict
|
|
local function func(abc : FakeVec)
|
|
abc.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 7},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK_EQ(2, frag.result->acResults.entryMap.size());
|
|
CHECK(frag.result->acResults.entryMap.count("zero"));
|
|
CHECK(frag.result->acResults.entryMap.count("dot"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "vec3_local_function_parameter")
|
|
{
|
|
const std::string source = R"(
|
|
--!strict
|
|
local function func(abc : FakeVec)
|
|
abc.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 7},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK_EQ(2, frag.result->acResults.entryMap.size());
|
|
CHECK(frag.result->acResults.entryMap.count("zero"));
|
|
CHECK(frag.result->acResults.entryMap.count("dot"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "function_parameter_not_recommending_out_of_scope_argument")
|
|
{
|
|
const std::string source = R"(
|
|
--!strict
|
|
local function foo(abd: FakeVec)
|
|
end
|
|
local function bar(abc : FakeVec)
|
|
a
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{5, 5},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK(frag.result->acResults.entryMap.count("abc"));
|
|
CHECK(!frag.result->acResults.entryMap.count("abd"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "bad_range_1")
|
|
{
|
|
const std::string source = R"(
|
|
local t = 1
|
|
)";
|
|
const std::string updated = R"(
|
|
t
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{2, 1},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto opt = linearSearchForBinding(frag.result->freshScope, "t");
|
|
REQUIRE(opt);
|
|
CHECK_EQ("number", toString(*opt));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "bad_range_2")
|
|
{
|
|
const std::string source = R"(
|
|
local t = 1
|
|
)";
|
|
const std::string updated = R"(
|
|
local t = 1
|
|
t
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{2, 1},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
auto opt = linearSearchForBinding(frag.result->freshScope, "t");
|
|
REQUIRE(opt);
|
|
CHECK_EQ("number", toString(*opt));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "bad_range_3")
|
|
{
|
|
// This test makes less sense since we don't have an updated check that
|
|
// includes l
|
|
// instead this will recommend nothing useful because `local t` hasn't
|
|
// been typechecked in the fresh module
|
|
const std::string source = R"(
|
|
l
|
|
)";
|
|
const std::string updated = R"(
|
|
local t = 1
|
|
l
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{2, 1},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.status == FragmentAutocompleteStatus::Success);
|
|
REQUIRE(frag.result);
|
|
}
|
|
);
|
|
}
|
|
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "do_not_recommend_results_in_multiline_comment")
|
|
{
|
|
ScopedFastFlag sff = {FFlag::LuauBetterCursorInCommentDetection, true};
|
|
std::string source = R"(--[[
|
|
)";
|
|
std::string dest = R"(--[[
|
|
a
|
|
)";
|
|
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{1, 1},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments_simple")
|
|
{
|
|
ScopedFastFlag sff = {FFlag::LuauBetterCursorInCommentDetection, true};
|
|
const std::string source = R"(
|
|
-- sel
|
|
-- retur
|
|
-- fo
|
|
-- if
|
|
-- end
|
|
-- the
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 6},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments_blocks")
|
|
{
|
|
const std::string source = R"(
|
|
--[[
|
|
comment 1
|
|
]] local
|
|
-- [[ comment 2]]
|
|
--
|
|
-- sdfsdfsdf
|
|
--[[comment 3]]
|
|
--[[
|
|
foo
|
|
bar
|
|
baz
|
|
]]
|
|
)";
|
|
ScopedFastFlag sff{FFlag::LuauBetterCursorInCommentDetection, true};
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 2},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK(!frag.result->acResults.entryMap.empty());
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{8, 6},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{10, 0},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments")
|
|
{
|
|
ScopedFastFlag sff = {FFlag::LuauBetterCursorInCommentDetection, true};
|
|
const std::string source = R"(
|
|
-- sel
|
|
-- retur
|
|
-- fo
|
|
--[[ sel ]]
|
|
local -- hello
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{1, 7},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{2, 9},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{3, 6},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{4, 9},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{5, 6},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK(!frag.result->acResults.entryMap.empty());
|
|
}
|
|
);
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{5, 14},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments_in_incremental_fragment")
|
|
{
|
|
const std::string source = R"(
|
|
local x = 5
|
|
if x == 5
|
|
)";
|
|
const std::string updated = R"(
|
|
local x = 5
|
|
if x == 5 then -- a comment
|
|
)";
|
|
ScopedFastFlag sff = {FFlag::LuauBetterCursorInCommentDetection, true};
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{2, 28},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
CHECK(frag.result == std::nullopt);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "fragment_autocomplete_handles_parse_errors")
|
|
{
|
|
|
|
ScopedFastInt sfi{FInt::LuauParseErrorLimit, 1};
|
|
const std::string source = R"(
|
|
|
|
)";
|
|
const std::string updated = R"(
|
|
type A = <>random non code text here
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
updated,
|
|
Position{1, 38},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK(frag.result->acResults.entryMap.empty());
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "require_tracing")
|
|
{
|
|
fileResolver.source["MainModule/A"] = R"(
|
|
return { x = 0 }
|
|
)";
|
|
|
|
fileResolver.source["MainModule"] = R"(
|
|
local result = require(script.A)
|
|
local x = 1 + result.
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
fileResolver.source["MainModule"],
|
|
fileResolver.source["MainModule"],
|
|
Position{2, 21},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
CHECK(frag.result->acResults.entryMap.size() == 1);
|
|
CHECK(frag.result->acResults.entryMap.count("x"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "fragment_ac_must_traverse_typeof_and_not_ice")
|
|
{
|
|
// This test ensures that we traverse typeof expressions for defs that are being referred to in the fragment
|
|
// In this case, we want to ensure we populate the incremental environment with the reference to `m`
|
|
// Without this, we would ice as we will refer to the local `m` before it's declaration
|
|
ScopedFastFlag sff{FFlag::LuauMixedModeDefFinderTraversesTypeOf, true};
|
|
const std::string source = R"(
|
|
--!strict
|
|
local m = {}
|
|
-- and here
|
|
function m:m1() end
|
|
type nt = typeof(m)
|
|
|
|
return m
|
|
)";
|
|
const std::string updated = R"(
|
|
--!strict
|
|
local m = {}
|
|
-- and here
|
|
function m:m1() end
|
|
type nt = typeof(m)
|
|
l
|
|
return m
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(source, updated, Position{6, 2}, [](FragmentAutocompleteStatusResult& _) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "duped_alias")
|
|
{
|
|
const std::string source = R"(
|
|
type a = typeof({})
|
|
|
|
)";
|
|
const std::string dest = R"(
|
|
type a = typeof({})
|
|
type a = typeof({})
|
|
)";
|
|
|
|
// Re-parsing and typechecking a type alias in the fragment that was defined in the base module will assert in ConstraintGenerator::checkAliases
|
|
// unless we don't clone it This will let the incremental pass re-generate the type binding, and we will expect to see it in the type bindings
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{2, 20},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result);
|
|
Scope* sc = frag.result->freshScope;
|
|
CHECK(1 == sc->privateTypeBindings.count("a"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "mutually_recursive_alias")
|
|
{
|
|
const std::string source = R"(
|
|
type U = {f : number, g : U}
|
|
|
|
)";
|
|
const std::string dest = R"(
|
|
type U = {f : number, g : V}
|
|
type V = {h : number, i : U?}
|
|
)";
|
|
|
|
// Re-parsing and typechecking a type alias in the fragment that was defined in the base module will assert in ConstraintGenerator::checkAliases
|
|
// unless we don't clone it This will let the incremental pass re-generate the type binding, and we will expect to see it in the type bindings
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{2, 30},
|
|
[](FragmentAutocompleteStatusResult& frag)
|
|
{
|
|
REQUIRE(frag.result->freshScope);
|
|
Scope* scope = frag.result->freshScope;
|
|
CHECK(1 == scope->privateTypeBindings.count("U"));
|
|
CHECK(1 == scope->privateTypeBindings.count("V"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "generalization_crash_when_old_solver_freetypes_have_no_bounds_set")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauFreeTypesMustHaveBounds, true};
|
|
const std::string source = R"(
|
|
local UserInputService = game:GetService("UserInputService");
|
|
|
|
local Camera = workspace.CurrentCamera;
|
|
|
|
UserInputService.InputBegan:Connect(function(Input)
|
|
if (Input.KeyCode == Enum.KeyCode.One) then
|
|
local Up = Input.Foo
|
|
local Vector = -(Up:Unit)
|
|
end
|
|
end)
|
|
)";
|
|
|
|
const std::string dest = R"(
|
|
local UserInputService = game:GetService("UserInputService");
|
|
|
|
local Camera = workspace.CurrentCamera;
|
|
|
|
UserInputService.InputBegan:Connect(function(Input)
|
|
if (Input.KeyCode == Enum.KeyCode.One) then
|
|
local Up = Input.Foo
|
|
local Vector = -(Up:Unit())
|
|
end
|
|
end)
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(source, dest, Position{8, 36}, [](FragmentAutocompleteStatusResult& _) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "fragment_autocomplete_ensures_memory_isolation")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauCloneIncrementalModule, true};
|
|
ToStringOptions opt;
|
|
opt.exhaustive = true;
|
|
opt.exhaustive = true;
|
|
opt.functionTypeArguments = true;
|
|
opt.maxTableLength = 0;
|
|
opt.maxTypeLength = 0;
|
|
|
|
auto checkAndExamine = [&](const std::string& src, const std::string& idName, const std::string& idString)
|
|
{
|
|
checkWithOptions(src);
|
|
auto id = getType(idName, true);
|
|
LUAU_ASSERT(id);
|
|
CHECK_EQ(Luau::toString(*id, opt), idString);
|
|
};
|
|
|
|
auto getTypeFromModule = [](ModulePtr module, const std::string& name) -> std::optional<TypeId>
|
|
{
|
|
if (!module->hasModuleScope())
|
|
return std::nullopt;
|
|
return lookupName(module->getModuleScope(), name);
|
|
};
|
|
|
|
auto fragmentACAndCheck = [&](const std::string& updated, const Position& pos, const std::string& idName)
|
|
{
|
|
FragmentAutocompleteStatusResult frag = autocompleteFragment(updated, pos, std::nullopt);
|
|
REQUIRE(frag.result);
|
|
auto fragId = getTypeFromModule(frag.result->incrementalModule, idName);
|
|
LUAU_ASSERT(fragId);
|
|
|
|
auto srcId = getType(idName, true);
|
|
LUAU_ASSERT(srcId);
|
|
|
|
CHECK((*fragId)->owningArena != (*srcId)->owningArena);
|
|
CHECK(&(frag.result->incrementalModule->internalTypes) == (*fragId)->owningArena);
|
|
};
|
|
|
|
const std::string source = R"(local module = {}
|
|
f
|
|
return module)";
|
|
|
|
const std::string updated1 = R"(local module = {}
|
|
function module.a
|
|
return module)";
|
|
|
|
const std::string updated2 = R"(local module = {}
|
|
function module.ab
|
|
return module)";
|
|
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, false};
|
|
checkAndExamine(source, "module", "{ }");
|
|
// [TODO] CLI-140762 we shouldn't mutate stale module in autocompleteFragment
|
|
// early return since the following checking will fail, which it shouldn't!
|
|
fragmentACAndCheck(updated1, Position{1, 17}, "module");
|
|
fragmentACAndCheck(updated2, Position{1, 18}, "module");
|
|
}
|
|
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
checkAndExamine(source, "module", "{ }");
|
|
// [TODO] CLI-140762 we shouldn't mutate stale module in autocompleteFragment
|
|
// early return since the following checking will fail, which it shouldn't!
|
|
fragmentACAndCheck(updated1, Position{1, 17}, "module");
|
|
fragmentACAndCheck(updated2, Position{1, 18}, "module");
|
|
}
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "fragment_autocomplete_shouldnt_crash_on_cross_module_mutation")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauCloneIncrementalModule, true};
|
|
const std::string source = R"(local module = {}
|
|
function module.
|
|
return module
|
|
)";
|
|
|
|
const std::string updated = R"(local module = {}
|
|
function module.f
|
|
return module
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(source, updated, Position{1, 18}, [](FragmentAutocompleteStatusResult& result) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "ice_caused_by_mixed_mode_use")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauAutocompleteUsesModuleForTypeCompatibility, true};
|
|
const std::string source =
|
|
std::string("--[[\n\tPackage link auto-generated by Rotriever\n]]\nlocal PackageIndex = script.Parent._Index\n\nlocal Package = ") +
|
|
"require(PackageIndex[\"ReactOtter\"][\"ReactOtter\"])\n\nexport type Goal = Package.Goal\nexport type SpringOptions " +
|
|
"= Package.SpringOptions\n\n\nreturn Pa";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{11, 9},
|
|
[](FragmentAutocompleteStatusResult& _) {
|
|
|
|
}
|
|
);
|
|
autocompleteFragmentInBothSolvers(source, source, Position{11, 9}, [](auto& _) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "free_type_in_old_solver_shouldnt_trigger_not_null_assertion")
|
|
{
|
|
|
|
const std::string source = R"(--!strict
|
|
local foo
|
|
local a, z = foo()
|
|
|
|
local e = foo().x
|
|
|
|
local f = foo().y
|
|
|
|
z
|
|
)";
|
|
|
|
const std::string dest = R"(--!strict
|
|
local foo
|
|
local a, z = foo()
|
|
|
|
local e = foo().x
|
|
|
|
local f = foo().y
|
|
|
|
z:a
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(source, dest, Position{8, 3}, [](FragmentAutocompleteStatusResult& _) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "interior_free_types_assertion_caused_by_free_type_inheriting_null_scope_from_table")
|
|
{
|
|
ScopedFastFlag sff{FFlag::LuauTrackInteriorFreeTypesOnScope, true};
|
|
const std::string source = R"(--!strict
|
|
local foo
|
|
local a = foo()
|
|
|
|
local e = foo().x
|
|
|
|
local f = foo().y
|
|
|
|
|
|
)";
|
|
|
|
const std::string dest = R"(--!strict
|
|
local foo
|
|
local a = foo()
|
|
|
|
local e = foo().x
|
|
|
|
local f = foo().y
|
|
|
|
z = a.P.E
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(source, dest, Position{8, 9}, [](FragmentAutocompleteStatusResult& _) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "NotNull_nil_scope_assertion_caused_by_free_type_inheriting_null_scope_from_table")
|
|
{
|
|
const std::string source = R"(--!strict
|
|
local foo
|
|
local a = foo()
|
|
|
|
local e = foo().x
|
|
|
|
local f = foo().y
|
|
|
|
|
|
)";
|
|
|
|
const std::string dest = R"(--!strict
|
|
local foo
|
|
local a = foo()
|
|
|
|
local e = foo().x
|
|
|
|
local f = foo().y
|
|
|
|
z = a.P.E
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(source, dest, Position{8, 9}, [](FragmentAutocompleteStatusResult& _) {});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "user_defined_type_function_local")
|
|
{
|
|
ScopedFastFlag luauUserTypeFunTypecheck{FFlag::LuauUserTypeFunTypecheck, true};
|
|
|
|
const std::string source = R"(--!strict
|
|
type function foo(x: type): type
|
|
if x.tag == "singleton" then
|
|
local t = x:value()
|
|
|
|
return types.unionof(types.singleton(t), types.singleton(nil))
|
|
end
|
|
|
|
return types.number
|
|
end
|
|
)";
|
|
|
|
const std::string dest = R"(--!strict
|
|
type function foo(x: type): type
|
|
if x.tag == "singleton" then
|
|
local t = x:value()
|
|
x
|
|
return types.unionof(types.singleton(t), types.singleton(nil))
|
|
end
|
|
|
|
return types.number
|
|
end
|
|
)";
|
|
|
|
// Only checking in new solver as old solver doesn't handle type functions and constraint solver will ICE
|
|
ScopedFastFlag sff{FFlag::LuauSolverV2, true};
|
|
this->check(source, getOptions());
|
|
|
|
FragmentAutocompleteStatusResult result = autocompleteFragment(dest, Position{4, 9}, std::nullopt);
|
|
CHECK(result.status != FragmentAutocompleteStatus::InternalIce);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "for_loop_recommends")
|
|
{
|
|
const std::string source = R"(
|
|
local testArr: {{a: number, b: number}} = {
|
|
{a = 1, b = 2},
|
|
{a = 2, b = 4},
|
|
}
|
|
|
|
for _, v in testArr do
|
|
|
|
end
|
|
)";
|
|
|
|
const std::string dest = R"(
|
|
local testArr: {{a: number, b: number}} = {
|
|
{a = 1, b = 2},
|
|
{a = 2, b = 4},
|
|
}
|
|
|
|
for _, v in testArr do
|
|
print(v.
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{7, 12},
|
|
[](FragmentAutocompleteStatusResult& result)
|
|
{
|
|
CHECK(result.status != FragmentAutocompleteStatus::InternalIce);
|
|
CHECK(result.result);
|
|
CHECK(!result.result->acResults.entryMap.empty());
|
|
CHECK(result.result->acResults.entryMap.count("a"));
|
|
CHECK(result.result->acResults.entryMap.count("b"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "for_loop_recommends")
|
|
{
|
|
const std::string source = R"(
|
|
local testArr: {string} = {
|
|
"a",
|
|
"b",
|
|
}
|
|
|
|
for _, v in testArr do
|
|
|
|
end
|
|
)";
|
|
|
|
const std::string dest = R"(
|
|
local testArr: {string} = {
|
|
"a",
|
|
"b",
|
|
}
|
|
|
|
for _, v in testArr do
|
|
print(v:)
|
|
end
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{7, 12},
|
|
[](FragmentAutocompleteStatusResult& result)
|
|
{
|
|
CHECK(result.status != FragmentAutocompleteStatus::InternalIce);
|
|
CHECK(result.result);
|
|
CHECK(!result.result->acResults.entryMap.empty());
|
|
CHECK(result.result->acResults.entryMap.count("upper"));
|
|
CHECK(result.result->acResults.entryMap.count("sub"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "expr_function")
|
|
{
|
|
const std::string source = R"(
|
|
local t = {}
|
|
type Input = {x : string}
|
|
function t.Do(fn : (Input) -> ())
|
|
if t.x == "a" then
|
|
return
|
|
end
|
|
end
|
|
|
|
t.Do(function (f)
|
|
f
|
|
end)
|
|
)";
|
|
|
|
const std::string dest = R"(
|
|
local t = {}
|
|
type Input = {x : string}
|
|
function t.Do(fn : (Input) -> ())
|
|
if t.x == "a" then
|
|
return
|
|
end
|
|
end
|
|
|
|
t.Do(function (f)
|
|
f.
|
|
end)
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{10, 6},
|
|
[](FragmentAutocompleteStatusResult& status)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == status.status);
|
|
REQUIRE(status.result);
|
|
CHECK(!status.result->acResults.entryMap.empty());
|
|
CHECK(status.result->acResults.entryMap.count("x"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "differ_1")
|
|
{
|
|
const std::string source = R"()";
|
|
const std::string dest = R"(local tbl = { foo = 1, bar = 2 };
|
|
tbl.b)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{1, 5},
|
|
[](FragmentAutocompleteStatusResult& status)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == status.status);
|
|
REQUIRE(status.result);
|
|
CHECK(!status.result->acResults.entryMap.empty());
|
|
CHECK(status.result->acResults.entryMap.count("foo"));
|
|
CHECK(status.result->acResults.entryMap.count("bar"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_test_both_empty")
|
|
{
|
|
SourceModule stale;
|
|
SourceModule fresh;
|
|
ParseResult o = parseHelper_(stale, R"()");
|
|
ParseResult n = parseHelper_(stale, R"()");
|
|
auto pos = Luau::blockDiffStart(o.root, n.root, nullptr);
|
|
CHECK(pos == std::nullopt);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_test_both_empty_e2e")
|
|
{
|
|
const std::string source = R"()";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
source,
|
|
Position{0, 0},
|
|
[](FragmentAutocompleteStatusResult& result)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == result.status);
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_1")
|
|
{
|
|
SourceModule stale;
|
|
SourceModule fresh;
|
|
ParseResult o = parseHelper_(stale, R"()");
|
|
ParseResult n = parseHelper_(fresh, R"(local x = 4
|
|
local y = 3
|
|
local z = 3)");
|
|
auto pos = Luau::blockDiffStart(o.root, n.root, n.root->body.data[2]);
|
|
REQUIRE(pos);
|
|
CHECK(*pos == Position{0, 0});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_1_e2e")
|
|
{
|
|
const std::string source = R"()";
|
|
const std::string dest = R"(local f1 = 4
|
|
local f2 = "a"
|
|
local f3 = f
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{2, 12},
|
|
[](FragmentAutocompleteStatusResult& result)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == result.status);
|
|
REQUIRE(result.result);
|
|
CHECK(!result.result->acResults.entryMap.empty());
|
|
CHECK(result.result->acResults.entryMap.count("f1"));
|
|
CHECK(result.result->acResults.entryMap.count("f2"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_1_e2e_in_the_middle")
|
|
{
|
|
const std::string source = R"()";
|
|
const std::string dest = R"(local f1 = 4
|
|
local f2 = f
|
|
local f3 = f
|
|
)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{1, 12},
|
|
[](FragmentAutocompleteStatusResult& result)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == result.status);
|
|
REQUIRE(result.result);
|
|
CHECK(!result.result->acResults.entryMap.empty());
|
|
CHECK(result.result->acResults.entryMap.count("f1"));
|
|
CHECK(!result.result->acResults.entryMap.count("f3"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_2")
|
|
{
|
|
SourceModule stale;
|
|
SourceModule fresh;
|
|
ParseResult o = parseHelper_(stale, R"(local x = 4)");
|
|
ParseResult n = parseHelper_(fresh, R"(local x = 4
|
|
local y = 3
|
|
local z = 3)");
|
|
auto pos = Luau::blockDiffStart(o.root, n.root, n.root->body.data[1]);
|
|
REQUIRE(pos);
|
|
CHECK(*pos == Position{1, 0});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_3")
|
|
{
|
|
SourceModule stale;
|
|
SourceModule fresh;
|
|
ParseResult o = parseHelper_(stale, R"(local x = 4
|
|
local y = 2 + 1)");
|
|
ParseResult n = parseHelper_(fresh, R"(local x = 4
|
|
local y = 3
|
|
local z = 3
|
|
local foo = 8)");
|
|
auto pos = Luau::blockDiffStart(o.root, n.root, n.root->body.data[3]);
|
|
REQUIRE(pos);
|
|
CHECK(*pos == Position{1, 0});
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_3")
|
|
{
|
|
const std::string source = R"(local f1 = 4
|
|
local f2 = 2 + 1)";
|
|
const std::string dest = R"(local f1 = 4
|
|
local f2 = 3
|
|
local f3 = 3
|
|
local foo = 8 + )";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{3, 16},
|
|
[](FragmentAutocompleteStatusResult& result)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == result.status);
|
|
REQUIRE(result.result);
|
|
CHECK(!result.result->acResults.entryMap.empty());
|
|
CHECK(result.result->acResults.entryMap.count("f1"));
|
|
CHECK(result.result->acResults.entryMap.count("f2"));
|
|
CHECK(result.result->acResults.entryMap.count("f3"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "block_diff_added_locals_fake_similarity")
|
|
{
|
|
// Captures the bad behaviour of block based diffs
|
|
SourceModule stale;
|
|
SourceModule fresh;
|
|
ParseResult o = parseHelper_(stale, R"(local x = 4
|
|
local y = true
|
|
local z = 2 + 1)");
|
|
ParseResult n = parseHelper_(fresh, R"(local x = 4
|
|
local y = "tr"
|
|
local z = 3
|
|
local foo = 8)");
|
|
auto pos = Luau::blockDiffStart(o.root, n.root, n.root->body.data[2]);
|
|
REQUIRE(pos);
|
|
CHECK(*pos == Position{2, 0});
|
|
}
|
|
|
|
#if 0
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "TypeCorrectLocalReturn_assert")
|
|
{
|
|
const std::string source = R"()";
|
|
const std::string dest = R"(local function target(a: number, b: string) return a + #b end
|
|
local function bar1(a: string) reutrn a .. 'x' end
|
|
local function bar2(a: number) return -a end
|
|
return target(bar)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{3, 17},
|
|
[](FragmentAutocompleteStatusResult& status)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == status.status);
|
|
REQUIRE(status.result);
|
|
CHECK(!status.result->acResults.entryMap.empty());
|
|
CHECK(status.result->acResults.entryMap.count("bar1"));
|
|
CHECK(status.result->acResults.entryMap.count("bar2"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "TypeCorrectLocalRank_assert")
|
|
{
|
|
const std::string source = R"()";
|
|
const std::string dest = R"(local function target(a: number, b: string) return a + #b end
|
|
local bar1 = 'hello'
|
|
local bar2 = 4
|
|
return target(bar)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{3, 17},
|
|
[](FragmentAutocompleteStatusResult& status)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == status.status);
|
|
REQUIRE(status.result);
|
|
CHECK(!status.result->acResults.entryMap.empty());
|
|
CHECK(status.result->acResults.entryMap.count("bar1"));
|
|
CHECK(status.result->acResults.entryMap.count("bar2"));
|
|
}
|
|
);
|
|
}
|
|
#endif
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "str_metata_table_finished_defining")
|
|
{
|
|
const std::string source = R"(local function foobar(): string return "" end
|
|
local foo = f)";
|
|
const std::string dest = R"(local function foobar(): string return "" end
|
|
local foo = foobar()
|
|
foo:)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{2, 4},
|
|
[](FragmentAutocompleteStatusResult& res)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == res.status);
|
|
REQUIRE(res.result);
|
|
CHECK(!res.result->acResults.entryMap.empty());
|
|
CHECK(res.result->acResults.entryMap.count("len"));
|
|
CHECK(res.result->acResults.entryMap.count("gsub"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "str_metata_table_redef")
|
|
{
|
|
const std::string source = R"(local x = 42)";
|
|
const std::string dest = R"(local x = 42
|
|
local x = ""
|
|
x:)";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{2, 2},
|
|
[](FragmentAutocompleteStatusResult& res)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == res.status);
|
|
REQUIRE(res.result);
|
|
CHECK(!res.result->acResults.entryMap.empty());
|
|
CHECK(res.result->acResults.entryMap.count("len"));
|
|
CHECK(res.result->acResults.entryMap.count("gsub"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "diff_multiple_blocks_on_same_line")
|
|
{
|
|
const std::string source = R"(
|
|
do local function foo() end; local x = ""; end do local function bar() end)";
|
|
const std::string dest = R"(
|
|
do local function foo() end; local x = ""; end do local function bar() end local x = {a : number}; b end )";
|
|
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{1, 101},
|
|
[](FragmentAutocompleteStatusResult& res)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == res.status);
|
|
REQUIRE(res.result);
|
|
CHECK(!res.result->acResults.entryMap.empty());
|
|
CHECK(res.result->acResults.entryMap.count("bar"));
|
|
CHECK(!res.result->acResults.entryMap.count("foo"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "nested_blocks_else_simple")
|
|
{
|
|
const std::string source = R"(
|
|
local function foo(t : {foo : string})
|
|
local x = t.foo
|
|
do
|
|
if t then
|
|
end
|
|
end
|
|
end
|
|
)";
|
|
const std::string dest = R"(
|
|
local function foo(t : {foo : string})
|
|
local x = t.foo
|
|
do
|
|
if t then
|
|
x:
|
|
end
|
|
end
|
|
end
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{5, 14},
|
|
[](FragmentAutocompleteStatusResult& res)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == res.status);
|
|
REQUIRE(res.result);
|
|
CHECK(!res.result->acResults.entryMap.empty());
|
|
CHECK(res.result->acResults.entryMap.count("gsub"));
|
|
CHECK(res.result->acResults.entryMap.count("len"));
|
|
}
|
|
);
|
|
}
|
|
|
|
TEST_CASE_FIXTURE(FragmentAutocompleteBuiltinsFixture, "nested_blocks_else_difficult_2")
|
|
{
|
|
const std::string source = R"(
|
|
local function foo(t : {foo : number})
|
|
do
|
|
if t then
|
|
end
|
|
end
|
|
end
|
|
)";
|
|
const std::string dest = R"(
|
|
local function foo(t : {foo : number})
|
|
do
|
|
if t then
|
|
else
|
|
local x = 4
|
|
return x + t.
|
|
end
|
|
end
|
|
end
|
|
)";
|
|
autocompleteFragmentInBothSolvers(
|
|
source,
|
|
dest,
|
|
Position{6, 24},
|
|
[](FragmentAutocompleteStatusResult& res)
|
|
{
|
|
CHECK(FragmentAutocompleteStatus::Success == res.status);
|
|
REQUIRE(res.result);
|
|
CHECK(!res.result->acResults.entryMap.empty());
|
|
CHECK(res.result->acResults.entryMap.count("foo"));
|
|
}
|
|
);
|
|
}
|
|
// NOLINTEND(bugprone-unchecked-optional-access)
|
|
|
|
TEST_SUITE_END();
|