7#include "absl/strings/str_cat.h"
8#include "absl/strings/str_format.h"
9#include "absl/strings/str_join.h"
18 escaped.reserve(input.size());
19 for (
char c : input) {
22 escaped.append(
"\\\\");
25 escaped.append(
"\\\"");
28 escaped.append(
"\\n");
31 escaped.append(
"\\r");
34 escaped.append(
"\\t");
44 const std::vector<std::string>& values) {
46 for (
size_t i = 0; i < values.size(); ++i) {
57 static bool initialized =
false;
66 std::unique_ptr<resources::CommandHandler> handler,
68 std::string name = handler->GetName();
74 for (
const auto& alias : metadata.
aliases) {
83 std::vector<std::unique_ptr<resources::CommandHandler>> handlers) {
84 for (
auto& handler : handlers) {
85 std::string name = handler->GetName();
92 auto descriptor = handler->Describe();
95 metadata.
usage = handler->GetUsage();
101 if (name.find(
"resource-") == 0) {
103 metadata.
description =
"Resource inspection and search";
104 if (name ==
"resource-list") {
105 metadata.
examples = {
"z3ed resource-list --type=dungeon --format=json",
106 "z3ed resource-list --type=sprite --format=table"};
108 }
else if (name.find(
"dungeon-") == 0) {
110 metadata.
description =
"Dungeon inspection and editing";
111 if (name ==
"dungeon-describe-room") {
113 "z3ed dungeon-describe-room --room=5 --format=json"};
114 }
else if (name ==
"dungeon-place-sprite") {
116 "Place a dungeon sprite (dry-run by default, --write to apply)";
118 "z3ed dungeon-place-sprite --room=0x77 --id=0xA3 --x=16 --y=21 "
119 "--subtype=4 --rom=Roms/oos168.sfc --format=json",
120 "z3ed dungeon-place-sprite --room=0x77 --id=0xA3 --x=16 --y=21 "
121 "--subtype=4 --write --rom=/tmp/oos-work.sfc --format=json"};
122 }
else if (name ==
"dungeon-remove-sprite") {
124 "Remove a dungeon sprite by index or exact coordinates";
126 "z3ed dungeon-remove-sprite --room=0x77 --index=2 "
127 "--rom=/tmp/oos-work.sfc --format=json",
128 "z3ed dungeon-remove-sprite --room=0x77 --x=16 --y=21 --write "
129 "--rom=/tmp/oos-work.sfc --format=json"};
130 }
else if (name ==
"dungeon-place-object") {
132 "Place a dungeon object (tracks, rails, doors, etc.)";
134 "z3ed dungeon-place-object --room=0x98 --id=0x0031 --x=20 --y=20 "
135 "--size=4 --rom=/tmp/oos-work.sfc --format=json",
136 "z3ed dungeon-place-object --room=0x98 --id=0x0031 --x=20 --y=20 "
137 "--size=4 --write --rom=/tmp/oos-work.sfc --format=json"};
138 }
else if (name ==
"dungeon-oracle-preflight") {
140 "Oracle ROM safety preflight: water-fill region/table, collision "
141 "maps, and optional required-room checks";
143 "z3ed dungeon-oracle-preflight --rom oos168.sfc --format=json",
144 "z3ed dungeon-oracle-preflight --rom oos168.sfc "
145 "--required-collision-rooms=0x25,0x27 --format=json",
146 "z3ed dungeon-oracle-preflight --rom oos168.sfc "
147 "--report=/tmp/preflight.json"};
148 }
else if (name ==
"dungeon-set-collision-tile") {
150 "Set one or more custom collision tiles in a dungeon room";
152 "z3ed dungeon-set-collision-tile --room=0xB8 "
153 "--tiles=\"10,5,0xB7;50,45,0xBA\" --rom=/tmp/oos-work.sfc "
155 "z3ed dungeon-set-collision-tile --room=0xB8 "
156 "--tiles=\"10,5,0xB7;50,45,0xBA\" --write "
157 "--rom=/tmp/oos-work.sfc --format=json"};
159 }
else if (name.find(
"overworld-") == 0) {
161 metadata.
description =
"Overworld inspection and editing";
162 if (name ==
"overworld-find-tile") {
164 "z3ed overworld-find-tile --tile=0x42 --format=json"};
166 }
else if (name.find(
"project-bundle-") == 0) {
169 if (name ==
"project-bundle-verify") {
170 metadata.
description =
"Project bundle verification";
172 "z3ed project-bundle-verify --project MyProject.yazeproj "
174 "z3ed project-bundle-verify --project project.yaze "
175 "--report report.json",
176 "z3ed project-bundle-verify --project MyProject.yazeproj "
177 "--check-rom-hash --format=json"};
178 }
else if (name ==
"project-bundle-pack") {
179 metadata.
description =
"Pack bundle into zip archive";
181 "z3ed project-bundle-pack --project MyProject.yazeproj "
182 "--out MyProject.zip --format=json",
183 "z3ed project-bundle-pack --project MyProject.yazeproj "
184 "--out MyProject.zip --overwrite"};
185 }
else if (name ==
"project-bundle-unpack") {
186 metadata.
description =
"Unpack zip archive into bundle";
188 "z3ed project-bundle-unpack --archive MyProject.zip "
189 "--out ./projects --format=json",
190 "z3ed project-bundle-unpack --archive MyProject.zip "
191 "--out ./projects --overwrite",
192 "z3ed project-bundle-unpack --archive MyProject.zip "
193 "--out ./projects --keep-partial-output",
194 "z3ed project-bundle-unpack --archive MyProject.zip "
195 "--out ./projects --dry-run"};
197 }
else if (name.find(
"rom-") == 0) {
199 metadata.
description =
"ROM inspection and validation";
200 if (name ==
"rom-info") {
201 metadata.
examples = {
"z3ed rom-info --rom=zelda3.sfc"};
202 }
else if (name ==
"rom-validate") {
203 metadata.
examples = {
"z3ed rom-validate --rom=zelda3.sfc"};
204 }
else if (name ==
"rom-doctor") {
205 metadata.
examples = {
"z3ed rom-doctor --rom=zelda3.sfc --format=json"};
206 }
else if (name ==
"rom-diff") {
208 "z3ed rom-diff --rom_a=base.sfc --rom_b=target.sfc"};
209 }
else if (name ==
"rom-compare") {
211 "z3ed rom-compare --rom=target.sfc --baseline=vanilla.sfc"};
213 }
else if (name.find(
"emulator-") == 0) {
215 metadata.
description =
"Emulator control and debugging";
217 if (name ==
"emulator-set-breakpoint") {
219 "z3ed emulator-set-breakpoint --address=0x83D7 --description='NMI "
222 }
else if (name.find(
"mesen-") == 0) {
224 metadata.
description =
"Mesen2 socket automation and introspection";
226 if (name ==
"mesen-state-capture") {
228 "z3ed mesen-state-capture --state=foo.state "
229 "--rom-file=Roms/oos168x.sfc --scenario=d6_room_88",
230 "z3ed mesen-state-capture --state=Roms/SaveStates/d6_room_88.state "
231 "--rom-file=Roms/oos168x.sfc --slot=1 "
232 "--states-dir=~/Library/Application\\ Support/Mesen2/SaveStates"};
233 }
else if (name ==
"mesen-state-hook") {
235 "z3ed mesen-state-hook --state=foo.state "
236 "--rom-file=Roms/oos168x.sfc --scenario=d6_room_88",
237 "z3ed mesen-state-hook --state=foo.state "
238 "--rom-file=Roms/oos168x.sfc --format=json"};
240 }
else if (name.find(
"gui-") == 0) {
244 }
else if (name.find(
"hex-") == 0) {
247 }
else if (name.find(
"palette-") == 0) {
250 }
else if (name.find(
"sprite-") == 0) {
253 }
else if (name.find(
"message-") == 0 || name.find(
"dialogue-") == 0) {
255 metadata.
description = name.find(
"message-") == 0 ?
"Message inspection"
256 :
"Dialogue inspection";
257 }
else if (name.find(
"music-") == 0) {
260 }
else if (name.find(
"oracle-") == 0) {
262 metadata.
description =
"Oracle-of-Secrets project tooling";
263 if (name ==
"oracle-menu-index") {
265 "z3ed oracle-menu-index --project=/path/to/oracle-of-secrets "
267 "z3ed oracle-menu-index --table=Menu_ItemCursorPositions "
268 "--missing-bins --format=json"};
269 }
else if (name ==
"oracle-menu-set-offset") {
271 "z3ed oracle-menu-set-offset "
272 "--asm=Menu/menu_select_item.asm "
273 "--table=Menu_ItemCursorPositions --index=0 --row=7 --col=2",
274 "z3ed oracle-menu-set-offset "
275 "--asm=Menu/menu_select_item.asm "
276 "--table=Menu_ItemCursorPositions --index=0 --row=7 --col=2 "
278 }
else if (name ==
"oracle-menu-validate") {
280 "z3ed oracle-menu-validate --project=/path/to/oracle-of-secrets",
281 "z3ed oracle-menu-validate --project=/path/to/oracle-of-secrets "
282 "--strict --max-row=31 --max-col=31"};
283 }
else if (name ==
"oracle-smoke-check") {
285 "z3ed oracle-smoke-check --rom oos168.sfc --format=json",
286 "z3ed oracle-smoke-check --rom oos168.sfc --strict-readiness "
288 "z3ed oracle-smoke-check --rom oos168.sfc "
289 "--min-d6-track-rooms=4 --format=json",
290 "z3ed oracle-smoke-check --rom oos168.sfc "
291 "--report=/tmp/smoke.json"};
293 }
else if (name ==
"simple-chat" || name ==
"chat") {
299 "z3ed simple-chat --rom=zelda3.sfc",
300 "z3ed simple-chat \"What dungeons exist?\" --rom=zelda3.sfc"};
301 }
else if (name.find(
"tools-") == 0) {
303 if (name ==
"tools-list") {
304 metadata.
description =
"List available test helper tools";
307 }
else if (name ==
"tools-harness-state") {
308 metadata.
description =
"Generate WRAM state for test harnesses";
310 "z3ed tools-harness-state --rom=zelda3.sfc --output=state.h"};
311 }
else if (name ==
"tools-extract-values") {
312 metadata.
description =
"Extract vanilla ROM values";
313 metadata.
examples = {
"z3ed tools-extract-values --rom=zelda3.sfc"};
314 }
else if (name ==
"tools-extract-golden") {
315 metadata.
description =
"Extract comprehensive golden data for testing";
317 "z3ed tools-extract-golden --rom=zelda3.sfc --output=golden.h"};
318 }
else if (name ==
"tools-patch-v3") {
319 metadata.
description =
"Create v3 ZSCustomOverworld patched ROM";
321 "z3ed tools-patch-v3 --rom=zelda3.sfc --output=patched.sfc"};
325 }
else if (name.find(
"test-") == 0) {
327 metadata.
description =
"Test discovery and execution";
328 if (name ==
"test-list") {
330 metadata.
examples = {
"z3ed test-list",
"z3ed test-list --format json"};
331 }
else if (name ==
"test-run") {
332 metadata.
examples = {
"z3ed test-run --label stable",
333 "z3ed test-run --label gui"};
334 }
else if (name ==
"test-status") {
336 metadata.
examples = {
"z3ed test-status --format json"};
344 if (!descriptor.summary.empty() &&
345 descriptor.summary !=
"Command summary not provided.") {
350 if (!descriptor.todo_reference.empty() &&
351 descriptor.todo_reference !=
"todo#unassigned") {
355 Register(std::move(handler), metadata);
363 return it->second.get();
367 auto alias_it =
aliases_.find(name);
369 auto handler_it =
handlers_.find(alias_it->second);
371 return handler_it->second.get();
379 const std::string& name)
const {
381 std::string canonical_name = name;
382 auto alias_it =
aliases_.find(name);
384 canonical_name = alias_it->second;
387 auto it =
metadata_.find(canonical_name);
388 return (it !=
metadata_.end()) ? &it->second :
nullptr;
392 const std::string& category)
const {
393 std::vector<std::string> result;
394 for (
const auto& [name, metadata] :
metadata_) {
395 if (metadata.category == category) {
396 result.push_back(name);
403 std::vector<std::string> categories;
404 for (
const auto& [_, metadata] :
metadata_) {
405 if (std::find(categories.begin(), categories.end(), metadata.category) ==
407 categories.push_back(metadata.category);
414 std::vector<std::string> result;
415 for (
const auto& [name, metadata] :
metadata_) {
416 if (metadata.available_to_agent) {
417 result.push_back(name);
424 std::ostringstream out;
425 out <<
"{\n \"commands\": [\n";
428 for (
const auto& [_, metadata] :
metadata_) {
434 out <<
" \"name\": \"" << EscapeJson(metadata.name) <<
"\",\n";
435 out <<
" \"category\": \"" << EscapeJson(metadata.category) <<
"\",\n";
436 out <<
" \"description\": \"" << EscapeJson(metadata.description)
438 out <<
" \"usage\": \"" << EscapeJson(metadata.usage) <<
"\",\n";
439 out <<
" \"available_to_agent\": "
440 << (metadata.available_to_agent ?
"true" :
"false") <<
",\n";
441 out <<
" \"requires_rom\": "
442 << (metadata.requires_rom ?
"true" :
"false") <<
",\n";
443 out <<
" \"requires_grpc\": "
444 << (metadata.requires_grpc ?
"true" :
"false") <<
",\n";
445 if (!metadata.todo_reference.empty()) {
446 out <<
" \"todo_reference\": \""
447 << EscapeJson(metadata.todo_reference) <<
"\",\n";
449 out <<
" \"todo_reference\": \"\",\n";
451 out <<
" \"aliases\": ";
452 AppendStringArray(out, metadata.aliases);
454 out <<
" \"examples\": ";
455 AppendStringArray(out, metadata.examples);
467 return absl::StrFormat(
"Command '%s' not found", name);
470 std::ostringstream help;
471 help <<
"\n\033[1;36m" << metadata->name <<
"\033[0m - "
472 << metadata->description <<
"\n\n";
473 help <<
"\033[1;33mUsage:\033[0m\n";
474 help <<
" " << metadata->usage <<
"\n\n";
476 if (!metadata->examples.empty()) {
477 help <<
"\033[1;33mExamples:\033[0m\n";
478 for (
const auto& example : metadata->examples) {
479 help <<
" " << example <<
"\n";
484 if (metadata->requires_rom) {
485 help <<
"\033[1;33mRequires:\033[0m ROM file (--rom=<path>)\n";
487 if (metadata->requires_grpc) {
488 help <<
"\033[1;33mRequires:\033[0m YAZE running with gRPC enabled\n";
491 if (!metadata->aliases.empty()) {
492 help <<
"\n\033[1;33mAliases:\033[0m "
493 << absl::StrJoin(metadata->aliases,
", ") <<
"\n";
500 const std::string& category)
const {
502 if (commands.empty()) {
503 return absl::StrFormat(
"No commands in category '%s'", category);
506 std::ostringstream help;
507 help <<
"\n\033[1;36m" << category <<
" commands:\033[0m\n\n";
509 for (
const auto& cmd : commands) {
512 help <<
" \033[1;33m" << cmd <<
"\033[0m\n";
513 help <<
" " << metadata->description <<
"\n";
514 if (!metadata->usage.empty()) {
515 help <<
" Usage: " << metadata->usage <<
"\n";
525 std::ostringstream help;
526 help <<
"\n\033[1;36mAll z3ed Commands:\033[0m\n\n";
529 for (
const auto& category : categories) {
537 const std::vector<std::string>& args,
539 std::string* captured_output) {
540 auto* handler =
Get(name);
542 return absl::NotFoundError(absl::StrFormat(
"Command '%s' not found", name));
545 const bool command_help_requested =
546 std::find(args.begin(), args.end(),
"--help") != args.end() ||
547 std::find(args.begin(), args.end(),
"-h") != args.end();
548 if (command_help_requested) {
550 if (captured_output !=
nullptr) {
551 *captured_output = help;
553 std::cout << help <<
"\n";
555 return absl::OkStatus();
558 absl::Status status = handler->Run(args, rom_context, captured_output);
562 if (!status.ok() && status.code() == absl::StatusCode::kInvalidArgument &&
565 if (captured_output !=
nullptr) {
566 if (!captured_output->empty()) {
567 captured_output->append(
"\n\n");
569 captured_output->append(help);
571 std::cout << help <<
"\n";
579 return Get(name) !=
nullptr;
The Rom class is used to load, save, and modify Rom data. This is a generic SNES ROM container and do...
Single source of truth for all z3ed commands.
void RegisterCliCommands()
std::map< std::string, std::string > aliases_
std::vector< std::string > GetCommandsInCategory(const std::string &category) const
Get all commands in a category.
static CommandRegistry & Instance()
std::string GenerateCategoryHelp(const std::string &category) const
Generate category help text.
std::string ExportFunctionSchemas() const
Export function schemas for AI tool calling (JSON)
const CommandMetadata * GetMetadata(const std::string &name) const
Get command metadata.
std::string GenerateCompleteHelp() const
Generate complete help text (all commands)
std::vector< std::string > GetAgentCommands() const
Get all commands available to AI agents.
std::map< std::string, CommandMetadata > metadata_
absl::Status Execute(const std::string &name, const std::vector< std::string > &args, Rom *rom_context=nullptr, std::string *captured_output=nullptr)
Execute a command by name.
std::map< std::string, std::unique_ptr< resources::CommandHandler > > handlers_
resources::CommandHandler * Get(const std::string &name) const
Get a command handler by name or alias.
bool HasCommand(const std::string &name) const
Check if command exists.
std::vector< std::string > GetCategories() const
Get all categories.
void Register(std::unique_ptr< resources::CommandHandler > handler, const CommandMetadata &metadata)
Register a command handler.
std::string GenerateHelp(const std::string &name) const
Generate help text for a command.
void RegisterHandlers(std::vector< std::unique_ptr< resources::CommandHandler > > handlers)
Register a set of command handlers (idempotent)
Base class for CLI command handlers.
std::string EscapeJson(const std::string &input)
void AppendStringArray(std::ostringstream &out, const std::vector< std::string > &values)
std::vector< std::unique_ptr< resources::CommandHandler > > CreateCliCommandHandlers()
Factory function to create all CLI-level command handlers.