mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 18:30:54 +01:00
Hello all! Another week, another Luau release! # Change to `lua_setuserdatametatable` This release fixes #1710: `lua_setuserdatametatable` is being changed so that it _only_ operates on the top of the stack: the `idx` parameter is being removed. Prior to this, `lua_setuserdatametable` would set the metatable of the value in the stack at `idx`, but _always_ pop the top of the stack. The old behavior is available in this release as `lua_setuserdatametatable_DEPRECATED`. # General This release exposes a generalized implementation of require-by-string's autocomplete logic. `FileResolver` can now be optionally constructed with a `RequireSuggester`, which provides an interface for converting a given module to a `RequireNode`. Consumers of this new API implement a `RequireNode` to define how modules are represented in their embedded context, and the new API manages the logic specific to require-by-string, including providing suggestions for require aliases. This enhancement moves toward integrating require-by-string's semantics into the language itself, rather than merely providing a specification for community members to implement themselves. # New Type Solver * Fixed a source of potential `Luau::follow detected a Type cycle` internal compiler exceptions when assigning a global to itself. * Fixed an issue whereby `*no-refine*` (a type which should not be visible at the end of type checking) was not being properly elided, causing inference of class-like tables to become unreadable / induce crashes in autocomplete. * Fixed a case of incomplete constraint solving when performing basic math in a loop # Fragment Autocomplete * Fixed several crashes related to not properly filling in scope information for the fragments * Fixed a source of memory corruption by isolating the return type of a fragment when it is type checked. * Improved performance by opting not to clone persistent types for the fragment (e.g.: built in types) # Internal Contributors Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@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> --------- 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: 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>
172 lines
5.6 KiB
C++
172 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
|
|
#include "Luau/FileResolver.h"
|
|
|
|
#include "Luau/Common.h"
|
|
#include "Luau/StringUtils.h"
|
|
|
|
#include <algorithm>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <string_view>
|
|
#include <utility>
|
|
|
|
LUAU_FASTFLAGVARIABLE(LuauExposeRequireByStringAutocomplete)
|
|
LUAU_FASTFLAGVARIABLE(LuauEscapeCharactersInRequireSuggestions)
|
|
LUAU_FASTFLAGVARIABLE(LuauHideImpossibleRequireSuggestions)
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
static std::optional<RequireSuggestions> processRequireSuggestions(std::optional<RequireSuggestions> suggestions)
|
|
{
|
|
if (!suggestions)
|
|
return suggestions;
|
|
|
|
if (FFlag::LuauEscapeCharactersInRequireSuggestions)
|
|
{
|
|
for (RequireSuggestion& suggestion : *suggestions)
|
|
{
|
|
suggestion.fullPath = escape(suggestion.fullPath);
|
|
}
|
|
}
|
|
|
|
return suggestions;
|
|
}
|
|
|
|
static RequireSuggestions makeSuggestionsFromAliases(std::vector<RequireAlias> aliases)
|
|
{
|
|
RequireSuggestions result;
|
|
for (RequireAlias& alias : aliases)
|
|
{
|
|
RequireSuggestion suggestion;
|
|
suggestion.label = "@" + std::move(alias.alias);
|
|
suggestion.fullPath = suggestion.label;
|
|
suggestion.tags = std::move(alias.tags);
|
|
result.push_back(std::move(suggestion));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static RequireSuggestions makeSuggestionsForFirstComponent(std::unique_ptr<RequireNode> node)
|
|
{
|
|
RequireSuggestions result = makeSuggestionsFromAliases(node->getAvailableAliases());
|
|
result.push_back(RequireSuggestion{"./", "./", {}});
|
|
result.push_back(RequireSuggestion{"../", "../", {}});
|
|
return result;
|
|
}
|
|
|
|
static RequireSuggestions makeSuggestionsFromNode(std::unique_ptr<RequireNode> node, const std::string_view path, bool isPartialPath)
|
|
{
|
|
LUAU_ASSERT(!path.empty());
|
|
|
|
RequireSuggestions result;
|
|
|
|
const size_t lastSlashInPath = path.find_last_of('/');
|
|
|
|
if (lastSlashInPath != std::string_view::npos)
|
|
{
|
|
// Add a suggestion for the parent directory
|
|
RequireSuggestion parentSuggestion;
|
|
parentSuggestion.label = "..";
|
|
|
|
// TODO: after exposing require-by-string's path normalization API, this
|
|
// if-else can be replaced. Instead, we can simply normalize the result
|
|
// of inserting ".." at the end of the current path.
|
|
if (lastSlashInPath >= 2 && path.substr(lastSlashInPath - 2, 3) == "../")
|
|
{
|
|
parentSuggestion.fullPath = path.substr(0, lastSlashInPath + 1);
|
|
parentSuggestion.fullPath += "..";
|
|
}
|
|
else
|
|
{
|
|
parentSuggestion.fullPath = path.substr(0, lastSlashInPath);
|
|
}
|
|
|
|
result.push_back(std::move(parentSuggestion));
|
|
}
|
|
|
|
std::string fullPathPrefix;
|
|
if (isPartialPath)
|
|
{
|
|
// ./path/to/chi -> ./path/to/
|
|
fullPathPrefix += path.substr(0, lastSlashInPath + 1);
|
|
}
|
|
else
|
|
{
|
|
if (path.back() == '/')
|
|
{
|
|
// ./path/to/ -> ./path/to/
|
|
fullPathPrefix += path;
|
|
}
|
|
else
|
|
{
|
|
// ./path/to -> ./path/to/
|
|
fullPathPrefix += path;
|
|
fullPathPrefix += "/";
|
|
}
|
|
}
|
|
|
|
for (const std::unique_ptr<RequireNode>& child : node->getChildren())
|
|
{
|
|
if (!child)
|
|
continue;
|
|
|
|
std::string pathComponent = child->getPathComponent();
|
|
if (FFlag::LuauHideImpossibleRequireSuggestions)
|
|
{
|
|
// If path component contains a slash, it cannot be required by string.
|
|
// There's no point suggesting it.
|
|
if (pathComponent.find('/') != std::string::npos)
|
|
continue;
|
|
}
|
|
|
|
RequireSuggestion suggestion;
|
|
suggestion.label = isPartialPath || path.back() == '/' ? child->getLabel() : "/" + child->getLabel();
|
|
suggestion.fullPath = fullPathPrefix + std::move(pathComponent);
|
|
suggestion.tags = child->getTags();
|
|
result.push_back(std::move(suggestion));
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
std::optional<RequireSuggestions> RequireSuggester::getRequireSuggestionsImpl(const ModuleName& requirer, const std::optional<std::string>& path)
|
|
const
|
|
{
|
|
if (!path)
|
|
return std::nullopt;
|
|
|
|
std::unique_ptr<RequireNode> requirerNode = getNode(requirer);
|
|
if (!requirerNode)
|
|
return std::nullopt;
|
|
|
|
const size_t slashPos = path->find_last_of('/');
|
|
|
|
if (slashPos == std::string::npos)
|
|
return makeSuggestionsForFirstComponent(std::move(requirerNode));
|
|
|
|
// If path already points at a Node, return the Node's children as paths.
|
|
if (std::unique_ptr<RequireNode> node = requirerNode->resolvePathToNode(*path))
|
|
return makeSuggestionsFromNode(std::move(node), *path, /* isPartialPath = */ false);
|
|
|
|
// Otherwise, recover a partial path and use this to generate suggestions.
|
|
if (std::unique_ptr<RequireNode> partialNode = requirerNode->resolvePathToNode(path->substr(0, slashPos)))
|
|
return makeSuggestionsFromNode(std::move(partialNode), *path, /* isPartialPath = */ true);
|
|
|
|
return std::nullopt;
|
|
}
|
|
|
|
std::optional<RequireSuggestions> RequireSuggester::getRequireSuggestions(const ModuleName& requirer, const std::optional<std::string>& path) const
|
|
{
|
|
return processRequireSuggestions(getRequireSuggestionsImpl(requirer, path));
|
|
}
|
|
|
|
std::optional<RequireSuggestions> FileResolver::getRequireSuggestions(const ModuleName& requirer, const std::optional<std::string>& path) const
|
|
{
|
|
if (!FFlag::LuauExposeRequireByStringAutocomplete)
|
|
return std::nullopt;
|
|
|
|
return requireSuggester ? requireSuggester->getRequireSuggestions(requirer, path) : std::nullopt;
|
|
}
|
|
|
|
} // namespace Luau
|