// 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 #include #include #include #include #include #include using namespace Luau; LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete) LUAU_FASTFLAG(LuauIncrementalAutocompleteCommentDetection) 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(LuauModuleHoldsAstRoot) 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) static std::optional nullCallback(std::string tag, std::optional ptr, std::optional 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 struct FragmentAutocompleteFixtureImpl : BaseType { static_assert(std::is_base_of_v, "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 luauModuleHoldsAstRoot{FFlag::LuauModuleHoldsAstRoot, 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}; FragmentAutocompleteFixtureImpl() : BaseType(true) { } CheckResult checkWithOptions(const std::string& source) { return this->check(source, getOptions()); } ParseResult parseHelper(std::string document) { SourceModule& source = getSource(); ParseOptions parseOptions; parseOptions.captureComments = true; ParseResult parseResult = Parser::parse(document.c_str(), document.length(), *source.names, *source.allocator, parseOptions); return parseResult; } 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 parseFragment( const std::string& document, const Position& cursorPos, std::optional 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 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 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 assertions, std::optional 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 typecheckFragmentForModule( const ModuleName& module, const std::string& document, Position cursorPos, std::optional 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 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(); return *source; } private: std::unique_ptr source = std::make_unique(); }; struct FragmentAutocompleteFixture : FragmentAutocompleteFixtureImpl { FragmentAutocompleteFixture() : FragmentAutocompleteFixtureImpl() { 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 { FragmentAutocompleteBuiltinsFixture() : FragmentAutocompleteFixtureImpl() { 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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()); } 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(); 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(); 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(); 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(); 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(); 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(); 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(); 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(); REQUIRE(bin); CHECK_EQ(AstExprBinary::Op::Add, bin->op); auto lhs = bin->left->as(); auto rhs = bin->right->as(); 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(); 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(); REQUIRE(bin); CHECK_EQ(AstExprBinary::Op::Add, bin->op); auto lhs = bin->left->as(); auto rhs = bin->right->as(); 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()); CHECK_GE(callFragment->ancestry.size(), 2); AstNode* back = callFragment->ancestry.back(); CHECK(back->is()); 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()); 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()); CHECK_GE(stringFragment->ancestry.size(), 1); back = stringFragment->ancestry.back(); auto asString = back->as(); 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()); CHECK_GE(fragment->ancestry.size(), 2); AstNode* back = fragment->ancestry.back(); CHECK(back->is()); 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()); 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(); // NOLINTEND(bugprone-unchecked-optional-access) 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(FragmentAutocompleteFixture, "typecheck_fragment_handles_stale_module") { ScopedFastFlag sff(FFlag::LuauModuleHoldsAstRoot, false); const std::string sourceName = "MainModule"; fileResolver.source[sourceName] = "local x = 5"; CheckResult checkResult = frontend.check(sourceName, getOptions()); LUAU_REQUIRE_NO_ERRORS(checkResult); auto [result, _] = typecheckFragmentForModule(sourceName, fileResolver.source[sourceName], Luau::Position(0, 0)); CHECK_EQ(result, FragmentTypeCheckStatus::Success); frontend.markDirty(sourceName); frontend.parse(sourceName); CHECK_NE(frontend.getSourceModule(sourceName), nullptr); auto [result2, __] = typecheckFragmentForModule(sourceName, fileResolver.source[sourceName], Luau::Position(0, 0)); CHECK_EQ(result2, FragmentTypeCheckStatus::SkipAutocomplete); } 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 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 { 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(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(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::LuauIncrementalAutocompleteCommentDetection, true}, {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::LuauIncrementalAutocompleteCommentDetection, true}, {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::LuauIncrementalAutocompleteCommentDetection, true}, {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::LuauIncrementalAutocompleteCommentDetection, true}, {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(FragmentAutocompleteFixture, "fragment_autocomplete_handles_stale_module") { ScopedFastFlag sff{FFlag::LuauModuleHoldsAstRoot, false}; const std::string sourceName = "MainModule"; fileResolver.source[sourceName] = "local x = 5"; frontend.check(sourceName, getOptions()); frontend.markDirty(sourceName); frontend.parse(sourceName); FragmentAutocompleteStatusResult frag = autocompleteFragmentForModule(sourceName, fileResolver.source[sourceName], Luau::Position(0, 0)); REQUIRE(frag.result); CHECK(frag.result->acResults.entryMap.empty()); CHECK_EQ(frag.result->incrementalModule, nullptr); } 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 { 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") { ScopedFastFlag sff{FFlag::LuauTrackInteriorFreeTypesOnScope, false}; 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_SUITE_END();