mirror of
https://github.com/luau-lang/luau.git
synced 2025-05-04 10:33:46 +01:00
We have lots of new changes for you! # What's Changed ## General - We updated Luau's license year to 2025! - We fixed a bug where large amounts of errors were being printed when deep intersections of unions error. ## Require-by-String This release introduces the `Luau.Require` library, which exposes the runtime semantics of require-by-string, including support for the new `@self` alias described in [this RFC](https://github.com/luau-lang/rfcs/pull/109). The library operates on a virtualized filesystem, allowing consumers to specify navigation rules without assuming a filesystem context. Documentation in `Require.h` explains how to enable the library, and the `setupState` function in Repl.cpp demonstrates how we've integrated it into the luau CLI tool. Note that the interface in `Require.h` is written in C, which enables any application written in a language with a C foreign-function interface to link against this library and enable require-by-string. This makes it straightforward for any application embedding Luau to support require-by-string, provided that it defines or operates within an environment resembling a virtual filesystem. The core navigation semantics of require-by-string have additionally been pulled into the `Luau.RequireNavigator` library. While `Luau.Require` internally depends on `Luau.RequireNavigator`, the latter does not depend on the Luau VM. This library provides an interface for inspecting require-by-string's navigation behavior and therefore serves as a useful dependency for static tooling. Documentation for `Luau.RequireNavigator` is available in `RequireNavigator.h`. ## Autocomplete - We fixed a memory leak in fragment autocomplete! ## New Solver And Old Solver - We've found a infinite iteration error over a type pack. We added a way to detect this error and throw an `InternalCompileError` instead. - We fix `table.freeze` not accounting for the first argument not getting type stated. We fall back to regular inference instead. - We fix a crash in the old solver with `length_error`. - We fix a crash in the new solver stemming from generalization reentrancy. Now we correctly generalize interior free types that do not appear in a function signature. - We fix a nil refinement. (Fixes https://github.com/luau-lang/luau/issues/1687 and https://github.com/luau-lang/luau/issues/1451) ### 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> Full Changelog: https://github.com/luau-lang/luau/compare/0.668...0.669 --------- 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: 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>
119 lines
3.4 KiB
C++
119 lines
3.4 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
#include "Luau/RequirerUtils.h"
|
|
|
|
#include "Luau/FileUtils.h"
|
|
|
|
#include <algorithm>
|
|
#include <string>
|
|
#include <string_view>
|
|
|
|
static std::pair<PathResult::Status, std::string> getSuffixWithAmbiguityCheck(const std::string& path)
|
|
{
|
|
bool found = false;
|
|
std::string suffix;
|
|
|
|
for (const char* potentialSuffix : {".luau", ".lua"})
|
|
{
|
|
if (isFile(path + potentialSuffix))
|
|
{
|
|
if (found)
|
|
return {PathResult::Status::AMBIGUOUS, ""};
|
|
|
|
suffix = potentialSuffix;
|
|
found = true;
|
|
}
|
|
}
|
|
if (isDirectory(path))
|
|
{
|
|
if (found)
|
|
return {PathResult::Status::AMBIGUOUS, ""};
|
|
|
|
for (const char* potentialSuffix : {"/init.luau", "/init.lua"})
|
|
{
|
|
if (isFile(path + potentialSuffix))
|
|
{
|
|
if (found)
|
|
return {PathResult::Status::AMBIGUOUS, ""};
|
|
|
|
suffix = potentialSuffix;
|
|
found = true;
|
|
}
|
|
}
|
|
|
|
found = true;
|
|
}
|
|
|
|
if (!found)
|
|
return {PathResult::Status::NOT_FOUND, ""};
|
|
|
|
return {PathResult::Status::SUCCESS, suffix};
|
|
}
|
|
|
|
static PathResult addSuffix(PathResult partialResult)
|
|
{
|
|
if (partialResult.status != PathResult::Status::SUCCESS)
|
|
return partialResult;
|
|
|
|
auto [status, suffix] = getSuffixWithAmbiguityCheck(partialResult.absPath);
|
|
if (status != PathResult::Status::SUCCESS)
|
|
return PathResult{status};
|
|
|
|
partialResult.suffix = std::move(suffix);
|
|
return partialResult;
|
|
}
|
|
|
|
PathResult getStdInResult()
|
|
{
|
|
std::optional<std::string> cwd = getCurrentWorkingDirectory();
|
|
if (!cwd)
|
|
return PathResult{PathResult::Status::NOT_FOUND};
|
|
|
|
std::replace(cwd->begin(), cwd->end(), '\\', '/');
|
|
|
|
return PathResult{PathResult::Status::SUCCESS, *cwd + "/stdin", "./stdin", ""};
|
|
}
|
|
|
|
PathResult getAbsolutePathResult(const std::string& path)
|
|
{
|
|
return addSuffix(PathResult{PathResult::Status::SUCCESS, path});
|
|
}
|
|
|
|
PathResult tryGetRelativePathResult(const std::string& path)
|
|
{
|
|
if (isAbsolutePath(path))
|
|
return getAbsolutePathResult(path);
|
|
|
|
std::optional<std::string> cwd = getCurrentWorkingDirectory();
|
|
if (!cwd)
|
|
return PathResult{PathResult::Status::NOT_FOUND};
|
|
|
|
std::optional<std::string> resolvedAbsPath = resolvePath(path, *cwd + "/stdin");
|
|
if (!resolvedAbsPath)
|
|
return PathResult{PathResult::Status::NOT_FOUND};
|
|
|
|
return addSuffix(PathResult{PathResult::Status::SUCCESS, std::move(*resolvedAbsPath), path});
|
|
}
|
|
|
|
PathResult getParent(const std::string& absPath, const std::string& relPath)
|
|
{
|
|
std::optional<std::string> parent = getParentPath(absPath);
|
|
if (!parent)
|
|
return PathResult{PathResult::Status::NOT_FOUND};
|
|
|
|
return addSuffix(PathResult{PathResult::Status::SUCCESS, *parent, normalizePath(relPath + "/..")});
|
|
}
|
|
|
|
PathResult getChild(const std::string& absPath, const std::string& relPath, const std::string& name)
|
|
{
|
|
return addSuffix(PathResult{PathResult::Status::SUCCESS, joinPaths(absPath, name), joinPaths(relPath, name)});
|
|
}
|
|
|
|
bool isFilePresent(const std::string& path, const std::string& suffix)
|
|
{
|
|
return isFile(path + suffix);
|
|
}
|
|
|
|
std::optional<std::string> getFileContents(const std::string& path, const std::string& suffix)
|
|
{
|
|
return readFile(path + suffix);
|
|
}
|