// 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 "Luau/Ast.h" #include "Luau/AstQuery.h" #include "Luau/Autocomplete.h" #include "Luau/Common.h" #include "Luau/EqSatSimplification.h" #include "Luau/ModuleResolver.h" #include "Luau/Parser.h" #include "Luau/ParseOptions.h" #include "Luau/Module.h" #include "Luau/TimeTrace.h" #include "Luau/UnifierSharedState.h" #include "Luau/TypeFunction.h" #include "Luau/DataFlowGraph.h" #include "Luau/ConstraintGenerator.h" #include "Luau/ConstraintSolver.h" #include "Luau/Frontend.h" #include "Luau/Parser.h" #include "Luau/ParseOptions.h" #include "Luau/Module.h" #include "Luau/Clone.h" #include "AutocompleteCore.h" LUAU_FASTINT(LuauTypeInferRecursionLimit); LUAU_FASTINT(LuauTypeInferIterationLimit); LUAU_FASTINT(LuauTarjanChildLimit) LUAU_FASTFLAG(LuauAllowFragmentParsing); LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete) LUAU_FASTFLAGVARIABLE(LuauIncrementalAutocompleteBugfixes) LUAU_FASTFLAG(LuauReferenceAllocatorInNewSolver) LUAU_FASTFLAGVARIABLE(LuauMixedModeDefFinderTraversesTypeOf) LUAU_FASTFLAG(LuauBetterReverseDependencyTracking) LUAU_FASTFLAGVARIABLE(LuauCloneIncrementalModule) namespace { template void copyModuleVec(std::vector& result, const std::vector& input) { result.insert(result.end(), input.begin(), input.end()); } template void copyModuleMap(Luau::DenseHashMap& result, const Luau::DenseHashMap& input) { for (auto [k, v] : input) result[k] = v; } } // namespace namespace Luau { template void cloneModuleMap(TypeArena& destArena, CloneState& cloneState, const Luau::DenseHashMap& source, Luau::DenseHashMap& dest) { for (auto [k, v] : source) { dest[k] = Luau::clone(v, destArena, cloneState); } } struct MixedModeIncrementalTCDefFinder : public AstVisitor { bool visit(AstExprLocal* local) override { referencedLocalDefs.emplace_back(local->local, local); return true; } bool visit(AstTypeTypeof* node) override { // We need to traverse typeof expressions because they may refer to locals that we need // to populate the local environment for fragment typechecking. For example, `typeof(m)` // requires that we find the local/global `m` and place it in the environment. // The default behaviour here is to return false, and have individual visitors override // the specific behaviour they need. return FFlag::LuauMixedModeDefFinderTraversesTypeOf; } // ast defs is just a mapping from expr -> def in general // will get built up by the dfg builder // localDefs, we need to copy over std::vector> referencedLocalDefs; }; void cloneAndSquashScopes( CloneState& cloneState, const Scope* staleScope, const ModulePtr& staleModule, NotNull destArena, NotNull dfg, AstStatBlock* program, Scope* destScope ) { std::vector scopes; for (const Scope* current = staleScope; current; current = current->parent.get()) { scopes.emplace_back(current); } // in reverse order (we need to clone the parents and override defs as we go down the list) for (auto it = scopes.rbegin(); it != scopes.rend(); ++it) { const Scope* curr = *it; // Clone the lvalue types for (const auto& [def, ty] : curr->lvalueTypes) destScope->lvalueTypes[def] = Luau::clone(ty, *destArena, cloneState); // Clone the rvalueRefinements for (const auto& [def, ty] : curr->rvalueRefinements) destScope->rvalueRefinements[def] = Luau::clone(ty, *destArena, cloneState); for (const auto& [n, m] : curr->importedTypeBindings) { std::unordered_map importedBindingTypes; for (const auto& [v, tf] : m) importedBindingTypes[v] = Luau::clone(tf, *destArena, cloneState); destScope->importedTypeBindings[n] = m; } // Finally, clone up the bindings for (const auto& [s, b] : curr->bindings) { destScope->bindings[s] = Luau::clone(b, *destArena, cloneState); } } // The above code associates defs with TypeId's in the scope // so that lookup to locals will succeed. MixedModeIncrementalTCDefFinder finder; program->visit(&finder); std::vector> locals = std::move(finder.referencedLocalDefs); for (auto [loc, expr] : locals) { if (std::optional binding = staleScope->linearSearchForBinding(loc->name.value, true)) { destScope->lvalueTypes[dfg->getDef(expr)] = Luau::clone(binding->typeId, *destArena, cloneState); } } return; } static FrontendModuleResolver& getModuleResolver(Frontend& frontend, std::optional options) { if (FFlag::LuauSolverV2 || !options) return frontend.moduleResolver; return options->forAutocomplete ? frontend.moduleResolverForAutocomplete : frontend.moduleResolver; } FragmentAutocompleteAncestryResult findAncestryForFragmentParse(AstStatBlock* root, const Position& cursorPos) { std::vector ancestry = findAncestryAtPositionForAutocomplete(root, cursorPos); // Should always contain the root AstStat LUAU_ASSERT(ancestry.size() >= 1); DenseHashMap localMap{AstName()}; std::vector localStack; AstStat* nearestStatement = nullptr; for (AstNode* node : ancestry) { if (auto block = node->as()) { for (auto stat : block->body) { if (stat->location.begin <= cursorPos) nearestStatement = stat; if (stat->location.begin < cursorPos && stat->location.begin.line < cursorPos.line) { // This statement precedes the current one if (auto loc = stat->as()) { for (auto v : loc->vars) { localStack.push_back(v); localMap[v->name] = v; } } else if (auto locFun = stat->as()) { localStack.push_back(locFun->name); localMap[locFun->name->name] = locFun->name; if (locFun->location.contains(cursorPos)) { for (AstLocal* loc : locFun->func->args) { localStack.push_back(loc); localMap[loc->name] = loc; } } } else if (auto globFun = stat->as()) { if (globFun->location.contains(cursorPos)) { for (AstLocal* loc : globFun->func->args) { localStack.push_back(loc); localMap[loc->name] = loc; } } } } } } } if (!nearestStatement) nearestStatement = ancestry[0]->asStat(); LUAU_ASSERT(nearestStatement); return {std::move(localMap), std::move(localStack), std::move(ancestry), std::move(nearestStatement)}; } /** * Get document offsets is a function that takes a source text document as well as a start position and end position(line, column) in that * document and attempts to get the concrete text between those points. It returns a pair of: * - start offset that represents an index in the source `char*` corresponding to startPos * - length, that represents how many more bytes to read to get to endPos. * Example - your document is "foo bar baz" and getDocumentOffsets is passed (0, 4), (0, 8). This function returns the pair {3, 5} * which corresponds to the string " bar " */ std::pair getDocumentOffsets(const std::string_view& src, const Position& startPos, const Position& endPos) { size_t lineCount = 0; size_t colCount = 0; size_t docOffset = 0; size_t startOffset = 0; size_t endOffset = 0; bool foundStart = false; bool foundEnd = false; for (char c : src) { if (foundStart && foundEnd) break; if (startPos.line == lineCount && startPos.column == colCount) { foundStart = true; startOffset = docOffset; } if (endPos.line == lineCount && endPos.column == colCount) { endOffset = docOffset; while (endOffset < src.size() && src[endOffset] != '\n') endOffset++; foundEnd = true; } // We put a cursor position that extends beyond the extents of the current line if (foundStart && !foundEnd && (lineCount > endPos.line)) { foundEnd = true; endOffset = docOffset - 1; } if (c == '\n') { lineCount++; colCount = 0; } else { colCount++; } docOffset++; } if (foundStart && !foundEnd) endOffset = src.length(); size_t min = std::min(startOffset, endOffset); size_t len = std::max(startOffset, endOffset) - min; return {min, len}; } ScopePtr findClosestScope(const ModulePtr& module, const AstStat* nearestStatement) { LUAU_ASSERT(module->hasModuleScope()); ScopePtr closest = module->getModuleScope(); // find the scope the nearest statement belonged to. for (auto [loc, sc] : module->scopes) { if (loc.encloses(nearestStatement->location) && closest->location.begin <= loc.begin) closest = sc; } return closest; } std::optional parseFragment( const SourceModule& srcModule, std::string_view src, const Position& cursorPos, std::optional fragmentEndPosition ) { FragmentAutocompleteAncestryResult result = findAncestryForFragmentParse(srcModule.root, cursorPos); AstStat* nearestStatement = result.nearestStatement; const Location& rootSpan = srcModule.root->location; // Did we append vs did we insert inline bool appended = cursorPos >= rootSpan.end; // statement spans multiple lines bool multiline = nearestStatement->location.begin.line != nearestStatement->location.end.line; const Position endPos = fragmentEndPosition.value_or(cursorPos); // We start by re-parsing everything (we'll refine this as we go) Position startPos = srcModule.root->location.begin; // If we added to the end of the sourceModule, use the end of the nearest location if (appended && multiline) startPos = nearestStatement->location.end; // Statement spans one line && cursorPos is either on the same line or after else if (!multiline && cursorPos.line >= nearestStatement->location.end.line) startPos = nearestStatement->location.begin; else if (multiline && nearestStatement->location.end.line < cursorPos.line) startPos = nearestStatement->location.end; else startPos = nearestStatement->location.begin; auto [offsetStart, parseLength] = getDocumentOffsets(src, startPos, endPos); const char* srcStart = src.data() + offsetStart; std::string_view dbg = src.substr(offsetStart, parseLength); const std::shared_ptr& nameTbl = srcModule.names; FragmentParseResult fragmentResult; fragmentResult.fragmentToParse = std::string(dbg.data(), parseLength); // For the duration of the incremental parse, we want to allow the name table to re-use duplicate names ParseOptions opts; opts.allowDeclarationSyntax = false; opts.captureComments = true; opts.parseFragment = FragmentParseResumeSettings{std::move(result.localMap), std::move(result.localStack), startPos}; ParseResult p = Luau::Parser::parse(srcStart, parseLength, *nameTbl, *fragmentResult.alloc.get(), opts); // This means we threw a ParseError and we should decline to offer autocomplete here. if (p.root == nullptr) return std::nullopt; std::vector fabricatedAncestry = std::move(result.ancestry); // Get the ancestry for the fragment at the offset cursor position. // Consumers have the option to request with fragment end position, so we cannot just use the end position of our parse result as the // cursor position. Instead, use the cursor position calculated as an offset from our start position. std::vector fragmentAncestry = findAncestryAtPositionForAutocomplete(p.root, cursorPos); fabricatedAncestry.insert(fabricatedAncestry.end(), fragmentAncestry.begin(), fragmentAncestry.end()); if (nearestStatement == nullptr) nearestStatement = p.root; fragmentResult.root = std::move(p.root); fragmentResult.ancestry = std::move(fabricatedAncestry); fragmentResult.nearestStatement = nearestStatement; fragmentResult.commentLocations = std::move(p.commentLocations); return fragmentResult; } ModulePtr cloneModule(CloneState& cloneState, const ModulePtr& source, std::unique_ptr alloc) { freeze(source->internalTypes); freeze(source->interfaceTypes); ModulePtr incremental = std::make_shared(); incremental->name = source->name; incremental->humanReadableName = source->humanReadableName; incremental->allocator = std::move(alloc); // Clone types cloneModuleMap(incremental->internalTypes, cloneState, source->astTypes, incremental->astTypes); cloneModuleMap(incremental->internalTypes, cloneState, source->astTypePacks, incremental->astTypePacks); cloneModuleMap(incremental->internalTypes, cloneState, source->astExpectedTypes, incremental->astExpectedTypes); cloneModuleMap(incremental->internalTypes, cloneState, source->astOverloadResolvedTypes, incremental->astOverloadResolvedTypes); cloneModuleMap(incremental->internalTypes, cloneState, source->astForInNextTypes, incremental->astForInNextTypes); copyModuleMap(incremental->astScopes, source->astScopes); return incremental; } ModulePtr copyModule(const ModulePtr& result, std::unique_ptr alloc) { ModulePtr incrementalModule = std::make_shared(); incrementalModule->name = result->name; incrementalModule->humanReadableName = "Incremental$" + result->humanReadableName; incrementalModule->internalTypes.owningModule = incrementalModule.get(); incrementalModule->interfaceTypes.owningModule = incrementalModule.get(); incrementalModule->allocator = std::move(alloc); // Don't need to keep this alive (it's already on the source module) copyModuleVec(incrementalModule->scopes, result->scopes); copyModuleMap(incrementalModule->astTypes, result->astTypes); copyModuleMap(incrementalModule->astTypePacks, result->astTypePacks); copyModuleMap(incrementalModule->astExpectedTypes, result->astExpectedTypes); // Don't need to clone astOriginalCallTypes copyModuleMap(incrementalModule->astOverloadResolvedTypes, result->astOverloadResolvedTypes); // Don't need to clone astForInNextTypes copyModuleMap(incrementalModule->astForInNextTypes, result->astForInNextTypes); // Don't need to clone astResolvedTypes // Don't need to clone astResolvedTypePacks // Don't need to clone upperBoundContributors copyModuleMap(incrementalModule->astScopes, result->astScopes); // Don't need to clone declared Globals; return incrementalModule; } void mixedModeCompatibility( const ScopePtr& bottomScopeStale, const ScopePtr& myFakeScope, const ModulePtr& stale, NotNull dfg, AstStatBlock* program ) { // This code does the following // traverse program // look for ast refs for locals // ask for the corresponding defId from dfg // given that defId, and that expression, in the incremental module, map lvalue types from defID to MixedModeIncrementalTCDefFinder finder; program->visit(&finder); std::vector> locals = std::move(finder.referencedLocalDefs); for (auto [loc, expr] : locals) { if (std::optional binding = bottomScopeStale->linearSearchForBinding(loc->name.value, true)) { myFakeScope->lvalueTypes[dfg->getDef(expr)] = binding->typeId; } } } FragmentTypeCheckResult typecheckFragment_( Frontend& frontend, AstStatBlock* root, const ModulePtr& stale, const ScopePtr& closestScope, const Position& cursorPos, std::unique_ptr astAllocator, const FrontendOptions& opts ) { freeze(stale->internalTypes); freeze(stale->interfaceTypes); CloneState cloneState{frontend.builtinTypes}; ModulePtr incrementalModule = FFlag::LuauCloneIncrementalModule ? cloneModule(cloneState, stale, std::move(astAllocator)) : copyModule(stale, std::move(astAllocator)); incrementalModule->checkedInNewSolver = true; unfreeze(incrementalModule->internalTypes); unfreeze(incrementalModule->interfaceTypes); /// Setup typecheck limits TypeCheckLimits limits; if (opts.moduleTimeLimitSec) limits.finishTime = TimeTrace::getClock() + *opts.moduleTimeLimitSec; else limits.finishTime = std::nullopt; limits.cancellationToken = opts.cancellationToken; /// Icehandler NotNull iceHandler{&frontend.iceHandler}; /// Make the shared state for the unifier (recursion + iteration limits) UnifierSharedState unifierState{iceHandler}; unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit; unifierState.counters.iterationLimit = limits.unifierIterationLimit.value_or(FInt::LuauTypeInferIterationLimit); /// Initialize the normalizer Normalizer normalizer{&incrementalModule->internalTypes, frontend.builtinTypes, NotNull{&unifierState}}; /// User defined type functions runtime TypeFunctionRuntime typeFunctionRuntime(iceHandler, NotNull{&limits}); /// Create a DataFlowGraph just for the surrounding context DataFlowGraph dfg = DataFlowGraphBuilder::build(root, NotNull{&incrementalModule->defArena}, NotNull{&incrementalModule->keyArena}, iceHandler); SimplifierPtr simplifier = newSimplifier(NotNull{&incrementalModule->internalTypes}, frontend.builtinTypes); FrontendModuleResolver& resolver = getModuleResolver(frontend, opts); /// Contraint Generator ConstraintGenerator cg{ incrementalModule, NotNull{&normalizer}, NotNull{simplifier.get()}, NotNull{&typeFunctionRuntime}, NotNull{&resolver}, frontend.builtinTypes, iceHandler, stale->getModuleScope(), nullptr, nullptr, NotNull{&dfg}, {} }; std::shared_ptr freshChildOfNearestScope = nullptr; if (FFlag::LuauCloneIncrementalModule) { freshChildOfNearestScope = std::make_shared(closestScope); incrementalModule->scopes.emplace_back(root->location, freshChildOfNearestScope); cg.rootScope = freshChildOfNearestScope.get(); cloneAndSquashScopes( cloneState, closestScope.get(), stale, NotNull{&incrementalModule->internalTypes}, NotNull{&dfg}, root, freshChildOfNearestScope.get() ); cg.visitFragmentRoot(freshChildOfNearestScope, root); } else { // Any additions to the scope must occur in a fresh scope cg.rootScope = stale->getModuleScope().get(); freshChildOfNearestScope = std::make_shared(closestScope); incrementalModule->scopes.emplace_back(root->location, freshChildOfNearestScope); mixedModeCompatibility(closestScope, freshChildOfNearestScope, stale, NotNull{&dfg}, root); // closest Scope -> children = { ...., freshChildOfNearestScope} // We need to trim nearestChild from the scope hierarcy closestScope->children.emplace_back(freshChildOfNearestScope.get()); cg.visitFragmentRoot(freshChildOfNearestScope, root); // Trim nearestChild from the closestScope Scope* back = closestScope->children.back().get(); LUAU_ASSERT(back == freshChildOfNearestScope.get()); closestScope->children.pop_back(); } /// Initialize the constraint solver and run it ConstraintSolver cs{ NotNull{&normalizer}, NotNull{simplifier.get()}, NotNull{&typeFunctionRuntime}, NotNull(cg.rootScope), borrowConstraints(cg.constraints), incrementalModule->name, NotNull{&resolver}, {}, nullptr, NotNull{&dfg}, limits }; try { cs.run(); } catch (const TimeLimitError&) { stale->timeout = true; } catch (const UserCancelError&) { stale->cancelled = true; } // In frontend we would forbid internal types // because this is just for autocomplete, we don't actually care // We also don't even need to typecheck - just synthesize types as best as we can freeze(incrementalModule->internalTypes); freeze(incrementalModule->interfaceTypes); return {std::move(incrementalModule), std::move(freshChildOfNearestScope)}; } std::pair typecheckFragment( Frontend& frontend, const ModuleName& moduleName, const Position& cursorPos, std::optional opts, std::string_view src, std::optional fragmentEndPosition ) { if (FFlag::LuauBetterReverseDependencyTracking) { if (!frontend.allModuleDependenciesValid(moduleName, opts && opts->forAutocomplete)) return {FragmentTypeCheckStatus::SkipAutocomplete, {}}; } const SourceModule* sourceModule = frontend.getSourceModule(moduleName); if (!sourceModule) { LUAU_ASSERT(!"Expected Source Module for fragment typecheck"); return {}; } FrontendModuleResolver& resolver = getModuleResolver(frontend, opts); ModulePtr module = resolver.getModule(moduleName); if (!module) { LUAU_ASSERT(!"Expected Module for fragment typecheck"); return {}; } if (FFlag::LuauIncrementalAutocompleteBugfixes && FFlag::LuauReferenceAllocatorInNewSolver) { if (sourceModule->allocator.get() != module->allocator.get()) { return {FragmentTypeCheckStatus::SkipAutocomplete, {}}; } } auto tryParse = parseFragment(*sourceModule, src, cursorPos, fragmentEndPosition); if (!tryParse) return {FragmentTypeCheckStatus::SkipAutocomplete, {}}; FragmentParseResult& parseResult = *tryParse; if (isWithinComment(parseResult.commentLocations, fragmentEndPosition.value_or(cursorPos))) return {FragmentTypeCheckStatus::SkipAutocomplete, {}}; FrontendOptions frontendOptions = opts.value_or(frontend.options); const ScopePtr& closestScope = findClosestScope(module, parseResult.nearestStatement); FragmentTypeCheckResult result = typecheckFragment_(frontend, parseResult.root, module, closestScope, cursorPos, std::move(parseResult.alloc), frontendOptions); result.ancestry = std::move(parseResult.ancestry); return {FragmentTypeCheckStatus::Success, result}; } FragmentAutocompleteResult fragmentAutocomplete( Frontend& frontend, std::string_view src, const ModuleName& moduleName, Position cursorPosition, std::optional opts, StringCompletionCallback callback, std::optional fragmentEndPosition ) { LUAU_ASSERT(FFlag::LuauAllowFragmentParsing); LUAU_ASSERT(FFlag::LuauAutocompleteRefactorsForIncrementalAutocomplete); const SourceModule* sourceModule = frontend.getSourceModule(moduleName); if (!sourceModule) { LUAU_ASSERT(!"Expected Source Module for fragment typecheck"); return {}; } // If the cursor is within a comment in the stale source module we should avoid providing a recommendation if (isWithinComment(*sourceModule, fragmentEndPosition.value_or(cursorPosition))) return {}; auto [tcStatus, tcResult] = typecheckFragment(frontend, moduleName, cursorPosition, opts, src, fragmentEndPosition); if (tcStatus == FragmentTypeCheckStatus::SkipAutocomplete) return {}; auto globalScope = (opts && opts->forAutocomplete) ? frontend.globalsForAutocomplete.globalScope.get() : frontend.globals.globalScope.get(); TypeArena arenaForFragmentAutocomplete; auto result = Luau::autocomplete_( tcResult.incrementalModule, frontend.builtinTypes, &arenaForFragmentAutocomplete, tcResult.ancestry, globalScope, tcResult.freshScope, cursorPosition, frontend.fileResolver, callback ); return {std::move(tcResult.incrementalModule), tcResult.freshScope.get(), std::move(arenaForFragmentAutocomplete), std::move(result)}; } } // namespace Luau