forked from tact-lang/tact
-
Notifications
You must be signed in to change notification settings - Fork 0
/
tact.js
executable file
·197 lines (180 loc) · 7.51 KB
/
tact.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
#!/usr/bin/env node
const pkg = require("../package.json");
// eslint-disable-next-line @typescript-eslint/no-var-requires
const main = require("../dist/node.js");
const meowModule = import("meow");
// eslint-disable-next-line @typescript-eslint/no-var-requires
const { execFileSync } = require("child_process");
void meowModule.then(
/** @param meow {import('meow/build/index')} */
(meow) => {
const cli = meow.default(
`
Usage
$ tact [...flags] (--config CONFIG | FILE)
Flags
-c, --config CONFIG Specify path to config file (tact.config.json)
-p, --project ...names Build only the specified project name(s) from the config file
-q, --quiet Suppress compiler log output
--with-decompilation Full compilation followed by decompilation of produced binary code
--func Output intermediate FunC code and exit
--check Perform syntax and type checking, then exit
-e, --eval EXPRESSION Evaluate a Tact expression and exit
-v, --version Print Tact compiler version and exit
-h, --help Display this text and exit
Examples
$ tact --version
${pkg.version}
Learn more about Tact: https://docs.tact-lang.org
Join Telegram group: https://t.me/tactlang
Follow X/Twitter account: https://twitter.com/tact_language`,
{
importMeta: {
url: new URL("file://" + __dirname + __filename).toString(),
},
description: `Command-line utility for the Tact compiler:\n${pkg.description}`,
autoVersion: false,
flags: {
config: {
shortFlag: "c",
type: "string",
isRequired: (flags, _) => {
// Require a config when the projects are specified
// AND version/help are not specified
// AND eval is not specified
return (
flags.projects.length !== 0 &&
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
!flags.version &&
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
!flags.help &&
!flags.eval
);
},
},
projects: {
shortFlag: "p",
type: "string",
isMultiple: true,
},
quiet: { shortFlag: "q", type: "boolean", default: false },
withDecompilation: { type: "boolean", default: false },
func: { type: "boolean", default: false },
check: { type: "boolean", default: false },
eval: { shortFlag: "e", type: "string" },
version: { shortFlag: "v", type: "boolean" },
help: { shortFlag: "h", type: "boolean" },
},
allowUnknownFlags: false,
},
);
// Helper function to write less in following checks
const isEmptyConfigAndInput = () => {
return cli.flags.config === undefined && cli.input.length === 0;
};
// Show help regardless of other flags
if (cli.flags.help) {
cli.showHelp(0);
}
// Show version regardless of other flags
if (cli.flags.version) {
console.log(pkg.version);
// if working inside a git repository
// also print the current git commit hash
try {
const gitCommit = execFileSync("git", ["rev-parse", "HEAD"], {
encoding: "utf8",
stdio: ["ignore", "pipe", "ignore"],
}).trim();
console.log(`git commit: ${gitCommit}`);
} finally {
process.exit(0);
}
}
// Evaluate expression regardless of other flags
if (cli.flags.eval) {
const result = main.parseAndEvalExpression(cli.flags.eval);
switch (result.kind) {
case "ok":
{
console.log(result.value);
process.exit(0);
}
break;
case "error": {
console.log(result.message);
process.exit(30);
}
}
}
// Disallow specifying both config or Tact source file at the same time
if (cli.flags.config !== undefined && cli.input.length > 0) {
console.log(
"Error: Both config and Tact file can't be simultaneously specified, pick one!",
);
cli.showHelp();
}
// Disallow specifying several exclusive compilation mode flags
const compilationModeFlags = [
cli.flags.check,
cli.flags.func,
cli.flags.withDecompilation,
];
const numOfCompilationModeFlagsSet = compilationModeFlags.filter(
(flag) => flag,
).length;
if (numOfCompilationModeFlagsSet > 1) {
console.log(
"Error: Flags --with-decompilation, --func and --check are mutually exclusive!",
);
cli.showHelp();
}
// Disallow using compilation mode flags without a config or a file specified
if (isEmptyConfigAndInput() && numOfCompilationModeFlagsSet > 0) {
console.log(
"Error: Either config or Tact file have to be specified!",
);
cli.showHelp();
}
// Disallow specifying more than one Tact file
if (cli.input.length > 1) {
console.log(
"Error: Only one Tact file can be specified at a time. If you want more, provide a config!",
);
cli.showHelp();
}
// Show help when all flags and inputs are empty
// Note, that version/help flags are already processed above and don't need to be mentioned here
if (
isEmptyConfigAndInput() &&
numOfCompilationModeFlagsSet === 0 &&
cli.flags.projects.length === 0
) {
cli.showHelp(0);
}
// Compilation mode
const mode = cli.flags.check
? "checkOnly"
: cli.flags.func
? "funcOnly"
: cli.flags.withDecompilation
? "fullWithDecompilation"
: undefined;
// TODO: all flags on the cli should take precedence over flags in the config
// Make a nice model for it in the src/node.ts instead of the current mess
// Consider making overwrites right here or something.
// Main command
void main
.run({
fileName: cli.input.at(0),
configPath: cli.flags.config,
projectNames: cli.flags.projects ?? [],
additionalCliOptions: { mode },
suppressLog: cli.flags.quiet,
})
.then((response) => {
// https://nodejs.org/docs/v20.12.1/api/process.html#exit-codes
process.exit(response.ok ? 0 : 30);
});
},
);