mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-05 11:20:54 +01:00
## New Type Solver 1. Update resolved types for singleton unions and intersections to avoid crashing when type checking type assertions. 2. Generalize free return type pack of a function type inferred at call site to ensure that the free type does not leak to another module. 3. Fix crash from cyclic indexers by reducing if possible or producing an error otherwise. 4. Fix handling of irreducible type functions to prevent type inference from failing. 5. Fix handling of recursive metatables to avoid infinite recursion. ## New and Old Type Solver Fix accidental capture of all exceptions in multi-threaded typechecking by converting all typechecking exceptions to `InternalCompilerError` and only capturing those. ## Fragment Autocomplete 1. Add a block based diff algorithm based on class index and span for re-typechecking. This reduces the granularity of fragment autocomplete to avoid flakiness when the fragment does not have enough type information. 2. Fix bugs arising from incorrect scope selection for autocompletion. ## Roundtrippable AST Store type alias location in `TypeFun` class to ensure it is accessible for exported types as part of the public interface. ## Build System 1. Bump minimum supported CMake version to 3.10 since GitHub is phasing out the currently supported minimum version 3.0, released 11 years ago. 2. Fix compilation when `HARDSTACKTESTS` is enabled. ## Miscellaneous Flag removals and cleanup of unused code. ## Internal Contributors Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Talha Pathan <tpathan@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> ## External Contributors Thanks to [@grh-official](https://github.com/grh-official) for PR #1759 **Full Changelog**: https://github.com/luau-lang/luau/compare/0.667...0.668 --------- Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Varun Saini <61795485+vrn-sn@users.noreply.github.com> Co-authored-by: Alexander Youngblood <ayoungblood@roblox.com> Co-authored-by: Menarul Alam <malam@roblox.com> Co-authored-by: Vighnesh <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com>
196 lines
5.7 KiB
C++
196 lines
5.7 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
#pragma once
|
|
|
|
#include "Luau/Ast.h"
|
|
#include "Luau/Parser.h"
|
|
#include "Luau/AutocompleteTypes.h"
|
|
#include "Luau/DenseHash.h"
|
|
#include "Luau/Module.h"
|
|
#include "Luau/Frontend.h"
|
|
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
namespace Luau
|
|
{
|
|
struct FrontendOptions;
|
|
|
|
enum class FragmentAutocompleteWaypoint
|
|
{
|
|
ParseFragmentEnd,
|
|
CloneModuleStart,
|
|
CloneModuleEnd,
|
|
DfgBuildEnd,
|
|
CloneAndSquashScopeStart,
|
|
CloneAndSquashScopeEnd,
|
|
ConstraintSolverStart,
|
|
ConstraintSolverEnd,
|
|
TypecheckFragmentEnd,
|
|
AutocompleteEnd,
|
|
COUNT,
|
|
};
|
|
|
|
class IFragmentAutocompleteReporter
|
|
{
|
|
public:
|
|
virtual void reportWaypoint(FragmentAutocompleteWaypoint) = 0;
|
|
virtual void reportFragmentString(std::string_view) = 0;
|
|
};
|
|
|
|
enum class FragmentTypeCheckStatus
|
|
{
|
|
SkipAutocomplete,
|
|
Success,
|
|
};
|
|
|
|
struct FragmentAutocompleteAncestryResult
|
|
{
|
|
DenseHashMap<AstName, AstLocal*> localMap{AstName()};
|
|
std::vector<AstLocal*> localStack;
|
|
std::vector<AstNode*> ancestry;
|
|
AstStat* nearestStatement = nullptr;
|
|
AstStatBlock* parentBlock = nullptr;
|
|
Location fragmentSelectionRegion;
|
|
};
|
|
|
|
struct FragmentParseResult
|
|
{
|
|
std::string fragmentToParse;
|
|
AstStatBlock* root = nullptr;
|
|
std::vector<AstNode*> ancestry;
|
|
AstStat* nearestStatement = nullptr;
|
|
std::vector<Comment> commentLocations;
|
|
std::unique_ptr<Allocator> alloc = std::make_unique<Allocator>();
|
|
Position scopePos{0, 0};
|
|
};
|
|
|
|
struct FragmentTypeCheckResult
|
|
{
|
|
ModulePtr incrementalModule = nullptr;
|
|
ScopePtr freshScope;
|
|
std::vector<AstNode*> ancestry;
|
|
};
|
|
|
|
struct FragmentAutocompleteResult
|
|
{
|
|
ModulePtr incrementalModule;
|
|
Scope* freshScope;
|
|
TypeArena arenaForAutocomplete;
|
|
AutocompleteResult acResults;
|
|
};
|
|
|
|
struct FragmentRegion
|
|
{
|
|
Location fragmentLocation;
|
|
AstStat* nearestStatement = nullptr; // used for tests
|
|
AstStatBlock* parentBlock = nullptr; // used for scope detection
|
|
};
|
|
|
|
std::optional<Position> blockDiffStart(AstStatBlock* blockOld, AstStatBlock* blockNew, AstStat* nearestStatementNewAst);
|
|
FragmentRegion getFragmentRegion(AstStatBlock* root, const Position& cursorPosition);
|
|
FragmentAutocompleteAncestryResult findAncestryForFragmentParse(AstStatBlock* stale, const Position& cursorPos, AstStatBlock* lastGoodParse);
|
|
FragmentAutocompleteAncestryResult findAncestryForFragmentParse_DEPRECATED(AstStatBlock* root, const Position& cursorPos);
|
|
|
|
std::optional<FragmentParseResult> parseFragment_DEPRECATED(
|
|
AstStatBlock* root,
|
|
AstNameTable* names,
|
|
std::string_view src,
|
|
const Position& cursorPos,
|
|
std::optional<Position> fragmentEndPosition
|
|
);
|
|
|
|
std::optional<FragmentParseResult> parseFragment(
|
|
AstStatBlock* stale,
|
|
AstStatBlock* mostRecentParse,
|
|
AstNameTable* names,
|
|
std::string_view src,
|
|
const Position& cursorPos,
|
|
std::optional<Position> fragmentEndPosition
|
|
);
|
|
|
|
std::pair<FragmentTypeCheckStatus, FragmentTypeCheckResult> typecheckFragment(
|
|
Frontend& frontend,
|
|
const ModuleName& moduleName,
|
|
const Position& cursorPos,
|
|
std::optional<FrontendOptions> opts,
|
|
std::string_view src,
|
|
std::optional<Position> fragmentEndPosition,
|
|
AstStatBlock* recentParse = nullptr,
|
|
IFragmentAutocompleteReporter* reporter = nullptr
|
|
);
|
|
|
|
FragmentAutocompleteResult fragmentAutocomplete(
|
|
Frontend& frontend,
|
|
std::string_view src,
|
|
const ModuleName& moduleName,
|
|
Position cursorPosition,
|
|
std::optional<FrontendOptions> opts,
|
|
StringCompletionCallback callback,
|
|
std::optional<Position> fragmentEndPosition = std::nullopt,
|
|
AstStatBlock* recentParse = nullptr,
|
|
IFragmentAutocompleteReporter* reporter = nullptr
|
|
);
|
|
|
|
enum class FragmentAutocompleteStatus
|
|
{
|
|
Success,
|
|
FragmentTypeCheckFail,
|
|
InternalIce
|
|
};
|
|
|
|
struct FragmentAutocompleteStatusResult
|
|
{
|
|
FragmentAutocompleteStatus status;
|
|
std::optional<FragmentAutocompleteResult> result;
|
|
};
|
|
|
|
struct FragmentContext
|
|
{
|
|
std::string_view newSrc;
|
|
const ParseResult& freshParse;
|
|
std::optional<FrontendOptions> opts;
|
|
std::optional<Position> DEPRECATED_fragmentEndPosition;
|
|
IFragmentAutocompleteReporter* reporter = nullptr;
|
|
};
|
|
|
|
/**
|
|
* @brief Attempts to compute autocomplete suggestions from the fragment context.
|
|
*
|
|
* This function computes autocomplete suggestions using outdated frontend typechecking data
|
|
* by patching the fragment context of the new script source content.
|
|
*
|
|
* @param frontend The Luau Frontend data structure, which may contain outdated typechecking data.
|
|
*
|
|
* @param moduleName The name of the target module, specifying which script the caller wants to request autocomplete for.
|
|
*
|
|
* @param cursorPosition The position in the script where the caller wants to trigger autocomplete.
|
|
*
|
|
* @param context The fragment context that this API will use to patch the outdated typechecking data.
|
|
*
|
|
* @param stringCompletionCB A callback function that provides autocomplete suggestions for string contexts.
|
|
*
|
|
* @return
|
|
* The status indicating whether `fragmentAutocomplete` ran successfully or failed, along with the reason for failure.
|
|
* Also includes autocomplete suggestions if the status is successful.
|
|
*
|
|
* @usage
|
|
* FragmentAutocompleteStatusResult acStatusResult;
|
|
* if (shouldFragmentAC)
|
|
* acStatusResult = Luau::tryFragmentAutocomplete(...);
|
|
*
|
|
* if (acStatusResult.status != Successful)
|
|
* {
|
|
* frontend.check(moduleName, options);
|
|
* acStatusResult.acResult = Luau::autocomplete(...);
|
|
* }
|
|
* return convertResultWithContext(acStatusResult.acResult);
|
|
*/
|
|
FragmentAutocompleteStatusResult tryFragmentAutocomplete(
|
|
Frontend& frontend,
|
|
const ModuleName& moduleName,
|
|
Position cursorPosition,
|
|
FragmentContext context,
|
|
StringCompletionCallback stringCompletionCB
|
|
);
|
|
|
|
} // namespace Luau
|