2021-10-29 21:25:12 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/JsonEncoder.h"
|
|
|
|
|
|
|
|
#include "Luau/Ast.h"
|
|
|
|
#include "Luau/StringUtils.h"
|
2021-11-05 02:34:35 +00:00
|
|
|
#include "Luau/Common.h"
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
|
|
|
struct AstJsonEncoder : public AstVisitor
|
|
|
|
{
|
|
|
|
static constexpr int CHUNK_SIZE = 1024;
|
|
|
|
std::vector<std::string> chunks;
|
|
|
|
bool comma = false;
|
|
|
|
|
|
|
|
AstJsonEncoder()
|
|
|
|
{
|
|
|
|
newChunk();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string str()
|
|
|
|
{
|
|
|
|
return join(chunks, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool pushComma()
|
|
|
|
{
|
|
|
|
bool c = comma;
|
|
|
|
comma = false;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void popComma(bool c)
|
|
|
|
{
|
|
|
|
comma = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void newChunk()
|
|
|
|
{
|
|
|
|
chunks.emplace_back();
|
|
|
|
chunks.back().reserve(CHUNK_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void appendChunk(std::string_view sv)
|
|
|
|
{
|
|
|
|
if (sv.size() > CHUNK_SIZE)
|
|
|
|
{
|
|
|
|
chunks.emplace_back(sv);
|
|
|
|
newChunk();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& chunk = chunks.back();
|
|
|
|
if (chunk.size() + sv.size() < CHUNK_SIZE)
|
|
|
|
{
|
|
|
|
chunk.append(sv.data(), sv.size());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t prefix = CHUNK_SIZE - chunk.size();
|
|
|
|
chunk.append(sv.data(), prefix);
|
|
|
|
newChunk();
|
|
|
|
|
|
|
|
chunks.back().append(sv.data() + prefix, sv.size() - prefix);
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeRaw(std::string_view sv)
|
|
|
|
{
|
|
|
|
appendChunk(sv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeRaw(char c)
|
|
|
|
{
|
|
|
|
writeRaw(std::string_view{&c, 1});
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void write(std::string_view propName, const T& value)
|
|
|
|
{
|
|
|
|
if (comma)
|
|
|
|
writeRaw(",");
|
|
|
|
comma = true;
|
|
|
|
writeRaw("\"");
|
|
|
|
writeRaw(propName);
|
|
|
|
writeRaw("\":");
|
|
|
|
write(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(bool b)
|
|
|
|
{
|
|
|
|
if (b)
|
|
|
|
writeRaw("true");
|
|
|
|
else
|
|
|
|
writeRaw("false");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(double d)
|
|
|
|
{
|
|
|
|
char b[256];
|
|
|
|
sprintf(b, "%g", d);
|
|
|
|
writeRaw(b);
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeString(std::string_view sv)
|
|
|
|
{
|
|
|
|
// TODO escape more accurately?
|
|
|
|
writeRaw("\"");
|
|
|
|
|
|
|
|
for (char c : sv)
|
|
|
|
{
|
|
|
|
if (c == '"')
|
|
|
|
writeRaw("\\\"");
|
|
|
|
else if (c == '\0')
|
|
|
|
writeRaw("\\\0");
|
|
|
|
else
|
|
|
|
writeRaw(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
writeRaw("\"");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(char c)
|
|
|
|
{
|
|
|
|
writeString(std::string_view(&c, 1));
|
|
|
|
}
|
|
|
|
void write(int i)
|
|
|
|
{
|
|
|
|
writeRaw(std::to_string(i));
|
|
|
|
}
|
|
|
|
void write(long i)
|
|
|
|
{
|
|
|
|
writeRaw(std::to_string(i));
|
|
|
|
}
|
|
|
|
void write(long long i)
|
|
|
|
{
|
|
|
|
writeRaw(std::to_string(i));
|
|
|
|
}
|
|
|
|
void write(unsigned int i)
|
|
|
|
{
|
|
|
|
writeRaw(std::to_string(i));
|
|
|
|
}
|
|
|
|
void write(unsigned long i)
|
|
|
|
{
|
|
|
|
writeRaw(std::to_string(i));
|
|
|
|
}
|
|
|
|
void write(unsigned long long i)
|
|
|
|
{
|
|
|
|
writeRaw(std::to_string(i));
|
|
|
|
}
|
2022-02-14 22:14:11 +00:00
|
|
|
void write(std::nullptr_t)
|
|
|
|
{
|
|
|
|
writeRaw("null");
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
void write(std::string_view str)
|
|
|
|
{
|
|
|
|
writeString(str);
|
|
|
|
}
|
2022-02-18 01:18:01 +00:00
|
|
|
void write(std::optional<AstName> name)
|
|
|
|
{
|
|
|
|
if (name)
|
|
|
|
write(*name);
|
|
|
|
else
|
|
|
|
writeRaw("null");
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
void write(AstName name)
|
|
|
|
{
|
|
|
|
writeString(name.value ? name.value : "");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const Position& position)
|
|
|
|
{
|
|
|
|
write(position.line);
|
|
|
|
writeRaw(",");
|
|
|
|
write(position.column);
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const Location& location)
|
|
|
|
{
|
|
|
|
writeRaw("\"");
|
|
|
|
write(location.begin);
|
|
|
|
writeRaw(" - ");
|
|
|
|
write(location.end);
|
|
|
|
writeRaw("\"");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(AstLocal* local)
|
|
|
|
{
|
2022-02-14 22:14:11 +00:00
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
if (local->annotation != nullptr)
|
|
|
|
write("type", local->annotation);
|
|
|
|
else
|
|
|
|
write("type", nullptr);
|
|
|
|
write("name", local->name);
|
|
|
|
write("location", local->location);
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void writeNode(AstNode* node)
|
|
|
|
{
|
|
|
|
write("location", node->location);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F>
|
|
|
|
void writeNode(AstNode* node, std::string_view name, F&& f)
|
|
|
|
{
|
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
write("type", name);
|
|
|
|
writeNode(node);
|
|
|
|
f();
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(AstNode* node)
|
|
|
|
{
|
|
|
|
node->visit(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprGroup* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprGroup", [&]() {
|
|
|
|
write("expr", node->expr);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprConstantNil* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprConstantNil", []() {});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprConstantBool* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprConstantBool", [&]() {
|
|
|
|
write("value", node->value);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprConstantNumber* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprConstantNumber", [&]() {
|
|
|
|
write("value", node->value);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprConstantString* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprConstantString", [&]() {
|
|
|
|
write("value", node->value);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprLocal* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprLocal", [&]() {
|
|
|
|
write("local", node->local);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprGlobal* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprGlobal", [&]() {
|
|
|
|
write("global", node->name);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprVarargs* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprVarargs", []() {});
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void write(AstArray<T> arr)
|
|
|
|
{
|
|
|
|
writeRaw("[");
|
|
|
|
bool comma = false;
|
|
|
|
for (const auto& a : arr)
|
|
|
|
{
|
|
|
|
if (comma)
|
|
|
|
writeRaw(",");
|
|
|
|
else
|
2021-12-07 16:14:35 +00:00
|
|
|
comma = true;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
write(a);
|
|
|
|
}
|
|
|
|
writeRaw("]");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(AstArray<char> arr)
|
|
|
|
{
|
|
|
|
write(std::string_view{arr.data, arr.size});
|
|
|
|
}
|
|
|
|
|
|
|
|
#define PROP(prop) write(#prop, node->prop)
|
|
|
|
|
|
|
|
void write(class AstExprCall* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprCall", [&]() {
|
|
|
|
PROP(func);
|
|
|
|
PROP(args);
|
|
|
|
PROP(self);
|
|
|
|
PROP(argLocation);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprIndexName* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprIndexName", [&]() {
|
|
|
|
PROP(expr);
|
|
|
|
PROP(index);
|
|
|
|
PROP(indexLocation);
|
|
|
|
PROP(op);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprIndexExpr* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprIndexExpr", [&]() {
|
|
|
|
PROP(expr);
|
|
|
|
PROP(index);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprFunction* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprFunction", [&]() {
|
|
|
|
PROP(generics);
|
|
|
|
PROP(genericPacks);
|
|
|
|
if (node->self)
|
|
|
|
PROP(self);
|
|
|
|
PROP(args);
|
2022-02-18 01:18:01 +00:00
|
|
|
if (node->returnAnnotation)
|
2021-10-29 21:25:12 +01:00
|
|
|
PROP(returnAnnotation);
|
|
|
|
PROP(vararg);
|
|
|
|
PROP(varargLocation);
|
|
|
|
if (node->varargAnnotation)
|
|
|
|
PROP(varargAnnotation);
|
|
|
|
|
|
|
|
PROP(body);
|
|
|
|
PROP(functionDepth);
|
|
|
|
PROP(debugname);
|
|
|
|
PROP(hasEnd);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-18 01:18:01 +00:00
|
|
|
void write(const std::optional<AstTypeList>& typeList)
|
|
|
|
{
|
|
|
|
if (typeList)
|
|
|
|
write(*typeList);
|
|
|
|
else
|
|
|
|
writeRaw("null");
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
void write(const AstTypeList& typeList)
|
|
|
|
{
|
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
write("types", typeList.types);
|
|
|
|
if (typeList.tailType)
|
|
|
|
write("tailType", typeList.tailType);
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
|
|
|
}
|
|
|
|
|
2022-01-14 16:20:09 +00:00
|
|
|
void write(const AstGenericType& genericType)
|
|
|
|
{
|
2022-03-04 16:36:33 +00:00
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
write("name", genericType.name);
|
|
|
|
if (genericType.defaultValue)
|
|
|
|
write("type", genericType.defaultValue);
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
2022-01-14 16:20:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void write(const AstGenericTypePack& genericTypePack)
|
|
|
|
{
|
2022-03-04 16:36:33 +00:00
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
write("name", genericTypePack.name);
|
|
|
|
if (genericTypePack.defaultValue)
|
|
|
|
write("type", genericTypePack.defaultValue);
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
2022-01-14 16:20:09 +00:00
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
void write(AstExprTable::Item::Kind kind)
|
|
|
|
{
|
|
|
|
switch (kind)
|
|
|
|
{
|
|
|
|
case AstExprTable::Item::List:
|
|
|
|
return writeString("item");
|
|
|
|
case AstExprTable::Item::Record:
|
|
|
|
return writeString("record");
|
|
|
|
case AstExprTable::Item::General:
|
|
|
|
return writeString("general");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const AstExprTable::Item& item)
|
|
|
|
{
|
|
|
|
writeRaw("{");
|
Fix JsonEncoder for AstExprTable (#454)
JsonEncoder wasn't producing valid JSON for `AstExprTable`s. This PR fixes it. The new output looks like
```json
{
"type": "AstStatBlock",
"location": "0,0 - 6,4",
"body": [
{
"type": "AstStatLocal",
"location": "1,8 - 5,9",
"vars": [
{
"name": "x",
"location": "1,14 - 1,15"
}
],
"values": [
{
"type": "AstExprTable",
"location": "3,12 - 5,9",
"items": [
{
"kind": "record",
"key": {
"type": "AstExprConstantString",
"location": "4,12 - 4,15",
"value": "foo"
},
"value": {
"type": "AstExprConstantNumber",
"location": "4,18 - 4,21",
"value": 123
}
}
]
}
]
}
]
}
```
2022-04-08 19:26:47 +01:00
|
|
|
bool c = pushComma();
|
2021-10-29 21:25:12 +01:00
|
|
|
write("kind", item.kind);
|
|
|
|
switch (item.kind)
|
|
|
|
{
|
|
|
|
case AstExprTable::Item::List:
|
Fix JsonEncoder for AstExprTable (#454)
JsonEncoder wasn't producing valid JSON for `AstExprTable`s. This PR fixes it. The new output looks like
```json
{
"type": "AstStatBlock",
"location": "0,0 - 6,4",
"body": [
{
"type": "AstStatLocal",
"location": "1,8 - 5,9",
"vars": [
{
"name": "x",
"location": "1,14 - 1,15"
}
],
"values": [
{
"type": "AstExprTable",
"location": "3,12 - 5,9",
"items": [
{
"kind": "record",
"key": {
"type": "AstExprConstantString",
"location": "4,12 - 4,15",
"value": "foo"
},
"value": {
"type": "AstExprConstantNumber",
"location": "4,18 - 4,21",
"value": 123
}
}
]
}
]
}
]
}
```
2022-04-08 19:26:47 +01:00
|
|
|
write("value", item.value);
|
2021-10-29 21:25:12 +01:00
|
|
|
break;
|
|
|
|
default:
|
Fix JsonEncoder for AstExprTable (#454)
JsonEncoder wasn't producing valid JSON for `AstExprTable`s. This PR fixes it. The new output looks like
```json
{
"type": "AstStatBlock",
"location": "0,0 - 6,4",
"body": [
{
"type": "AstStatLocal",
"location": "1,8 - 5,9",
"vars": [
{
"name": "x",
"location": "1,14 - 1,15"
}
],
"values": [
{
"type": "AstExprTable",
"location": "3,12 - 5,9",
"items": [
{
"kind": "record",
"key": {
"type": "AstExprConstantString",
"location": "4,12 - 4,15",
"value": "foo"
},
"value": {
"type": "AstExprConstantNumber",
"location": "4,18 - 4,21",
"value": 123
}
}
]
}
]
}
]
}
```
2022-04-08 19:26:47 +01:00
|
|
|
write("key", item.key);
|
|
|
|
write("value", item.value);
|
2021-10-29 21:25:12 +01:00
|
|
|
break;
|
|
|
|
}
|
Fix JsonEncoder for AstExprTable (#454)
JsonEncoder wasn't producing valid JSON for `AstExprTable`s. This PR fixes it. The new output looks like
```json
{
"type": "AstStatBlock",
"location": "0,0 - 6,4",
"body": [
{
"type": "AstStatLocal",
"location": "1,8 - 5,9",
"vars": [
{
"name": "x",
"location": "1,14 - 1,15"
}
],
"values": [
{
"type": "AstExprTable",
"location": "3,12 - 5,9",
"items": [
{
"kind": "record",
"key": {
"type": "AstExprConstantString",
"location": "4,12 - 4,15",
"value": "foo"
},
"value": {
"type": "AstExprConstantNumber",
"location": "4,18 - 4,21",
"value": 123
}
}
]
}
]
}
]
}
```
2022-04-08 19:26:47 +01:00
|
|
|
popComma(c);
|
2021-10-29 21:25:12 +01:00
|
|
|
writeRaw("}");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprTable* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprTable", [&]() {
|
Fix JsonEncoder for AstExprTable (#454)
JsonEncoder wasn't producing valid JSON for `AstExprTable`s. This PR fixes it. The new output looks like
```json
{
"type": "AstStatBlock",
"location": "0,0 - 6,4",
"body": [
{
"type": "AstStatLocal",
"location": "1,8 - 5,9",
"vars": [
{
"name": "x",
"location": "1,14 - 1,15"
}
],
"values": [
{
"type": "AstExprTable",
"location": "3,12 - 5,9",
"items": [
{
"kind": "record",
"key": {
"type": "AstExprConstantString",
"location": "4,12 - 4,15",
"value": "foo"
},
"value": {
"type": "AstExprConstantNumber",
"location": "4,18 - 4,21",
"value": 123
}
}
]
}
]
}
]
}
```
2022-04-08 19:26:47 +01:00
|
|
|
PROP(items);
|
2021-10-29 21:25:12 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(AstExprUnary::Op op)
|
|
|
|
{
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case AstExprUnary::Not:
|
|
|
|
return writeString("not");
|
|
|
|
case AstExprUnary::Minus:
|
|
|
|
return writeString("minus");
|
|
|
|
case AstExprUnary::Len:
|
|
|
|
return writeString("len");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprUnary* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprUnary", [&]() {
|
|
|
|
PROP(op);
|
|
|
|
PROP(expr);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(AstExprBinary::Op op)
|
|
|
|
{
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case AstExprBinary::Add:
|
|
|
|
return writeString("Add");
|
|
|
|
case AstExprBinary::Sub:
|
|
|
|
return writeString("Sub");
|
|
|
|
case AstExprBinary::Mul:
|
|
|
|
return writeString("Mul");
|
|
|
|
case AstExprBinary::Div:
|
|
|
|
return writeString("Div");
|
|
|
|
case AstExprBinary::Mod:
|
|
|
|
return writeString("Mod");
|
|
|
|
case AstExprBinary::Pow:
|
|
|
|
return writeString("Pow");
|
|
|
|
case AstExprBinary::Concat:
|
|
|
|
return writeString("Concat");
|
|
|
|
case AstExprBinary::CompareNe:
|
|
|
|
return writeString("CompareNe");
|
|
|
|
case AstExprBinary::CompareEq:
|
|
|
|
return writeString("CompareEq");
|
|
|
|
case AstExprBinary::CompareLt:
|
|
|
|
return writeString("CompareLt");
|
|
|
|
case AstExprBinary::CompareLe:
|
|
|
|
return writeString("CompareLe");
|
|
|
|
case AstExprBinary::CompareGt:
|
|
|
|
return writeString("CompareGt");
|
|
|
|
case AstExprBinary::CompareGe:
|
|
|
|
return writeString("CompareGe");
|
|
|
|
case AstExprBinary::And:
|
|
|
|
return writeString("And");
|
|
|
|
case AstExprBinary::Or:
|
|
|
|
return writeString("Or");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprBinary* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprBinary", [&]() {
|
|
|
|
PROP(op);
|
|
|
|
PROP(left);
|
|
|
|
PROP(right);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprTypeAssertion* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprTypeAssertion", [&]() {
|
|
|
|
PROP(expr);
|
|
|
|
PROP(annotation);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstExprError* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstExprError", [&]() {
|
|
|
|
PROP(expressions);
|
|
|
|
PROP(messageIndex);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatBlock* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatBlock", [&]() {
|
|
|
|
writeRaw(",\"body\":[");
|
|
|
|
bool comma = false;
|
|
|
|
for (AstStat* stat : node->body)
|
|
|
|
{
|
|
|
|
if (comma)
|
|
|
|
writeRaw(",");
|
|
|
|
else
|
|
|
|
comma = true;
|
|
|
|
|
|
|
|
write(stat);
|
|
|
|
}
|
|
|
|
writeRaw("]");
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatIf* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatIf", [&]() {
|
|
|
|
PROP(condition);
|
|
|
|
PROP(thenbody);
|
|
|
|
if (node->elsebody)
|
|
|
|
PROP(elsebody);
|
2022-02-18 01:18:01 +00:00
|
|
|
write("hasThen", node->thenLocation.has_value());
|
2021-10-29 21:25:12 +01:00
|
|
|
PROP(hasEnd);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatWhile* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AtStatWhile", [&]() {
|
|
|
|
PROP(condition);
|
|
|
|
PROP(body);
|
|
|
|
PROP(hasDo);
|
|
|
|
PROP(hasEnd);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatRepeat* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatRepeat", [&]() {
|
|
|
|
PROP(condition);
|
|
|
|
PROP(body);
|
|
|
|
PROP(hasUntil);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatBreak* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatBreak", []() {});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatContinue* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatContinue", []() {});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatReturn* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatReturn", [&]() {
|
|
|
|
PROP(list);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatExpr* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatExpr", [&]() {
|
|
|
|
PROP(expr);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatLocal* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatLocal", [&]() {
|
|
|
|
PROP(vars);
|
|
|
|
PROP(values);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatFor* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatFor", [&]() {
|
|
|
|
PROP(var);
|
|
|
|
PROP(from);
|
|
|
|
PROP(to);
|
|
|
|
if (node->step)
|
|
|
|
PROP(step);
|
|
|
|
PROP(body);
|
|
|
|
PROP(hasDo);
|
|
|
|
PROP(hasEnd);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatForIn* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatForIn", [&]() {
|
|
|
|
PROP(vars);
|
|
|
|
PROP(values);
|
|
|
|
PROP(body);
|
|
|
|
PROP(hasIn);
|
|
|
|
PROP(hasDo);
|
|
|
|
PROP(hasEnd);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatAssign* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatAssign", [&]() {
|
|
|
|
PROP(vars);
|
|
|
|
PROP(values);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatCompoundAssign* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatCompoundAssign", [&]() {
|
|
|
|
PROP(op);
|
|
|
|
PROP(var);
|
|
|
|
PROP(value);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatFunction* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatFunction", [&]() {
|
|
|
|
PROP(name);
|
|
|
|
PROP(func);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatLocalFunction* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatLocalFunction", [&]() {
|
|
|
|
PROP(name);
|
|
|
|
PROP(func);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatTypeAlias* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatTypeAlias", [&]() {
|
|
|
|
PROP(name);
|
|
|
|
PROP(generics);
|
2021-12-03 06:41:04 +00:00
|
|
|
PROP(genericPacks);
|
2021-10-29 21:25:12 +01:00
|
|
|
PROP(type);
|
|
|
|
PROP(exported);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatDeclareFunction* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatDeclareFunction", [&]() {
|
|
|
|
PROP(name);
|
|
|
|
PROP(params);
|
|
|
|
PROP(retTypes);
|
|
|
|
PROP(generics);
|
|
|
|
PROP(genericPacks);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatDeclareGlobal* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatDeclareGlobal", [&]() {
|
|
|
|
PROP(name);
|
|
|
|
PROP(type);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const AstDeclaredClassProp& prop)
|
|
|
|
{
|
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
write("name", prop.name);
|
|
|
|
write("type", prop.ty);
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatDeclareClass* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatDeclareClass", [&]() {
|
|
|
|
PROP(name);
|
|
|
|
if (node->superName)
|
|
|
|
write("superName", *node->superName);
|
|
|
|
PROP(props);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstStatError* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstStatError", [&]() {
|
|
|
|
PROP(expressions);
|
|
|
|
PROP(statements);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-11-05 02:34:35 +00:00
|
|
|
void write(struct AstTypeOrPack node)
|
|
|
|
{
|
|
|
|
if (node.type)
|
|
|
|
write(node.type);
|
|
|
|
else
|
|
|
|
write(node.typePack);
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
void write(class AstTypeReference* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeReference", [&]() {
|
2022-02-18 01:18:01 +00:00
|
|
|
if (node->prefix)
|
2021-10-29 21:25:12 +01:00
|
|
|
PROP(prefix);
|
|
|
|
PROP(name);
|
2021-11-05 02:34:35 +00:00
|
|
|
PROP(parameters);
|
2021-10-29 21:25:12 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(const AstTableProp& prop)
|
|
|
|
{
|
|
|
|
writeRaw("{");
|
|
|
|
bool c = pushComma();
|
|
|
|
|
|
|
|
write("name", prop.name);
|
|
|
|
write("location", prop.location);
|
|
|
|
write("type", prop.type);
|
|
|
|
|
|
|
|
popComma(c);
|
|
|
|
writeRaw("}");
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypeTable* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeTable", [&]() {
|
|
|
|
PROP(props);
|
|
|
|
PROP(indexer);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypeFunction* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeFunction", [&]() {
|
|
|
|
PROP(generics);
|
|
|
|
PROP(genericPacks);
|
|
|
|
PROP(argTypes);
|
|
|
|
PROP(returnTypes);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypeTypeof* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeTypeof", [&]() {
|
|
|
|
PROP(expr);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypeUnion* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeUnion", [&]() {
|
|
|
|
PROP(types);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypeIntersection* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeIntersection", [&]() {
|
|
|
|
PROP(types);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypeError* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypeError", [&]() {
|
|
|
|
PROP(types);
|
|
|
|
PROP(messageIndex);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-11-05 02:34:35 +00:00
|
|
|
void write(class AstTypePackExplicit* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypePackExplicit", [&]() {
|
|
|
|
PROP(typeList);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
void write(class AstTypePackVariadic* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypePackVariadic", [&]() {
|
|
|
|
PROP(variadicType);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(class AstTypePackGeneric* node)
|
|
|
|
{
|
|
|
|
writeNode(node, "AstTypePackGeneric", [&]() {
|
|
|
|
PROP(genericName);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprGroup* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprConstantNil* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprConstantBool* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprConstantNumber* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprConstantString* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprLocal* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprGlobal* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprVarargs* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprCall* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprIndexName* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprIndexExpr* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprFunction* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprTable* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprUnary* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprBinary* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprTypeAssertion* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstExprError* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatBlock* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatIf* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatWhile* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatRepeat* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatBreak* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatContinue* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatReturn* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatExpr* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatLocal* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatFor* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatForIn* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatAssign* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatCompoundAssign* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatFunction* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatLocalFunction* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatTypeAlias* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatDeclareFunction* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatDeclareGlobal* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatDeclareClass* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstStatError* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeReference* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeTable* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeFunction* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeTypeof* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeUnion* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeIntersection* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypeError* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypePack* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-11-05 02:34:35 +00:00
|
|
|
bool visit(class AstTypePackExplicit* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
bool visit(class AstTypePackVariadic* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypePackGeneric* node) override
|
|
|
|
{
|
|
|
|
write(node);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::string toJson(AstNode* node)
|
|
|
|
{
|
|
|
|
AstJsonEncoder encoder;
|
|
|
|
node->visit(&encoder);
|
|
|
|
return encoder.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Luau
|