@statechange/xano-cli 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +251 -0
- package/dist/auth.d.ts +22 -0
- package/dist/auth.d.ts.map +1 -0
- package/dist/auth.js +94 -0
- package/dist/auth.js.map +1 -0
- package/dist/commands/audit.d.ts +6 -0
- package/dist/commands/audit.d.ts.map +1 -0
- package/dist/commands/audit.js +316 -0
- package/dist/commands/audit.js.map +1 -0
- package/dist/commands/auth.d.ts +6 -0
- package/dist/commands/auth.d.ts.map +1 -0
- package/dist/commands/auth.js +179 -0
- package/dist/commands/auth.js.map +1 -0
- package/dist/commands/health.d.ts +6 -0
- package/dist/commands/health.d.ts.map +1 -0
- package/dist/commands/health.js +115 -0
- package/dist/commands/health.js.map +1 -0
- package/dist/commands/history.d.ts +6 -0
- package/dist/commands/history.d.ts.map +1 -0
- package/dist/commands/history.js +250 -0
- package/dist/commands/history.js.map +1 -0
- package/dist/commands/inventory.d.ts +6 -0
- package/dist/commands/inventory.d.ts.map +1 -0
- package/dist/commands/inventory.js +282 -0
- package/dist/commands/inventory.js.map +1 -0
- package/dist/commands/logs.d.ts +6 -0
- package/dist/commands/logs.d.ts.map +1 -0
- package/dist/commands/logs.js +411 -0
- package/dist/commands/logs.js.map +1 -0
- package/dist/commands/performance.d.ts +6 -0
- package/dist/commands/performance.d.ts.map +1 -0
- package/dist/commands/performance.js +520 -0
- package/dist/commands/performance.js.map +1 -0
- package/dist/commands/secure.d.ts +6 -0
- package/dist/commands/secure.d.ts.map +1 -0
- package/dist/commands/secure.js +52 -0
- package/dist/commands/secure.js.map +1 -0
- package/dist/commands/xanoscript.d.ts +6 -0
- package/dist/commands/xanoscript.d.ts.map +1 -0
- package/dist/commands/xanoscript.js +216 -0
- package/dist/commands/xanoscript.js.map +1 -0
- package/dist/commands/xray.d.ts +6 -0
- package/dist/commands/xray.d.ts.map +1 -0
- package/dist/commands/xray.js +194 -0
- package/dist/commands/xray.js.map +1 -0
- package/dist/format.d.ts +10 -0
- package/dist/format.d.ts.map +1 -0
- package/dist/format.js +59 -0
- package/dist/format.js.map +1 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +43 -0
- package/dist/index.js.map +1 -0
- package/dist/performance/load-analysis.d.ts +33 -0
- package/dist/performance/load-analysis.d.ts.map +1 -0
- package/dist/performance/load-analysis.js +290 -0
- package/dist/performance/load-analysis.js.map +1 -0
- package/dist/performance/stack-rollup.d.ts +57 -0
- package/dist/performance/stack-rollup.d.ts.map +1 -0
- package/dist/performance/stack-rollup.js +108 -0
- package/dist/performance/stack-rollup.js.map +1 -0
- package/dist/registry-client.d.ts +81 -0
- package/dist/registry-client.d.ts.map +1 -0
- package/dist/registry-client.js +333 -0
- package/dist/registry-client.js.map +1 -0
- package/dist/xano-client.d.ts +103 -0
- package/dist/xano-client.d.ts.map +1 -0
- package/dist/xano-client.js +399 -0
- package/dist/xano-client.js.map +1 -0
- package/package.json +49 -0
- package/skills/performance-analysis/SKILL.md +135 -0
- package/skills/xano-cli/SKILL.md +158 -0
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* XanoScript CLI Commands — Generation & conversion using private API
|
|
3
|
+
*/
|
|
4
|
+
import { makeClient } from "../registry-client.js";
|
|
5
|
+
import { readFileSync, writeFileSync, mkdirSync } from "fs";
|
|
6
|
+
import { join, resolve } from "path";
|
|
7
|
+
const stdOptions = (cmd) => cmd
|
|
8
|
+
.option("--instance <instance>", "Xano instance (e.g., xq1a-abcd-1234.xano.io)")
|
|
9
|
+
.option("--workspace <workspace>", "Workspace ID")
|
|
10
|
+
.option("--branch <branch>", "Branch ID", "0")
|
|
11
|
+
.option("--token <token>", "Xano API token")
|
|
12
|
+
.option("--api-key <key>", "StateChange API key (overrides saved key)");
|
|
13
|
+
// Map CLI type names to XanoScript kind values
|
|
14
|
+
const TYPE_TO_KIND = {
|
|
15
|
+
function: "schema:function",
|
|
16
|
+
table: "schema:table",
|
|
17
|
+
api: "schema:query",
|
|
18
|
+
task: "schema:task",
|
|
19
|
+
trigger: "schema:trigger",
|
|
20
|
+
mcp_server: "schema:mcp_server",
|
|
21
|
+
addon: "schema:addon",
|
|
22
|
+
middleware: "schema:middleware",
|
|
23
|
+
};
|
|
24
|
+
async function fetchObjectsOfType(client, workspace, branchId, type) {
|
|
25
|
+
switch (type) {
|
|
26
|
+
case "function": {
|
|
27
|
+
const { functions } = await client.getFunctions(workspace, branchId);
|
|
28
|
+
return { items: functions, nameField: "name" };
|
|
29
|
+
}
|
|
30
|
+
case "table": {
|
|
31
|
+
const sink = await client.getWorkspaceSink(workspace);
|
|
32
|
+
return { items: sink.dbos ?? [], nameField: "name" };
|
|
33
|
+
}
|
|
34
|
+
case "api": {
|
|
35
|
+
const { queries } = await client.getAPIAppsAndQueries(workspace, branchId);
|
|
36
|
+
return { items: queries, nameField: "name" };
|
|
37
|
+
}
|
|
38
|
+
case "task": {
|
|
39
|
+
const tasks = await client.getTasks(workspace, branchId);
|
|
40
|
+
return { items: tasks, nameField: "name" };
|
|
41
|
+
}
|
|
42
|
+
case "trigger": {
|
|
43
|
+
const triggers = await client.getTriggers(workspace, branchId);
|
|
44
|
+
return { items: triggers, nameField: "name" };
|
|
45
|
+
}
|
|
46
|
+
case "mcp_server": {
|
|
47
|
+
const servers = await client.getMCPServers(workspace, branchId);
|
|
48
|
+
return { items: servers, nameField: "name" };
|
|
49
|
+
}
|
|
50
|
+
case "addon": {
|
|
51
|
+
const addons = await client.getAddons(workspace, branchId);
|
|
52
|
+
return { items: addons, nameField: "name" };
|
|
53
|
+
}
|
|
54
|
+
case "middleware": {
|
|
55
|
+
const mw = await client.getMiddleware(workspace, branchId);
|
|
56
|
+
return { items: mw, nameField: "name" };
|
|
57
|
+
}
|
|
58
|
+
default:
|
|
59
|
+
throw new Error(`Unknown type: ${type}. Valid types: ${Object.keys(TYPE_TO_KIND).join(", ")}`);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
function resolveKind(type, data) {
|
|
63
|
+
// Special handling for triggers based on obj_type
|
|
64
|
+
if (type === "trigger" && data) {
|
|
65
|
+
const objType = data.obj_type;
|
|
66
|
+
if (objType === "toolset")
|
|
67
|
+
return "schema:mcp_server_trigger";
|
|
68
|
+
if (objType === "database")
|
|
69
|
+
return "schema:table_trigger";
|
|
70
|
+
}
|
|
71
|
+
// Special handling for database type
|
|
72
|
+
if (type === "database")
|
|
73
|
+
return "schema:table";
|
|
74
|
+
const kind = TYPE_TO_KIND[type];
|
|
75
|
+
if (!kind) {
|
|
76
|
+
throw new Error(`Unknown type: ${type}. Valid types: ${Object.keys(TYPE_TO_KIND).join(", ")}`);
|
|
77
|
+
}
|
|
78
|
+
return kind;
|
|
79
|
+
}
|
|
80
|
+
function sanitizeFilename(name) {
|
|
81
|
+
return name.replace(/[^a-zA-Z0-9_\-. ]/g, "_");
|
|
82
|
+
}
|
|
83
|
+
export function createXanoScriptCommand(program) {
|
|
84
|
+
const xs = program.command("xanoscript").description("XanoScript generation & conversion");
|
|
85
|
+
stdOptions(xs
|
|
86
|
+
.command("generate")
|
|
87
|
+
.description("Generate XanoScript for a single object")
|
|
88
|
+
.argument("<type>", `Object type (${Object.keys(TYPE_TO_KIND).join(", ")})`)
|
|
89
|
+
.argument("<id>", "Object ID")).action(async (type, id, options) => {
|
|
90
|
+
try {
|
|
91
|
+
const { client, workspace, branchId } = await makeClient(options);
|
|
92
|
+
const objectId = parseInt(id);
|
|
93
|
+
// Fetch via sink and pluck by ID
|
|
94
|
+
const { items } = await fetchObjectsOfType(client, workspace, branchId, type);
|
|
95
|
+
const data = items.find((item) => item.id === objectId);
|
|
96
|
+
if (!data) {
|
|
97
|
+
console.error(`Error: ${type} with ID ${objectId} not found`);
|
|
98
|
+
process.exit(1);
|
|
99
|
+
}
|
|
100
|
+
const kind = resolveKind(type, data);
|
|
101
|
+
const result = await client.generateXanoScript(workspace, data, kind);
|
|
102
|
+
if (result.status === "success" && result.payload?.output) {
|
|
103
|
+
console.log(result.payload.output);
|
|
104
|
+
}
|
|
105
|
+
else if (result.payload?.message) {
|
|
106
|
+
if (!result.payload.doIgnore) {
|
|
107
|
+
console.error("Error:", result.payload.message);
|
|
108
|
+
process.exit(1);
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
console.error("Error: No XanoScript output returned");
|
|
113
|
+
process.exit(1);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
catch (error) {
|
|
117
|
+
console.error("Error:", error.message);
|
|
118
|
+
process.exit(1);
|
|
119
|
+
}
|
|
120
|
+
});
|
|
121
|
+
stdOptions(xs
|
|
122
|
+
.command("export-all")
|
|
123
|
+
.description("Bulk export all objects of a type to .xs files")
|
|
124
|
+
.option("--type <type>", `Object type (${Object.keys(TYPE_TO_KIND).join(", ")})`)
|
|
125
|
+
.option("--output-dir <dir>", "Output directory", "./xanoscript")).action(async (options) => {
|
|
126
|
+
try {
|
|
127
|
+
if (!options.type) {
|
|
128
|
+
console.error(`Error: --type required. Valid types: ${Object.keys(TYPE_TO_KIND).join(", ")}`);
|
|
129
|
+
process.exit(1);
|
|
130
|
+
}
|
|
131
|
+
const { client, workspace, branchId } = await makeClient(options);
|
|
132
|
+
const type = options.type;
|
|
133
|
+
const outputDir = resolve(options.outputDir, type);
|
|
134
|
+
mkdirSync(outputDir, { recursive: true });
|
|
135
|
+
console.log(`Fetching ${type} objects from workspace ${workspace}...\n`);
|
|
136
|
+
const { items, nameField } = await fetchObjectsOfType(client, workspace, branchId, type);
|
|
137
|
+
console.log(`Found ${items.length} ${type}(s). Generating XanoScript...\n`);
|
|
138
|
+
let success = 0;
|
|
139
|
+
let skipped = 0;
|
|
140
|
+
let errors = 0;
|
|
141
|
+
for (let i = 0; i < items.length; i++) {
|
|
142
|
+
const item = items[i];
|
|
143
|
+
const name = item[nameField] || `unnamed_${item.id}`;
|
|
144
|
+
process.stdout.write(` [${i + 1}/${items.length}] ${name}...`);
|
|
145
|
+
// Sink data already contains full objects
|
|
146
|
+
const kind = resolveKind(type, item);
|
|
147
|
+
const result = await client.generateXanoScript(workspace, item, kind);
|
|
148
|
+
if (result.status === "success" && result.payload?.output) {
|
|
149
|
+
const filename = `${sanitizeFilename(name)}.xs`;
|
|
150
|
+
const filepath = join(outputDir, filename);
|
|
151
|
+
writeFileSync(filepath, result.payload.output, "utf-8");
|
|
152
|
+
console.log(` ✅`);
|
|
153
|
+
success++;
|
|
154
|
+
}
|
|
155
|
+
else if (result.payload?.doIgnore) {
|
|
156
|
+
console.log(` ⏭️ skipped`);
|
|
157
|
+
skipped++;
|
|
158
|
+
}
|
|
159
|
+
else {
|
|
160
|
+
console.log(` ❌ ${result.payload?.message || "unknown error"}`);
|
|
161
|
+
errors++;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
console.log(`\nDone: ${success} exported, ${skipped} skipped, ${errors} errors`);
|
|
165
|
+
if (success > 0) {
|
|
166
|
+
console.log(`Output: ${outputDir}/`);
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
catch (error) {
|
|
170
|
+
console.error("Error:", error.message);
|
|
171
|
+
process.exit(1);
|
|
172
|
+
}
|
|
173
|
+
});
|
|
174
|
+
stdOptions(xs
|
|
175
|
+
.command("convert")
|
|
176
|
+
.description("Convert XanoScript (.xs file or stdin) back to Xano JSON")
|
|
177
|
+
.argument("[file]", "Path to .xs file (reads from stdin if omitted)")).action(async (file, options) => {
|
|
178
|
+
try {
|
|
179
|
+
const { client, workspace } = await makeClient(options);
|
|
180
|
+
let script;
|
|
181
|
+
if (file) {
|
|
182
|
+
script = readFileSync(file, "utf-8");
|
|
183
|
+
}
|
|
184
|
+
else {
|
|
185
|
+
// Read from stdin
|
|
186
|
+
const chunks = [];
|
|
187
|
+
for await (const chunk of process.stdin) {
|
|
188
|
+
chunks.push(chunk);
|
|
189
|
+
}
|
|
190
|
+
script = Buffer.concat(chunks).toString("utf-8");
|
|
191
|
+
}
|
|
192
|
+
if (!script.trim()) {
|
|
193
|
+
console.error("Error: No XanoScript input provided");
|
|
194
|
+
process.exit(1);
|
|
195
|
+
}
|
|
196
|
+
const result = await client.convertXanoScript(workspace, script);
|
|
197
|
+
if (result.status === "success") {
|
|
198
|
+
console.log(JSON.stringify(result.payload, null, 2));
|
|
199
|
+
}
|
|
200
|
+
else if (result.payload?.message) {
|
|
201
|
+
console.error("Error:", result.payload.message);
|
|
202
|
+
process.exit(1);
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
console.error("Error: Conversion failed");
|
|
206
|
+
process.exit(1);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
catch (error) {
|
|
210
|
+
console.error("Error:", error.message);
|
|
211
|
+
process.exit(1);
|
|
212
|
+
}
|
|
213
|
+
});
|
|
214
|
+
return xs;
|
|
215
|
+
}
|
|
216
|
+
//# sourceMappingURL=xanoscript.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"xanoscript.js","sourceRoot":"","sources":["../../src/commands/xanoscript.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"xray.d.ts","sourceRoot":"","sources":["../../src/commands/xray.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAUpC,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,OAAO,WA+MjD"}
|
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* X-Ray CLI Commands
|
|
3
|
+
*/
|
|
4
|
+
import { buildStepListFromXray, getFunctionDependencies, getStepWarnings, analyzeStack, } from "@statechange/xano-xray";
|
|
5
|
+
import { makeClient } from "../registry-client.js";
|
|
6
|
+
import { FORMAT_HELP, parseFormat, outputFormatted } from "../format.js";
|
|
7
|
+
export function createXRayCommand(program) {
|
|
8
|
+
const xray = program.command("xray").description("X-Ray analysis commands");
|
|
9
|
+
xray
|
|
10
|
+
.command("function")
|
|
11
|
+
.description("Analyze a function's X-Ray data")
|
|
12
|
+
.requiredOption("--id <id>", "Function ID")
|
|
13
|
+
.option("--instance <instance>", "Xano instance (e.g., app.xano.com)")
|
|
14
|
+
.option("--workspace <workspace>", "Workspace ID")
|
|
15
|
+
.option("--branch <branch>", "Branch ID", "0")
|
|
16
|
+
.option("--token <token>", "Xano API token")
|
|
17
|
+
.option("--api-key <key>", "StateChange API key (overrides saved key)")
|
|
18
|
+
.option("--format <format>", FORMAT_HELP, "table")
|
|
19
|
+
.action(async (options) => {
|
|
20
|
+
const { client, workspace: workspaceId, branchId } = await makeClient(options);
|
|
21
|
+
const functionId = parseInt(options.id);
|
|
22
|
+
const format = parseFormat(options.format);
|
|
23
|
+
try {
|
|
24
|
+
const log = format === "table" ? console.log.bind(console) : console.error.bind(console);
|
|
25
|
+
log(`Fetching function ${functionId}...`);
|
|
26
|
+
const func = await client.getFunction(functionId, workspaceId ?? undefined, branchId);
|
|
27
|
+
// Build step list
|
|
28
|
+
const steps = buildStepListFromXray(func);
|
|
29
|
+
// Analyze for warnings
|
|
30
|
+
const warnings = [];
|
|
31
|
+
steps.forEach((step) => {
|
|
32
|
+
const stepWarnings = getStepWarnings(step, func);
|
|
33
|
+
if (stepWarnings.length > 0) {
|
|
34
|
+
warnings.push({ step, warnings: stepWarnings });
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
// Analyze stack
|
|
38
|
+
const analysis = analyzeStack(func);
|
|
39
|
+
// Function dependencies
|
|
40
|
+
let dependencies = [];
|
|
41
|
+
if (workspaceId) {
|
|
42
|
+
try {
|
|
43
|
+
const { functions } = await client.getFunctions(workspaceId, branchId);
|
|
44
|
+
const depIds = getFunctionDependencies(func, { functions });
|
|
45
|
+
dependencies = Array.from(depIds).map((depId) => {
|
|
46
|
+
const dep = functions.find((f) => f.id === depId);
|
|
47
|
+
return { id: depId, name: dep?.name || `Function ${depId}` };
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
catch (e) {
|
|
51
|
+
// Silently fail if we can't get dependencies
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
const data = {
|
|
55
|
+
id: func.id,
|
|
56
|
+
name: func.name,
|
|
57
|
+
description: func.description || null,
|
|
58
|
+
total_steps: steps.length,
|
|
59
|
+
steps: steps.map((step) => ({
|
|
60
|
+
position: step.position2 || step.position,
|
|
61
|
+
name: step.name,
|
|
62
|
+
description: step.description || null,
|
|
63
|
+
disabled: !!(step.disabled || step.inheritedDisabled),
|
|
64
|
+
})),
|
|
65
|
+
warnings: warnings.map(({ step, warnings: sw }) => ({
|
|
66
|
+
position: step.position2 || step.position,
|
|
67
|
+
step_name: step.name,
|
|
68
|
+
issues: sw.map((w) => w.description),
|
|
69
|
+
})),
|
|
70
|
+
stack_analysis: {
|
|
71
|
+
has_errors: analysis.isError || false,
|
|
72
|
+
has_warnings: analysis.isWarning || false,
|
|
73
|
+
},
|
|
74
|
+
dependencies,
|
|
75
|
+
};
|
|
76
|
+
if (outputFormatted(format, data))
|
|
77
|
+
return;
|
|
78
|
+
// Table output
|
|
79
|
+
console.log(`\nFunction: ${func.name}`);
|
|
80
|
+
if (func.description) {
|
|
81
|
+
console.log(`Description: ${func.description}`);
|
|
82
|
+
}
|
|
83
|
+
console.log(`ID: ${func.id}`);
|
|
84
|
+
console.log(`\nTotal steps: ${steps.length}`);
|
|
85
|
+
console.log("\nStep hierarchy:");
|
|
86
|
+
steps.forEach((step) => {
|
|
87
|
+
const indent = " ".repeat(step.position2?.split(".").length || 0);
|
|
88
|
+
const status = step.disabled || step.inheritedDisabled ? "[DISABLED] " : "";
|
|
89
|
+
console.log(`${indent}${status}${step.position2 || step.position}: ${step.name}${step.description ? ` - ${step.description}` : ""}`);
|
|
90
|
+
});
|
|
91
|
+
if (warnings.length > 0) {
|
|
92
|
+
console.log("\nPerformance Warnings:");
|
|
93
|
+
warnings.forEach(({ step, warnings: stepWarnings }) => {
|
|
94
|
+
console.log(`\n ${step.position2 || step.position}: ${step.name}`);
|
|
95
|
+
stepWarnings.forEach((w) => {
|
|
96
|
+
console.log(` - ${w.description}`);
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
if (analysis.isError || analysis.isWarning) {
|
|
101
|
+
console.log("\nStack Analysis:");
|
|
102
|
+
if (analysis.isError)
|
|
103
|
+
console.log(" - Has errors");
|
|
104
|
+
if (analysis.isWarning)
|
|
105
|
+
console.log(" - Has warnings");
|
|
106
|
+
}
|
|
107
|
+
if (dependencies.length > 0) {
|
|
108
|
+
console.log("\nFunction Dependencies:");
|
|
109
|
+
dependencies.forEach((dep) => {
|
|
110
|
+
console.log(` - ${dep.name} (ID: ${dep.id})`);
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
catch (error) {
|
|
115
|
+
console.error("Error:", error.message);
|
|
116
|
+
process.exit(1);
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
xray
|
|
120
|
+
.command("scan-workspace")
|
|
121
|
+
.description("Scan all functions in workspace for X-Ray issues")
|
|
122
|
+
.option("--instance <instance>", "Xano instance (e.g., app.xano.com)")
|
|
123
|
+
.option("--workspace <workspace>", "Workspace ID")
|
|
124
|
+
.option("--branch <branch>", "Branch ID", "0")
|
|
125
|
+
.option("--token <token>", "Xano API token")
|
|
126
|
+
.option("--api-key <key>", "StateChange API key (overrides saved key)")
|
|
127
|
+
.option("--include-warnings", "Include warnings in output")
|
|
128
|
+
.option("--format <format>", FORMAT_HELP, "table")
|
|
129
|
+
.action(async (options) => {
|
|
130
|
+
const { client, workspace, branchId } = await makeClient(options);
|
|
131
|
+
const format = parseFormat(options.format);
|
|
132
|
+
try {
|
|
133
|
+
const log = format === "table" ? console.log.bind(console) : console.error.bind(console);
|
|
134
|
+
log(`Scanning workspace ${workspace}...\n`);
|
|
135
|
+
const { functions } = await client.getFunctions(workspace, branchId);
|
|
136
|
+
log(`Found ${functions.length} functions\n`);
|
|
137
|
+
const errors = [];
|
|
138
|
+
const warnings = [];
|
|
139
|
+
for (const func of functions) {
|
|
140
|
+
const analysis = analyzeStack(func);
|
|
141
|
+
if (analysis.isError) {
|
|
142
|
+
errors.push({ function: func, analysis });
|
|
143
|
+
}
|
|
144
|
+
if (options.includeWarnings && analysis.isWarning) {
|
|
145
|
+
warnings.push({ function: func, analysis });
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
const data = {
|
|
149
|
+
total_functions_scanned: functions.length,
|
|
150
|
+
errors: errors.map(({ function: f }) => ({
|
|
151
|
+
id: f.id,
|
|
152
|
+
name: f.name,
|
|
153
|
+
description: f.description || null,
|
|
154
|
+
})),
|
|
155
|
+
warnings: options.includeWarnings
|
|
156
|
+
? warnings.map(({ function: f }) => ({
|
|
157
|
+
id: f.id,
|
|
158
|
+
name: f.name,
|
|
159
|
+
description: f.description || null,
|
|
160
|
+
}))
|
|
161
|
+
: undefined,
|
|
162
|
+
};
|
|
163
|
+
if (outputFormatted(format, data))
|
|
164
|
+
return;
|
|
165
|
+
if (errors.length > 0) {
|
|
166
|
+
console.log(`Functions with errors (${errors.length}):\n`);
|
|
167
|
+
errors.forEach(({ function: func }) => {
|
|
168
|
+
console.log(` - ${func.name} (ID: ${func.id})`);
|
|
169
|
+
if (func.description)
|
|
170
|
+
console.log(` ${func.description}`);
|
|
171
|
+
});
|
|
172
|
+
console.log();
|
|
173
|
+
}
|
|
174
|
+
if (warnings.length > 0) {
|
|
175
|
+
console.log(`Functions with warnings (${warnings.length}):\n`);
|
|
176
|
+
warnings.forEach(({ function: func }) => {
|
|
177
|
+
console.log(` - ${func.name} (ID: ${func.id})`);
|
|
178
|
+
if (func.description)
|
|
179
|
+
console.log(` ${func.description}`);
|
|
180
|
+
});
|
|
181
|
+
console.log();
|
|
182
|
+
}
|
|
183
|
+
if (errors.length === 0 && warnings.length === 0) {
|
|
184
|
+
console.log("No issues found!");
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
catch (error) {
|
|
188
|
+
console.error("Error:", error.message);
|
|
189
|
+
process.exit(1);
|
|
190
|
+
}
|
|
191
|
+
});
|
|
192
|
+
return xray;
|
|
193
|
+
}
|
|
194
|
+
//# sourceMappingURL=xray.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"xray.js","sourceRoot":"","sources":["../../src/commands/xray.ts"],"names":[],"mappings":"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"}
|
package/dist/format.d.ts
ADDED
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared output formatting utilities
|
|
3
|
+
*/
|
|
4
|
+
export type OutputFormat = "table" | "json" | "yaml";
|
|
5
|
+
export declare const FORMAT_HELP = "Output format: table (human-readable), json, or yaml (recommended for AI/LLM consumption)";
|
|
6
|
+
export declare function parseFormat(value: string): OutputFormat;
|
|
7
|
+
export declare function toYaml(obj: any, indent?: number): string;
|
|
8
|
+
/** Output data in the requested format. For json/yaml, prints to stdout. Returns true if handled. */
|
|
9
|
+
export declare function outputFormatted(format: OutputFormat, data: any): boolean;
|
|
10
|
+
//# sourceMappingURL=format.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"format.d.ts","sourceRoot":"","sources":["../src/format.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,MAAM,MAAM,YAAY,GAAG,OAAO,GAAG,MAAM,GAAG,MAAM,CAAC;AAErD,eAAO,MAAM,WAAW,8FAA8F,CAAC;AAEvH,wBAAgB,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,YAAY,CAIvD;AAED,wBAAgB,MAAM,CAAC,GAAG,EAAE,GAAG,EAAE,MAAM,SAAI,GAAG,MAAM,CA+BnD;AAED,qGAAqG;AACrG,wBAAgB,eAAe,CAAC,MAAM,EAAE,YAAY,EAAE,IAAI,EAAE,GAAG,GAAG,OAAO,CAUxE"}
|
package/dist/format.js
ADDED
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared output formatting utilities
|
|
3
|
+
*/
|
|
4
|
+
export const FORMAT_HELP = "Output format: table (human-readable), json, or yaml (recommended for AI/LLM consumption)";
|
|
5
|
+
export function parseFormat(value) {
|
|
6
|
+
if (value === "json" || value === "yaml" || value === "table")
|
|
7
|
+
return value;
|
|
8
|
+
console.error(`Error: Invalid format "${value}". Use table, json, or yaml.`);
|
|
9
|
+
process.exit(1);
|
|
10
|
+
}
|
|
11
|
+
export function toYaml(obj, indent = 0) {
|
|
12
|
+
const pad = " ".repeat(indent);
|
|
13
|
+
if (obj === null || obj === undefined)
|
|
14
|
+
return `${pad}null`;
|
|
15
|
+
if (typeof obj === "string") {
|
|
16
|
+
if (obj.includes("\n") || obj.includes(": ") || obj.includes("#") || obj.includes("'") || /^\d/.test(obj) || obj === "" || obj === "true" || obj === "false" || obj === "null") {
|
|
17
|
+
return `${pad}"${obj.replace(/\\/g, '\\\\').replace(/"/g, '\\"')}"`;
|
|
18
|
+
}
|
|
19
|
+
return `${pad}${obj}`;
|
|
20
|
+
}
|
|
21
|
+
if (typeof obj === "number" || typeof obj === "boolean")
|
|
22
|
+
return `${pad}${obj}`;
|
|
23
|
+
if (Array.isArray(obj)) {
|
|
24
|
+
if (obj.length === 0)
|
|
25
|
+
return `${pad}[]`;
|
|
26
|
+
return obj.map((item) => {
|
|
27
|
+
if (typeof item === "object" && item !== null) {
|
|
28
|
+
const inner = toYaml(item, indent + 1).trimStart();
|
|
29
|
+
return `${pad}- ${inner}`;
|
|
30
|
+
}
|
|
31
|
+
return `${pad}- ${toYaml(item, 0).trimStart()}`;
|
|
32
|
+
}).join("\n");
|
|
33
|
+
}
|
|
34
|
+
if (typeof obj === "object") {
|
|
35
|
+
const entries = Object.entries(obj).filter(([, v]) => v !== undefined);
|
|
36
|
+
if (entries.length === 0)
|
|
37
|
+
return `${pad}{}`;
|
|
38
|
+
return entries.map(([key, value]) => {
|
|
39
|
+
if (typeof value === "object" && value !== null) {
|
|
40
|
+
return `${pad}${key}:\n${toYaml(value, indent + 1)}`;
|
|
41
|
+
}
|
|
42
|
+
return `${pad}${key}: ${toYaml(value, 0).trimStart()}`;
|
|
43
|
+
}).join("\n");
|
|
44
|
+
}
|
|
45
|
+
return `${pad}${String(obj)}`;
|
|
46
|
+
}
|
|
47
|
+
/** Output data in the requested format. For json/yaml, prints to stdout. Returns true if handled. */
|
|
48
|
+
export function outputFormatted(format, data) {
|
|
49
|
+
if (format === "json") {
|
|
50
|
+
console.log(JSON.stringify(data, null, 2));
|
|
51
|
+
return true;
|
|
52
|
+
}
|
|
53
|
+
if (format === "yaml") {
|
|
54
|
+
console.log(toYaml(data));
|
|
55
|
+
return true;
|
|
56
|
+
}
|
|
57
|
+
return false;
|
|
58
|
+
}
|
|
59
|
+
//# sourceMappingURL=format.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"format.js","sourceRoot":"","sources":["../src/format.ts"],"names":[],"mappings":"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"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA;;;GAGG"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* StateChange CLI for Xano
|
|
4
|
+
* Access private Xano APIs and performance insights
|
|
5
|
+
*/
|
|
6
|
+
import { Command } from "commander";
|
|
7
|
+
import { createXRayCommand } from "./commands/xray.js";
|
|
8
|
+
import { createPerformanceCommand } from "./commands/performance.js";
|
|
9
|
+
import { createAuditCommand } from "./commands/audit.js";
|
|
10
|
+
import { createSecureCommand } from "./commands/secure.js";
|
|
11
|
+
import { createAuthCommand } from "./commands/auth.js";
|
|
12
|
+
import { createInventoryCommand } from "./commands/inventory.js";
|
|
13
|
+
import { createHealthCommand } from "./commands/health.js";
|
|
14
|
+
import { createHistoryCommand } from "./commands/history.js";
|
|
15
|
+
import { createXanoScriptCommand } from "./commands/xanoscript.js";
|
|
16
|
+
import { createLogsCommand } from "./commands/logs.js";
|
|
17
|
+
import { flushSinkCache } from "./xano-client.js";
|
|
18
|
+
const program = new Command();
|
|
19
|
+
program
|
|
20
|
+
.name("sc-xano")
|
|
21
|
+
.description("StateChange CLI for Xano - workspace management, performance analysis, and operational insights")
|
|
22
|
+
.version("0.2.0");
|
|
23
|
+
// Add subcommands
|
|
24
|
+
createAuthCommand(program);
|
|
25
|
+
createXRayCommand(program);
|
|
26
|
+
createPerformanceCommand(program);
|
|
27
|
+
createAuditCommand(program);
|
|
28
|
+
createSecureCommand(program);
|
|
29
|
+
createInventoryCommand(program);
|
|
30
|
+
createHealthCommand(program);
|
|
31
|
+
createHistoryCommand(program);
|
|
32
|
+
createXanoScriptCommand(program);
|
|
33
|
+
createLogsCommand(program);
|
|
34
|
+
// Utility: flush cached sink data
|
|
35
|
+
program
|
|
36
|
+
.command("flush")
|
|
37
|
+
.description("Flush cached sink data (forces fresh API calls)")
|
|
38
|
+
.action(() => {
|
|
39
|
+
flushSinkCache();
|
|
40
|
+
console.log("Cache flushed.");
|
|
41
|
+
});
|
|
42
|
+
program.parse(process.argv);
|
|
43
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA;;;GAGG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,wBAAwB,EAAE,MAAM,2BAA2B,CAAC;AACrE,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,mBAAmB,EAAE,MAAM,sBAAsB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,sBAAsB,EAAE,MAAM,yBAAyB,CAAC;AACjE,OAAO,EAAE,mBAAmB,EAAE,MAAM,sBAAsB,CAAC;AAC3D,OAAO,EAAE,oBAAoB,EAAE,MAAM,uBAAuB,CAAC;AAC7D,OAAO,EAAE,uBAAuB,EAAE,MAAM,0BAA0B,CAAC;AACnE,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAElD,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,SAAS,CAAC;KACf,WAAW,CAAC,iGAAiG,CAAC;KAC9G,OAAO,CAAC,OAAO,CAAC,CAAC;AAEpB,kBAAkB;AAClB,iBAAiB,CAAC,OAAO,CAAC,CAAC;AAC3B,iBAAiB,CAAC,OAAO,CAAC,CAAC;AAC3B,wBAAwB,CAAC,OAAO,CAAC,CAAC;AAClC,kBAAkB,CAAC,OAAO,CAAC,CAAC;AAC5B,mBAAmB,CAAC,OAAO,CAAC,CAAC;AAC7B,sBAAsB,CAAC,OAAO,CAAC,CAAC;AAChC,mBAAmB,CAAC,OAAO,CAAC,CAAC;AAC7B,oBAAoB,CAAC,OAAO,CAAC,CAAC;AAC9B,uBAAuB,CAAC,OAAO,CAAC,CAAC;AACjC,iBAAiB,CAAC,OAAO,CAAC,CAAC;AAE3B,kCAAkC;AAClC,OAAO;KACJ,OAAO,CAAC,OAAO,CAAC;KAChB,WAAW,CAAC,iDAAiD,CAAC;KAC9D,MAAM,CAAC,GAAG,EAAE;IACX,cAAc,EAAE,CAAC;IACjB,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;AAChC,CAAC,CAAC,CAAC;AAEL,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Performance Load Analysis - Portable from src/workers/performance.ts
|
|
3
|
+
*/
|
|
4
|
+
import { XanoClient } from "../xano-client.js";
|
|
5
|
+
export interface LoadAnalysisRequest {
|
|
6
|
+
instance: string;
|
|
7
|
+
workspace: number;
|
|
8
|
+
branch_id: number;
|
|
9
|
+
lookBack: number;
|
|
10
|
+
apps?: any[];
|
|
11
|
+
queries?: any[];
|
|
12
|
+
}
|
|
13
|
+
export interface RequestSummary {
|
|
14
|
+
totalDuration: number;
|
|
15
|
+
totalRequests: number;
|
|
16
|
+
avgDuration: number;
|
|
17
|
+
type?: string;
|
|
18
|
+
name?: string;
|
|
19
|
+
description?: string;
|
|
20
|
+
verb?: string;
|
|
21
|
+
status?: number;
|
|
22
|
+
objectId?: number | string;
|
|
23
|
+
}
|
|
24
|
+
export interface HistoryAnalysis {
|
|
25
|
+
date: number;
|
|
26
|
+
lookBack: number;
|
|
27
|
+
requestSummary: Record<string, RequestSummary>;
|
|
28
|
+
totals: RequestSummary;
|
|
29
|
+
reportTime: number;
|
|
30
|
+
reportDuration: number;
|
|
31
|
+
}
|
|
32
|
+
export declare function generateLoadAnalysis(client: XanoClient, options: LoadAnalysisRequest): Promise<HistoryAnalysis>;
|
|
33
|
+
//# sourceMappingURL=load-analysis.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"load-analysis.d.ts","sourceRoot":"","sources":["../../src/performance/load-analysis.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAE/C,MAAM,WAAW,mBAAmB;IAClC,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;IACjB,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;IACb,OAAO,CAAC,EAAE,GAAG,EAAE,CAAC;CACjB;AAED,MAAM,WAAW,cAAc;IAC7B,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,MAAM,CAAC;IACtB,WAAW,EAAE,MAAM,CAAC;IACpB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,CAAC;CAC5B;AAED,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,cAAc,EAAE,MAAM,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC;IAC/C,MAAM,EAAE,cAAc,CAAC;IACvB,UAAU,EAAE,MAAM,CAAC;IACnB,cAAc,EAAE,MAAM,CAAC;CACxB;AAED,wBAAsB,oBAAoB,CACxC,MAAM,EAAE,UAAU,EAClB,OAAO,EAAE,mBAAmB,GAC3B,OAAO,CAAC,eAAe,CAAC,CAmV1B"}
|