mirror of
https://github.com/luau-lang/luau.git
synced 2025-01-07 11:59:11 +00:00
d518d14b92
### What's new * Fixed many of the false positive errors in indexing of table unions and table intersections * It is now possible to run custom checks over Luau AST during typechecking by setting `customModuleCheck` in `FrontendOptions` * Fixed codegen issue on arm, where number->vector cast could corrupt that number value for the next time it's read ### New Solver * `error` type now behaves as the bottom type during subtyping checks * Fixed the scope that is used in subtyping with generic types * Fixed `astOriginalCallTypes` table often used by LSP to match the old solver --- ### Internal Contributors Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
310 lines
11 KiB
C++
310 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/TypeCheckLimits.h"
|
|
#include "Luau/Variant.h"
|
|
#include "Luau/AnyTypeSummary.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 FrontendCancellationToken;
|
|
struct AnyTypeSummary;
|
|
|
|
struct LoadDefinitionFileResult
|
|
{
|
|
bool success;
|
|
ParseResult parseResult;
|
|
SourceModule sourceModule;
|
|
ModulePtr module;
|
|
};
|
|
|
|
std::optional<Mode> parseMode(const std::vector<HotComment>& hotcomments);
|
|
|
|
std::vector<std::string_view> parsePathExpr(const AstExpr& pathExpr);
|
|
|
|
// Exported only for convenient testing.
|
|
std::optional<ModuleName> pathExprToModuleName(const ModuleName& currentModuleName, const std::vector<std::string_view>& expr);
|
|
|
|
/** Try to convert an AST fragment into a ModuleName.
|
|
* Returns std::nullopt if the expression cannot be resolved. This will most likely happen in cases where
|
|
* the import path involves some dynamic computation that we cannot see into at typechecking time.
|
|
*
|
|
* Unintuitively, weirdly-formulated modules (like game.Parent.Parent.Parent.Foo) will successfully produce a ModuleName
|
|
* as long as it falls within the permitted syntax. This is ok because we will fail to find the module and produce an
|
|
* error when we try during typechecking.
|
|
*/
|
|
std::optional<ModuleName> pathExprToModuleName(const ModuleName& currentModuleName, const AstExpr& expr);
|
|
|
|
struct SourceNode
|
|
{
|
|
bool hasDirtySourceModule() const
|
|
{
|
|
return dirtySourceModule;
|
|
}
|
|
|
|
bool hasDirtyModule(bool forAutocomplete) const
|
|
{
|
|
return forAutocomplete ? dirtyModuleForAutocomplete : dirtyModule;
|
|
}
|
|
|
|
ModuleName name;
|
|
std::string humanReadableName;
|
|
DenseHashSet<ModuleName> requireSet{{}};
|
|
std::vector<std::pair<ModuleName, Location>> requireLocations;
|
|
bool dirtySourceModule = true;
|
|
bool dirtyModule = true;
|
|
bool dirtyModuleForAutocomplete = 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;
|
|
|
|
void 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 isDirty(const ModuleName& name, bool forAutocomplete = false) const;
|
|
void markDirty(const ModuleName& name, std::vector<ModuleName>* markedDirty = nullptr);
|
|
|
|
/** 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);
|
|
|
|
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);
|
|
|
|
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,
|
|
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,
|
|
std::function<void(const ModuleName&, const ScopePtr&)> prepareModuleScope,
|
|
FrontendOptions options,
|
|
TypeCheckLimits limits,
|
|
bool recordJsonLog,
|
|
std::function<void(const ModuleName&, std::string)> writeJsonLog
|
|
);
|
|
|
|
} // namespace Luau
|