Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Roblox
GitHub Repository: Roblox/luau
Path: blob/master/Require/src/Navigation.cpp
2725 views
1
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
2
3
#include "Navigation.h"
4
5
#include "Luau/Require.h"
6
#include "lua.h"
7
#include "lualib.h"
8
9
#include <chrono>
10
#include <string>
11
12
static constexpr size_t initalFileBufferSize = 1024;
13
static constexpr size_t initalIdentifierBufferSize = 64;
14
15
namespace Luau::Require
16
{
17
18
static NavigationContext::NavigateResult convertNavigateResult(luarequire_NavigateResult result)
19
{
20
if (result == NAVIGATE_SUCCESS)
21
return NavigationContext::NavigateResult::Success;
22
if (result == NAVIGATE_AMBIGUOUS)
23
return NavigationContext::NavigateResult::Ambiguous;
24
25
return NavigationContext::NavigateResult::NotFound;
26
}
27
28
static NavigationContext::ConfigStatus convertConfigStatus(luarequire_ConfigStatus status)
29
{
30
if (status == CONFIG_PRESENT_JSON)
31
return NavigationContext::ConfigStatus::PresentJson;
32
if (status == CONFIG_PRESENT_LUAU)
33
return NavigationContext::ConfigStatus::PresentLuau;
34
if (status == CONFIG_AMBIGUOUS)
35
return NavigationContext::ConfigStatus::Ambiguous;
36
37
return NavigationContext::ConfigStatus::Absent;
38
}
39
40
RuntimeNavigationContext::RuntimeNavigationContext(luarequire_Configuration* config, lua_State* L, void* ctx, std::string requirerChunkname)
41
: config(config)
42
, L(L)
43
, ctx(ctx)
44
, requirerChunkname(std::move(requirerChunkname))
45
{
46
luauConfigInit = [config, ctx, this](lua_State* L)
47
{
48
int timeout = config->get_luau_config_timeout ? config->get_luau_config_timeout(L, ctx) : 2000;
49
this->timer.start(timeout);
50
lua_setthreaddata(L, &this->timer);
51
};
52
53
luauConfigInterrupt = [](lua_State* L, int gc)
54
{
55
RuntimeLuauConfigTimer* timer = static_cast<RuntimeLuauConfigTimer*>(lua_getthreaddata(L));
56
LUAU_ASSERT(timer);
57
if (timer->isFinished())
58
luaL_errorL(L, "configuration execution timed out");
59
};
60
}
61
62
NavigationContext::NavigateResult RuntimeNavigationContext::resetToRequirer()
63
{
64
return convertNavigateResult(config->reset(L, ctx, requirerChunkname.c_str()));
65
}
66
67
NavigationContext::NavigateResult RuntimeNavigationContext::jumpToAlias(const std::string& path)
68
{
69
return convertNavigateResult(config->jump_to_alias(L, ctx, path.c_str()));
70
}
71
72
NavigationContext::NavigateResult RuntimeNavigationContext::toAliasOverride(const std::string& aliasUnprefixed)
73
{
74
if (!config->to_alias_override)
75
return NavigationContext::NavigateResult::NotFound;
76
return convertNavigateResult(config->to_alias_override(L, ctx, aliasUnprefixed.c_str()));
77
}
78
79
NavigationContext::NavigateResult RuntimeNavigationContext::toAliasFallback(const std::string& aliasUnprefixed)
80
{
81
if (!config->to_alias_fallback)
82
return NavigationContext::NavigateResult::NotFound;
83
return convertNavigateResult(config->to_alias_fallback(L, ctx, aliasUnprefixed.c_str()));
84
}
85
86
NavigationContext::NavigateResult RuntimeNavigationContext::toParent()
87
{
88
return convertNavigateResult(config->to_parent(L, ctx));
89
}
90
91
NavigationContext::NavigateResult RuntimeNavigationContext::toChild(const std::string& component)
92
{
93
return convertNavigateResult(config->to_child(L, ctx, component.c_str()));
94
}
95
96
bool RuntimeNavigationContext::isModulePresent() const
97
{
98
return config->is_module_present(L, ctx);
99
}
100
101
std::optional<std::string> RuntimeNavigationContext::getChunkname() const
102
{
103
return getStringFromCWriter(config->get_chunkname, initalIdentifierBufferSize);
104
}
105
106
std::optional<std::string> RuntimeNavigationContext::getLoadname() const
107
{
108
return getStringFromCWriter(config->get_loadname, initalIdentifierBufferSize);
109
}
110
111
std::optional<std::string> RuntimeNavigationContext::getCacheKey() const
112
{
113
return getStringFromCWriter(config->get_cache_key, initalIdentifierBufferSize);
114
}
115
116
NavigationContext::ConfigStatus RuntimeNavigationContext::getConfigStatus() const
117
{
118
return convertConfigStatus(config->get_config_status(L, ctx));
119
}
120
121
NavigationContext::ConfigBehavior RuntimeNavigationContext::getConfigBehavior() const
122
{
123
if (config->get_alias)
124
return ConfigBehavior::GetAlias;
125
return ConfigBehavior::GetConfig;
126
}
127
128
std::optional<std::string> RuntimeNavigationContext::getAlias(const std::string& alias) const
129
{
130
return getStringFromCWriterWithInput(config->get_alias, alias, initalIdentifierBufferSize);
131
}
132
133
std::optional<std::string> RuntimeNavigationContext::getConfig() const
134
{
135
return getStringFromCWriter(config->get_config, initalFileBufferSize);
136
}
137
138
std::optional<std::string> RuntimeNavigationContext::getStringFromCWriter(
139
luarequire_WriteResult (*writer)(lua_State* L, void* ctx, char* buffer, size_t buffer_size, size_t* size_out),
140
size_t initalBufferSize
141
) const
142
{
143
std::string buffer;
144
buffer.resize(initalBufferSize);
145
146
size_t size;
147
luarequire_WriteResult result = writer(L, ctx, buffer.data(), buffer.size(), &size);
148
if (result == WRITE_BUFFER_TOO_SMALL)
149
{
150
buffer.resize(size);
151
result = writer(L, ctx, buffer.data(), buffer.size(), &size);
152
}
153
154
if (result == WRITE_SUCCESS)
155
{
156
buffer.resize(size);
157
return buffer;
158
}
159
160
return std::nullopt;
161
}
162
163
std::optional<std::string> RuntimeNavigationContext::getStringFromCWriterWithInput(
164
luarequire_WriteResult (*writer)(lua_State* L, void* ctx, const char* input, char* buffer, size_t buffer_size, size_t* size_out),
165
std::string input,
166
size_t initalBufferSize
167
) const
168
{
169
std::string buffer;
170
buffer.resize(initalBufferSize);
171
172
size_t size;
173
luarequire_WriteResult result = writer(L, ctx, input.c_str(), buffer.data(), buffer.size(), &size);
174
if (result == WRITE_BUFFER_TOO_SMALL)
175
{
176
buffer.resize(size);
177
result = writer(L, ctx, input.c_str(), buffer.data(), buffer.size(), &size);
178
}
179
180
if (result == WRITE_SUCCESS)
181
{
182
buffer.resize(size);
183
return buffer;
184
}
185
186
return std::nullopt;
187
}
188
189
RuntimeErrorHandler::RuntimeErrorHandler(std::string requiredPath)
190
: errorPrefix("error requiring module \"" + std::move(requiredPath) + "\": ")
191
{
192
}
193
194
void RuntimeErrorHandler::reportError(std::string message)
195
{
196
errorMessage = errorPrefix + std::move(message);
197
}
198
199
const std::string& RuntimeErrorHandler::getReportedError() const
200
{
201
return errorMessage;
202
}
203
204
void RuntimeLuauConfigTimer::start(int timeoutMs)
205
{
206
startTime = std::chrono::steady_clock::now();
207
if (timeoutMs < 0)
208
timeoutDuration = std::nullopt; // Infinite timeout
209
else
210
timeoutDuration = std::chrono::milliseconds(timeoutMs);
211
}
212
213
bool RuntimeLuauConfigTimer::isFinished() const
214
{
215
if (!timeoutDuration)
216
return false;
217
return (std::chrono::steady_clock::now() - startTime) >= *timeoutDuration;
218
}
219
220
} // namespace Luau::Require
221
222