mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 02:10:53 +01:00
After a very auspicious release last week, we have a new bevy of changes for you! ## What's Changed ### Deprecated Attribute This release includes an implementation of the `@deprecated` attribute proposed in [this RFC](https://rfcs.luau.org/syntax-attribute-functions-deprecated.html). It relies on the new type solver to propagate deprecation information from function and method AST nodes to the corresponding type objects. These objects are queried by a linter pass when it encounters local, global, or indexed variables, to issue deprecation warnings. Uses of deprecated functions and methods in recursion are ignored. To support deprecation of class methods, the parser has been extended to allow attribute declarations on class methods. The implementation does not support parameters, so it is not currently possible for users to customize deprecation messages. ### General - Add a limit for normalization of function types. ### New Type Solver - Fix type checker to accept numbers as concat operands (Fixes #1671). - Fix user-defined type functions failing when used inside type aliases/nested calls (Fixes #1738, Fixes #1679). - Improve constraint generation for overloaded functions (in part thanks to @vvatheus in #1694). - Improve type inference for indexers on table literals, especially when passing table literals directly as a function call argument. - Equate regular error type and intersection with a negation of an error type. - Avoid swapping types in 2-part union when RHS is optional. - Use simplification when doing `~nil` refinements. - `len<>` now works on metatables without `__len` function. ### AST - Retain source information for `AstTypeUnion` and `AstTypeIntersection`. ### Transpiler - Print attributes on functions. ### Parser - Allow types in indexers to begin with string literals by @jackdotink in #1750. ### Autocomplete - Evaluate user-defined type functions in ill-formed source code to provide autocomplete. - Fix the start location of functions that have attributes. - Implement better fragment selection. ### Internal Contributors Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com> Co-authored-by: Aviral Goel <agoel@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Sora Kanosue <skanosue@roblox.com> Co-authored-by: Talha Pathan <tpathan@roblox.com> Co-authored-by: Varun Saini <vsaini@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> **Full Changelog**: https://github.com/luau-lang/luau/compare/0.666...0.667 --------- Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Varun Saini <61795485+vrn-sn@users.noreply.github.com> Co-authored-by: Menarul Alam <malam@roblox.com> Co-authored-by: Aviral Goel <agoel@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>
195 lines
5.6 KiB
C++
195 lines
5.6 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
|
|
};
|
|
|
|
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
|