diff --git a/CLI/src/Web.cpp b/CLI/src/Web.cpp index 416a79f2..f9a54cb0 100644 --- a/CLI/src/Web.cpp +++ b/CLI/src/Web.cpp @@ -5,6 +5,11 @@ #include "Luau/Common.h" +// Analysis files +#include "Luau/Frontend.h" +#include "Luau/FileResolver.h" +#include + #include #include @@ -112,3 +117,270 @@ extern "C" const char* executeScript(const char* source) return result.empty() ? NULL : result.c_str(); } + +// Analysis +namespace Luau +{ +static std::vector parsePathExpr(const AstExpr& pathExpr) +{ + const AstExprIndexName* indexName = pathExpr.as(); + if (!indexName) + return {}; + + std::vector segments{indexName->index.value}; + + while (true) + { + if (AstExprIndexName* in = indexName->expr->as()) + { + segments.push_back(in->index.value); + indexName = in; + continue; + } + else if (AstExprGlobal* indexNameAsGlobal = indexName->expr->as()) + { + segments.push_back(indexNameAsGlobal->name.value); + break; + } + else if (AstExprLocal* indexNameAsLocal = indexName->expr->as()) + { + segments.push_back(indexNameAsLocal->local->name.value); + break; + } + else + return {}; + } + + std::reverse(segments.begin(), segments.end()); + return segments; +} + + +std::optional pathExprToModuleName(const ModuleName& currentModuleName, const std::vector& segments) +{ + if (segments.empty()) + return std::nullopt; + + std::vector result; + + auto it = segments.begin(); + + if (*it == "script" && !currentModuleName.empty()) + { + result = split(currentModuleName, '/'); + ++it; + } + + for (; it != segments.end(); ++it) + { + if (result.size() > 1 && *it == "Parent") + result.pop_back(); + else + result.push_back(*it); + } + + return join(result, "/"); +} + +std::optional pathExprToModuleName(const ModuleName& currentModuleName, const AstExpr& pathExpr) +{ + std::vector segments = parsePathExpr(pathExpr); + return pathExprToModuleName(currentModuleName, segments); +} + +struct TestFileResolver + : FileResolver + , ModuleResolver +{ + std::optional resolveModuleInfo(const ModuleName& currentModuleName, const AstExpr& pathExpr) override; + + const ModulePtr getModule(const ModuleName& moduleName) const override; + + bool moduleExists(const ModuleName& moduleName) const override; + + std::optional readSource(const ModuleName& name) override; + + std::optional resolveModule(const ModuleInfo* context, AstExpr* expr) override; + + std::string getHumanReadableModuleName(const ModuleName& name) const override; + + std::optional getEnvironmentForModule(const ModuleName& name) const override; + + std::unordered_map source; + std::unordered_map sourceTypes; + std::unordered_map environments; +}; + +struct TestConfigResolver : ConfigResolver +{ + Config defaultConfig; + std::unordered_map configFiles; + + const Config& getConfig(const ModuleName& name) const override; +}; + +std::optional TestFileResolver::resolveModuleInfo(const ModuleName& currentModuleName, const AstExpr& pathExpr) +{ + if (auto name = pathExprToModuleName(currentModuleName, pathExpr)) + return {{*name, false}}; + + return std::nullopt; +} + +const ModulePtr TestFileResolver::getModule(const ModuleName& moduleName) const +{ + LUAU_ASSERT(false); + return nullptr; +} + +bool TestFileResolver::moduleExists(const ModuleName& moduleName) const +{ + auto it = source.find(moduleName); + return (it != source.end()); +} + +std::optional TestFileResolver::readSource(const ModuleName& name) +{ + auto it = source.find(name); + if (it == source.end()) + return std::nullopt; + + SourceCode::Type sourceType = SourceCode::Module; + + auto it2 = sourceTypes.find(name); + if (it2 != sourceTypes.end()) + sourceType = it2->second; + + return SourceCode{it->second, sourceType}; +} + +std::optional TestFileResolver::resolveModule(const ModuleInfo* context, AstExpr* expr) +{ + if (AstExprGlobal* g = expr->as()) + { + if (g->name == "game") + return ModuleInfo{"game"}; + if (g->name == "workspace") + return ModuleInfo{"workspace"}; + if (g->name == "script") + return context ? std::optional(*context) : std::nullopt; + } + else if (AstExprIndexName* i = expr->as(); i && context) + { + if (i->index == "Parent") + { + std::string_view view = context->name; + size_t lastSeparatorIndex = view.find_last_of('/'); + + if (lastSeparatorIndex == std::string_view::npos) + return std::nullopt; + + return ModuleInfo{ModuleName(view.substr(0, lastSeparatorIndex)), context->optional}; + } + else + { + return ModuleInfo{context->name + '/' + i->index.value, context->optional}; + } + } + else if (AstExprIndexExpr* i = expr->as(); i && context) + { + if (AstExprConstantString* index = i->index->as()) + { + return ModuleInfo{context->name + '/' + std::string(index->value.data, index->value.size), context->optional}; + } + } + else if (AstExprCall* call = expr->as(); call && call->self && call->args.size >= 1 && context) + { + if (AstExprConstantString* index = call->args.data[0]->as()) + { + AstName func = call->func->as()->index; + + if (func == "GetService" && context->name == "game") + return ModuleInfo{"game/" + std::string(index->value.data, index->value.size)}; + } + } + + return std::nullopt; +} + +std::string TestFileResolver::getHumanReadableModuleName(const ModuleName& name) const +{ + // We have a handful of tests that need to distinguish between a canonical + // ModuleName and the human-readable version so we apply a simple transform + // here: We replace all slashes with dots. + std::string result = name; + for (size_t i = 0; i < result.size(); ++i) + { + if (result[i] == '/') + result[i] = '.'; + } + + return result; +} + +std::optional TestFileResolver::getEnvironmentForModule(const ModuleName& name) const +{ + auto it = environments.find(name); + if (it != environments.end()) + return it->second; + + return std::nullopt; +} + +const Config& TestConfigResolver::getConfig(const ModuleName& name) const +{ + auto it = configFiles.find(name); + if (it != configFiles.end()) + return it->second; + + return defaultConfig; +} + +TestFileResolver fileResolver; +TestConfigResolver configResolver; + +CheckResult frontendCheck(Mode mode, const std::string& source, std::optional options) +{ + Luau::Frontend frontend(&fileResolver, &configResolver); + + ModuleName mm = "web"; + configResolver.defaultConfig.mode = mode; + fileResolver.source[mm] = std::move(source); + frontend.markDirty(mm); + + CheckResult result = frontend.check(mm, options); + + return result; +} + +std::string runAnalysis(const std::string& source) +{ + std::string strResult; + + CheckResult checkResult = frontendCheck(Mode::Strict, source, std::nullopt); + + // Collect errors + for (auto error : checkResult.errors) + { + strResult += toString(error) += "\n"; + } + + return strResult; +} + +}; // namespace Luau + +extern "C" const char* executeAnalysis(const char* source) +{ + // setup flags + for (Luau::FValue* flag = Luau::FValue::list; flag; flag = flag->next) + if (strncmp(flag->name, "Luau", 4) == 0) + flag->value = true; + + std::string result; + + // run Analysis + result = Luau::runAnalysis(source); + + return result.empty() ? NULL : result.c_str(); +} \ No newline at end of file diff --git a/CMakeLists.txt b/CMakeLists.txt index 5286fd9f..f750e30e 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -259,10 +259,10 @@ endif() if(LUAU_BUILD_WEB) target_compile_options(Luau.Web PRIVATE ${LUAU_OPTIONS}) - target_link_libraries(Luau.Web PRIVATE Luau.Compiler Luau.VM) + target_link_libraries(Luau.Web PRIVATE Luau.Compiler Luau.VM Luau.Analysis) # declare exported functions to emscripten - target_link_options(Luau.Web PRIVATE -sEXPORTED_FUNCTIONS=['_executeScript'] -sEXPORTED_RUNTIME_METHODS=['ccall','cwrap']) + target_link_options(Luau.Web PRIVATE -sEXPORTED_FUNCTIONS=['_executeScript','_executeAnalysis'] -sEXPORTED_RUNTIME_METHODS=['ccall','cwrap']) # add -fexceptions for emscripten to allow exceptions to be caught in C++ target_link_options(Luau.Web PRIVATE -fexceptions)