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>
324 lines
11 KiB
C++
324 lines
11 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/Config.h"
|
|
#include "Luau/GlobalTypes.h"
|
|
#include "Luau/Module.h"
|
|
#include "Luau/ModuleResolver.h"
|
|
#include "Luau/RequireTracer.h"
|
|
#include "Luau/Scope.h"
|
|
#include "Luau/Set.h"
|
|
#include "Luau/TypeCheckLimits.h"
|
|
#include "Luau/Variant.h"
|
|
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <vector>
|
|
#include <optional>
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
class AstStat;
|
|
class ParseError;
|
|
struct Frontend;
|
|
struct TypeError;
|
|
struct LintWarning;
|
|
struct GlobalTypes;
|
|
struct TypeChecker;
|
|
struct FileResolver;
|
|
struct ModuleResolver;
|
|
struct ParseResult;
|
|
struct HotComment;
|
|
struct BuildQueueItem;
|
|
struct BuildQueueWorkState;
|
|
struct FrontendCancellationToken;
|
|
|
|
struct LoadDefinitionFileResult
|
|
{
|
|
bool success;
|
|
ParseResult parseResult;
|
|
SourceModule sourceModule;
|
|
ModulePtr module;
|
|
};
|
|
|
|
std::optional<Mode> parseMode(const std::vector<HotComment>& hotcomments);
|
|
|
|
struct SourceNode
|
|
{
|
|
bool hasDirtySourceModule() const
|
|
{
|
|
return dirtySourceModule;
|
|
}
|
|
|
|
bool hasDirtyModule(bool forAutocomplete) const
|
|
{
|
|
return forAutocomplete ? dirtyModuleForAutocomplete : dirtyModule;
|
|
}
|
|
|
|
bool hasInvalidModuleDependency(bool forAutocomplete) const
|
|
{
|
|
return forAutocomplete ? invalidModuleDependencyForAutocomplete : invalidModuleDependency;
|
|
}
|
|
|
|
void setInvalidModuleDependency(bool value, bool forAutocomplete)
|
|
{
|
|
if (forAutocomplete)
|
|
invalidModuleDependencyForAutocomplete = value;
|
|
else
|
|
invalidModuleDependency = value;
|
|
}
|
|
|
|
ModuleName name;
|
|
std::string humanReadableName;
|
|
DenseHashSet<ModuleName> requireSet{{}};
|
|
std::vector<std::pair<ModuleName, Location>> requireLocations;
|
|
Set<ModuleName> dependents{{}};
|
|
|
|
bool dirtySourceModule = true;
|
|
bool dirtyModule = true;
|
|
bool dirtyModuleForAutocomplete = true;
|
|
|
|
bool invalidModuleDependency = true;
|
|
bool invalidModuleDependencyForAutocomplete = true;
|
|
|
|
double autocompleteLimitsMult = 1.0;
|
|
};
|
|
|
|
struct FrontendOptions
|
|
{
|
|
// When true, we retain full type information about every term in the AST.
|
|
// Setting this to false cuts back on RAM and is a good idea for batch
|
|
// jobs where the type graph is not deeply inspected after typechecking
|
|
// is complete.
|
|
bool retainFullTypeGraphs = false;
|
|
|
|
// Run typechecking only in mode required for autocomplete (strict mode in
|
|
// order to get more precise type information)
|
|
bool forAutocomplete = false;
|
|
|
|
bool runLintChecks = false;
|
|
|
|
// If not empty, randomly shuffle the constraint set before attempting to
|
|
// solve. Use this value to seed the random number generator.
|
|
std::optional<unsigned> randomizeConstraintResolutionSeed;
|
|
|
|
std::optional<LintOptions> enabledLintWarnings;
|
|
|
|
std::shared_ptr<FrontendCancellationToken> cancellationToken;
|
|
|
|
// Time limit for typechecking a single module
|
|
std::optional<double> moduleTimeLimitSec;
|
|
|
|
// When true, some internal complexity limits will be scaled down for modules that miss the limit set by moduleTimeLimitSec
|
|
bool applyInternalLimitScaling = false;
|
|
|
|
// An optional callback which is called for every *dirty* module was checked
|
|
// Is multi-threaded typechecking is used, this callback might be called from multiple threads and has to be thread-safe
|
|
std::function<void(const SourceModule& sourceModule, const Luau::Module& module)> customModuleCheck;
|
|
};
|
|
|
|
struct CheckResult
|
|
{
|
|
std::vector<TypeError> errors;
|
|
|
|
LintResult lintResult;
|
|
|
|
std::vector<ModuleName> timeoutHits;
|
|
};
|
|
|
|
struct FrontendModuleResolver : ModuleResolver
|
|
{
|
|
FrontendModuleResolver(Frontend* frontend);
|
|
|
|
const ModulePtr getModule(const ModuleName& moduleName) const override;
|
|
bool moduleExists(const ModuleName& moduleName) const override;
|
|
std::optional<ModuleInfo> resolveModuleInfo(const ModuleName& currentModuleName, const AstExpr& pathExpr) override;
|
|
std::string getHumanReadableModuleName(const ModuleName& moduleName) const override;
|
|
|
|
bool setModule(const ModuleName& moduleName, ModulePtr module);
|
|
void clearModules();
|
|
|
|
private:
|
|
Frontend* frontend;
|
|
|
|
mutable std::mutex moduleMutex;
|
|
std::unordered_map<ModuleName, ModulePtr> modules;
|
|
};
|
|
|
|
struct Frontend
|
|
{
|
|
struct Stats
|
|
{
|
|
size_t files = 0;
|
|
size_t lines = 0;
|
|
|
|
size_t filesStrict = 0;
|
|
size_t filesNonstrict = 0;
|
|
|
|
double timeRead = 0;
|
|
double timeParse = 0;
|
|
double timeCheck = 0;
|
|
double timeLint = 0;
|
|
};
|
|
|
|
Frontend(FileResolver* fileResolver, ConfigResolver* configResolver, const FrontendOptions& options = {});
|
|
|
|
// Parse module graph and prepare SourceNode/SourceModule data, including required dependencies without running typechecking
|
|
void parse(const ModuleName& name);
|
|
|
|
// Parse and typecheck module graph
|
|
CheckResult check(const ModuleName& name, std::optional<FrontendOptions> optionOverride = {}); // new shininess
|
|
|
|
bool allModuleDependenciesValid(const ModuleName& name, bool forAutocomplete = false) const;
|
|
|
|
bool isDirty(const ModuleName& name, bool forAutocomplete = false) const;
|
|
void markDirty(const ModuleName& name, std::vector<ModuleName>* markedDirty = nullptr);
|
|
|
|
void traverseDependents(const ModuleName& name, std::function<bool(SourceNode&)> processSubtree);
|
|
|
|
/** Borrow a pointer into the SourceModule cache.
|
|
*
|
|
* Returns nullptr if we don't have it. This could mean that the script
|
|
* doesn't exist, or simply that its contents have changed since the previous
|
|
* check, in which case we do not have its AST.
|
|
*
|
|
* IMPORTANT: this pointer is only valid until the next call to markDirty. Do not retain it.
|
|
*/
|
|
SourceModule* getSourceModule(const ModuleName& name);
|
|
const SourceModule* getSourceModule(const ModuleName& name) const;
|
|
|
|
void clearStats();
|
|
void clear();
|
|
|
|
ScopePtr addEnvironment(const std::string& environmentName);
|
|
ScopePtr getEnvironmentScope(const std::string& environmentName) const;
|
|
|
|
void registerBuiltinDefinition(const std::string& name, std::function<void(Frontend&, GlobalTypes&, ScopePtr)>);
|
|
void applyBuiltinDefinitionToEnvironment(const std::string& environmentName, const std::string& definitionName);
|
|
|
|
LoadDefinitionFileResult loadDefinitionFile(
|
|
GlobalTypes& globals,
|
|
ScopePtr targetScope,
|
|
std::string_view source,
|
|
const std::string& packageName,
|
|
bool captureComments,
|
|
bool typeCheckForAutocomplete = false
|
|
);
|
|
|
|
// Batch module checking. Queue modules and check them together, retrieve results with 'getCheckResult'
|
|
// If provided, 'executeTask' function is allowed to call the 'task' function on any thread and return without waiting for 'task' to complete
|
|
void queueModuleCheck(const std::vector<ModuleName>& names);
|
|
void queueModuleCheck(const ModuleName& name);
|
|
std::vector<ModuleName> checkQueuedModules(
|
|
std::optional<FrontendOptions> optionOverride = {},
|
|
std::function<void(std::function<void()> task)> executeTask = {},
|
|
std::function<bool(size_t done, size_t total)> progress = {}
|
|
);
|
|
|
|
std::optional<CheckResult> getCheckResult(const ModuleName& name, bool accumulateNested, bool forAutocomplete = false);
|
|
std::vector<ModuleName> getRequiredScripts(const ModuleName& name);
|
|
|
|
private:
|
|
ModulePtr check(
|
|
const SourceModule& sourceModule,
|
|
Mode mode,
|
|
std::vector<RequireCycle> requireCycles,
|
|
std::optional<ScopePtr> environmentScope,
|
|
bool forAutocomplete,
|
|
bool recordJsonLog,
|
|
TypeCheckLimits typeCheckLimits
|
|
);
|
|
|
|
std::pair<SourceNode*, SourceModule*> getSourceNode(const ModuleName& name);
|
|
SourceModule parse(const ModuleName& name, std::string_view src, const ParseOptions& parseOptions);
|
|
|
|
bool parseGraph(
|
|
std::vector<ModuleName>& buildQueue,
|
|
const ModuleName& root,
|
|
bool forAutocomplete,
|
|
std::function<bool(const ModuleName&)> canSkip = {}
|
|
);
|
|
|
|
void addBuildQueueItems(
|
|
std::vector<BuildQueueItem>& items,
|
|
std::vector<ModuleName>& buildQueue,
|
|
bool cycleDetected,
|
|
DenseHashSet<Luau::ModuleName>& seen,
|
|
const FrontendOptions& frontendOptions
|
|
);
|
|
void checkBuildQueueItem(BuildQueueItem& item);
|
|
void checkBuildQueueItems(std::vector<BuildQueueItem>& items);
|
|
void recordItemResult(const BuildQueueItem& item);
|
|
void performQueueItemTask(std::shared_ptr<BuildQueueWorkState> state, size_t itemPos);
|
|
void sendQueueItemTask(std::shared_ptr<BuildQueueWorkState> state, size_t itemPos);
|
|
void sendQueueCycleItemTask(std::shared_ptr<BuildQueueWorkState> state);
|
|
|
|
static LintResult classifyLints(const std::vector<LintWarning>& warnings, const Config& config);
|
|
|
|
ScopePtr getModuleEnvironment(const SourceModule& module, const Config& config, bool forAutocomplete) const;
|
|
|
|
std::unordered_map<std::string, ScopePtr> environments;
|
|
std::unordered_map<std::string, std::function<void(Frontend&, GlobalTypes&, ScopePtr)>> builtinDefinitions;
|
|
|
|
BuiltinTypes builtinTypes_;
|
|
|
|
public:
|
|
const NotNull<BuiltinTypes> builtinTypes;
|
|
|
|
FileResolver* fileResolver;
|
|
|
|
FrontendModuleResolver moduleResolver;
|
|
FrontendModuleResolver moduleResolverForAutocomplete;
|
|
|
|
GlobalTypes globals;
|
|
GlobalTypes globalsForAutocomplete;
|
|
|
|
ConfigResolver* configResolver;
|
|
FrontendOptions options;
|
|
InternalErrorReporter iceHandler;
|
|
std::function<void(const ModuleName& name, const ScopePtr& scope, bool forAutocomplete)> prepareModuleScope;
|
|
std::function<void(const ModuleName& name, std::string log)> writeJsonLog = {};
|
|
|
|
std::unordered_map<ModuleName, std::shared_ptr<SourceNode>> sourceNodes;
|
|
std::unordered_map<ModuleName, std::shared_ptr<SourceModule>> sourceModules;
|
|
std::unordered_map<ModuleName, RequireTraceResult> requireTrace;
|
|
|
|
Stats stats = {};
|
|
|
|
std::vector<ModuleName> moduleQueue;
|
|
};
|
|
|
|
ModulePtr check(
|
|
const SourceModule& sourceModule,
|
|
Mode mode,
|
|
const std::vector<RequireCycle>& requireCycles,
|
|
NotNull<BuiltinTypes> builtinTypes,
|
|
NotNull<InternalErrorReporter> iceHandler,
|
|
NotNull<ModuleResolver> moduleResolver,
|
|
NotNull<FileResolver> fileResolver,
|
|
const ScopePtr& globalScope,
|
|
const ScopePtr& typeFunctionScope,
|
|
std::function<void(const ModuleName&, const ScopePtr&)> prepareModuleScope,
|
|
FrontendOptions options,
|
|
TypeCheckLimits limits
|
|
);
|
|
|
|
ModulePtr check(
|
|
const SourceModule& sourceModule,
|
|
Mode mode,
|
|
const std::vector<RequireCycle>& requireCycles,
|
|
NotNull<BuiltinTypes> builtinTypes,
|
|
NotNull<InternalErrorReporter> iceHandler,
|
|
NotNull<ModuleResolver> moduleResolver,
|
|
NotNull<FileResolver> fileResolver,
|
|
const ScopePtr& globalScope,
|
|
const ScopePtr& typeFunctionScope,
|
|
std::function<void(const ModuleName&, const ScopePtr&)> prepareModuleScope,
|
|
FrontendOptions options,
|
|
TypeCheckLimits limits,
|
|
bool recordJsonLog,
|
|
std::function<void(const ModuleName&, std::string)> writeJsonLog
|
|
);
|
|
|
|
} // namespace Luau
|