@maind-dev/cli 0.1.0 → 0.3.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/build/ci-report.js +146 -0
- package/build/ci-report.js.map +1 -0
- package/build/ci-result.js +145 -0
- package/build/ci-result.js.map +1 -0
- package/build/index.js +29 -1
- package/build/index.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
// `maind ci-report` — report a CI failure as a draft-lesson candidate (ADR-207
|
|
2
|
+
// Phase 2, INGESTION). An agent running in the CI runner authors a lesson
|
|
3
|
+
// candidate (frontmatter + body) capturing the gotcha + fix; this command reads
|
|
4
|
+
// that file, attaches CI provenance from the runner env, and POSTs it to the
|
|
5
|
+
// maind ingest endpoint with the CI key. maind files it in the maintainer review
|
|
6
|
+
// inbox — never auto-published.
|
|
7
|
+
//
|
|
8
|
+
// Fail-open by contract: a failing report must never further break the build.
|
|
9
|
+
// Any error → stderr warning, exit 0. (This already runs in an on-failure step.)
|
|
10
|
+
import { createHash } from "node:crypto";
|
|
11
|
+
import { readFileSync } from "node:fs";
|
|
12
|
+
import { MissingApiKeyError, resolveAuth } from "./auth.js";
|
|
13
|
+
const DEFAULT_REPORT_URL = "https://app.maind.dev/api/ci/report";
|
|
14
|
+
export function parseCiReportArgs(argv) {
|
|
15
|
+
const get = (name) => {
|
|
16
|
+
const i = argv.indexOf(name);
|
|
17
|
+
if (i === -1 || i + 1 >= argv.length)
|
|
18
|
+
return undefined;
|
|
19
|
+
return argv[i + 1];
|
|
20
|
+
};
|
|
21
|
+
return {
|
|
22
|
+
lessonPath: get("--lesson") ?? null,
|
|
23
|
+
rationale: get("--rationale") ?? null,
|
|
24
|
+
failureType: get("--failure-type") ?? null,
|
|
25
|
+
signature: get("--signature") ?? null,
|
|
26
|
+
summary: get("--summary") ?? null,
|
|
27
|
+
provider: get("--provider") ?? "github-actions",
|
|
28
|
+
logsPath: get("--logs") ?? null,
|
|
29
|
+
remediation: get("--remediation") ?? null,
|
|
30
|
+
reportUrl: get("--report-url") ?? process.env.MAIND_REPORT_URL ?? DEFAULT_REPORT_URL,
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
/** Collects CI run metadata from the runner environment (GitHub Actions today). */
|
|
34
|
+
function ciEnv() {
|
|
35
|
+
const env = process.env;
|
|
36
|
+
const runId = env.GITHUB_RUN_ID;
|
|
37
|
+
const repo = env.GITHUB_REPOSITORY;
|
|
38
|
+
const server = env.GITHUB_SERVER_URL ?? "https://github.com";
|
|
39
|
+
return {
|
|
40
|
+
workflow_name: env.GITHUB_WORKFLOW || undefined,
|
|
41
|
+
run_id: runId || undefined,
|
|
42
|
+
run_url: runId && repo ? `${server}/${repo}/actions/runs/${runId}` : undefined,
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
function sha256(s) {
|
|
46
|
+
return createHash("sha256").update(s, "utf8").digest("hex");
|
|
47
|
+
}
|
|
48
|
+
/** Runs the report. Always returns 0 (fail-open). */
|
|
49
|
+
export async function runCiReport(opts) {
|
|
50
|
+
try {
|
|
51
|
+
if (!opts.lessonPath) {
|
|
52
|
+
warn("missing --lesson <path-to-authored-lesson.md>; nothing to report.");
|
|
53
|
+
return 0;
|
|
54
|
+
}
|
|
55
|
+
if (!opts.rationale) {
|
|
56
|
+
warn("missing --rationale; the agent should explain why this is a lesson.");
|
|
57
|
+
return 0;
|
|
58
|
+
}
|
|
59
|
+
if (!opts.failureType) {
|
|
60
|
+
warn("missing --failure-type (e.g. tsc-error, test-failure).");
|
|
61
|
+
return 0;
|
|
62
|
+
}
|
|
63
|
+
const lesson = readFileSync(opts.lessonPath, "utf8");
|
|
64
|
+
if (lesson.trim().length < 50) {
|
|
65
|
+
warn("authored lesson is too short — skipping report.");
|
|
66
|
+
return 0;
|
|
67
|
+
}
|
|
68
|
+
const env = ciEnv();
|
|
69
|
+
const runId = env.run_id ?? "local";
|
|
70
|
+
// Stable signature: explicit flag wins; else derive from failure-type + a
|
|
71
|
+
// hash of the lesson so the same recurring failure dedups server-side.
|
|
72
|
+
const signature = opts.signature ?? `${opts.failureType}:${sha256(opts.failureType + lesson).slice(0, 16)}`;
|
|
73
|
+
const provenance = {
|
|
74
|
+
ci_provider: opts.provider,
|
|
75
|
+
run_id: runId,
|
|
76
|
+
failure_type: opts.failureType,
|
|
77
|
+
failure_signature: signature,
|
|
78
|
+
draft_rationale: opts.rationale,
|
|
79
|
+
};
|
|
80
|
+
if (env.workflow_name)
|
|
81
|
+
provenance.workflow_name = env.workflow_name;
|
|
82
|
+
if (env.run_url)
|
|
83
|
+
provenance.run_url = env.run_url;
|
|
84
|
+
if (opts.summary)
|
|
85
|
+
provenance.failure_summary = opts.summary;
|
|
86
|
+
if (opts.remediation)
|
|
87
|
+
provenance.remediation_hint = opts.remediation;
|
|
88
|
+
if (opts.logsPath) {
|
|
89
|
+
try {
|
|
90
|
+
provenance.failure_logs = readFileSync(opts.logsPath, "utf8").slice(0, 8000);
|
|
91
|
+
}
|
|
92
|
+
catch {
|
|
93
|
+
// logs are best-effort
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
const auth = resolveAuth();
|
|
97
|
+
const body = JSON.stringify({
|
|
98
|
+
lesson,
|
|
99
|
+
content_hash: sha256(lesson),
|
|
100
|
+
ci_provenance: provenance,
|
|
101
|
+
});
|
|
102
|
+
let res;
|
|
103
|
+
try {
|
|
104
|
+
res = await fetch(opts.reportUrl, {
|
|
105
|
+
method: "POST",
|
|
106
|
+
headers: {
|
|
107
|
+
"content-type": "application/json",
|
|
108
|
+
authorization: `Bearer ${auth.apiKey}`,
|
|
109
|
+
},
|
|
110
|
+
body,
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
catch (err) {
|
|
114
|
+
warn(`network error posting to ${opts.reportUrl}: ${msg(err)}`);
|
|
115
|
+
return 0;
|
|
116
|
+
}
|
|
117
|
+
const text = await res.text().catch(() => "");
|
|
118
|
+
if (!res.ok) {
|
|
119
|
+
warn(`ingest rejected (HTTP ${res.status}): ${text.slice(0, 300)}`);
|
|
120
|
+
return 0;
|
|
121
|
+
}
|
|
122
|
+
let parsed = {};
|
|
123
|
+
try {
|
|
124
|
+
parsed = JSON.parse(text);
|
|
125
|
+
}
|
|
126
|
+
catch {
|
|
127
|
+
// non-JSON 200 — treat as success but unparseable
|
|
128
|
+
}
|
|
129
|
+
process.stderr.write(`maind ci-report: ${parsed.deduplicated ? "deduplicated" : "filed"} draft ${parsed.draft_id ?? "?"}` +
|
|
130
|
+
(parsed.review_url ? ` → ${parsed.review_url}` : "") +
|
|
131
|
+
"\n");
|
|
132
|
+
return 0;
|
|
133
|
+
}
|
|
134
|
+
catch (err) {
|
|
135
|
+
const reason = err instanceof MissingApiKeyError ? err.message : msg(err);
|
|
136
|
+
warn(`report failed, continuing: ${reason}`);
|
|
137
|
+
return 0;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
function msg(err) {
|
|
141
|
+
return err instanceof Error ? err.message : String(err);
|
|
142
|
+
}
|
|
143
|
+
function warn(line) {
|
|
144
|
+
process.stderr.write(`maind ci-report: ${line}\n`);
|
|
145
|
+
}
|
|
146
|
+
//# sourceMappingURL=ci-report.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ci-report.js","sourceRoot":"","sources":["../src/ci-report.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
// `maind ci-result` — report a CI build outcome (success|failure) together with
|
|
2
|
+
// the served-lesson manifest (ADR-207 Phase 3, FEEDBACK). For every lesson that
|
|
3
|
+
// was in the agent's context for this run (from `maind context`'s manifest.json),
|
|
4
|
+
// maind records a per-run CI signal. The aggregate "CI-confirmed N×" is shown to
|
|
5
|
+
// maintainers as a human-gated signal — it never changes a lesson's trust
|
|
6
|
+
// automatically. The signal means "this lesson was in context for N green runs"
|
|
7
|
+
// (correlation, not causation).
|
|
8
|
+
//
|
|
9
|
+
// Fail-open by contract: a failing report must never break the build. Any error
|
|
10
|
+
// → stderr warning, exit 0. Runs at the end of a job (success OR failure).
|
|
11
|
+
import { readFileSync } from "node:fs";
|
|
12
|
+
import { MissingApiKeyError, resolveAuth } from "./auth.js";
|
|
13
|
+
const DEFAULT_RESULT_URL = "https://app.maind.dev/api/ci/result";
|
|
14
|
+
export function parseCiResultArgs(argv) {
|
|
15
|
+
const get = (name) => {
|
|
16
|
+
const i = argv.indexOf(name);
|
|
17
|
+
if (i === -1 || i + 1 >= argv.length)
|
|
18
|
+
return undefined;
|
|
19
|
+
return argv[i + 1];
|
|
20
|
+
};
|
|
21
|
+
return {
|
|
22
|
+
outcome: get("--outcome") ?? null,
|
|
23
|
+
manifestPath: get("--manifest") ?? ".maind/manifest.json",
|
|
24
|
+
runRef: get("--run-ref") ?? null,
|
|
25
|
+
runUrl: get("--run-url") ?? null,
|
|
26
|
+
resultUrl: get("--result-url") ?? process.env.MAIND_RESULT_URL ?? DEFAULT_RESULT_URL,
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Normalizes common CI outcome spellings (incl. GitHub Actions `${{ job.status }}`
|
|
31
|
+
* → success|failure|cancelled) to the two states the server records.
|
|
32
|
+
*/
|
|
33
|
+
function normalizeOutcome(raw) {
|
|
34
|
+
const s = raw.trim().toLowerCase();
|
|
35
|
+
if (["success", "passed", "pass", "passing", "ok", "green", "0"].includes(s))
|
|
36
|
+
return "success";
|
|
37
|
+
if (["failure", "failed", "fail", "failing", "error", "red", "cancelled", "canceled"].includes(s)) {
|
|
38
|
+
return "failure";
|
|
39
|
+
}
|
|
40
|
+
return null;
|
|
41
|
+
}
|
|
42
|
+
/** Collects CI run metadata from the runner environment (GitHub Actions today). */
|
|
43
|
+
function ciEnv() {
|
|
44
|
+
const env = process.env;
|
|
45
|
+
const runId = env.GITHUB_RUN_ID;
|
|
46
|
+
const repo = env.GITHUB_REPOSITORY;
|
|
47
|
+
const server = env.GITHUB_SERVER_URL ?? "https://github.com";
|
|
48
|
+
return {
|
|
49
|
+
run_id: runId || undefined,
|
|
50
|
+
run_url: runId && repo ? `${server}/${repo}/actions/runs/${runId}` : undefined,
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
/** Runs the result report. Always returns 0 (fail-open). */
|
|
54
|
+
export async function runCiResult(opts) {
|
|
55
|
+
try {
|
|
56
|
+
if (!opts.outcome) {
|
|
57
|
+
warn("missing --outcome <success|failure> (e.g. --outcome ${{ job.status }}).");
|
|
58
|
+
return 0;
|
|
59
|
+
}
|
|
60
|
+
const outcome = normalizeOutcome(opts.outcome);
|
|
61
|
+
if (!outcome) {
|
|
62
|
+
warn(`unrecognized --outcome '${opts.outcome}'; expected success|failure.`);
|
|
63
|
+
return 0;
|
|
64
|
+
}
|
|
65
|
+
let manifestRaw;
|
|
66
|
+
try {
|
|
67
|
+
manifestRaw = readFileSync(opts.manifestPath, "utf8");
|
|
68
|
+
}
|
|
69
|
+
catch {
|
|
70
|
+
warn(`no manifest at ${opts.manifestPath} (did 'maind context' run first?); nothing to attribute.`);
|
|
71
|
+
return 0;
|
|
72
|
+
}
|
|
73
|
+
let manifest;
|
|
74
|
+
try {
|
|
75
|
+
manifest = JSON.parse(manifestRaw);
|
|
76
|
+
}
|
|
77
|
+
catch {
|
|
78
|
+
warn(`manifest at ${opts.manifestPath} is not valid JSON; skipping.`);
|
|
79
|
+
return 0;
|
|
80
|
+
}
|
|
81
|
+
const lessonIds = Array.isArray(manifest.served)
|
|
82
|
+
? manifest.served.map((s) => s.id).filter((id) => typeof id === "string" && id.length > 0)
|
|
83
|
+
: [];
|
|
84
|
+
if (lessonIds.length === 0) {
|
|
85
|
+
warn("manifest lists no served lessons; nothing to report.");
|
|
86
|
+
return 0;
|
|
87
|
+
}
|
|
88
|
+
const env = ciEnv();
|
|
89
|
+
const runRef = opts.runRef ?? manifest.run_ref ?? env.run_id ?? null;
|
|
90
|
+
if (!runRef) {
|
|
91
|
+
warn("no run ref (--run-ref / manifest.run_ref / $GITHUB_RUN_ID); cannot attribute an outcome.");
|
|
92
|
+
return 0;
|
|
93
|
+
}
|
|
94
|
+
const runUrl = opts.runUrl ?? env.run_url ?? null;
|
|
95
|
+
const auth = resolveAuth();
|
|
96
|
+
const body = JSON.stringify({
|
|
97
|
+
outcome,
|
|
98
|
+
run_ref: runRef,
|
|
99
|
+
run_url: runUrl,
|
|
100
|
+
lesson_ids: lessonIds,
|
|
101
|
+
});
|
|
102
|
+
let res;
|
|
103
|
+
try {
|
|
104
|
+
res = await fetch(opts.resultUrl, {
|
|
105
|
+
method: "POST",
|
|
106
|
+
headers: {
|
|
107
|
+
"content-type": "application/json",
|
|
108
|
+
authorization: `Bearer ${auth.apiKey}`,
|
|
109
|
+
},
|
|
110
|
+
body,
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
catch (err) {
|
|
114
|
+
warn(`network error posting to ${opts.resultUrl}: ${msg(err)}`);
|
|
115
|
+
return 0;
|
|
116
|
+
}
|
|
117
|
+
const text = await res.text().catch(() => "");
|
|
118
|
+
if (!res.ok) {
|
|
119
|
+
warn(`ingest rejected (HTTP ${res.status}): ${text.slice(0, 300)}`);
|
|
120
|
+
return 0;
|
|
121
|
+
}
|
|
122
|
+
let parsed = {};
|
|
123
|
+
try {
|
|
124
|
+
parsed = JSON.parse(text);
|
|
125
|
+
}
|
|
126
|
+
catch {
|
|
127
|
+
// non-JSON 200 — treat as success but unparseable
|
|
128
|
+
}
|
|
129
|
+
process.stderr.write(`maind ci-result: ${outcome} · recorded ${parsed.recorded ?? "?"} · skipped ${parsed.skipped ?? 0}` +
|
|
130
|
+
` (run ${runRef})\n`);
|
|
131
|
+
return 0;
|
|
132
|
+
}
|
|
133
|
+
catch (err) {
|
|
134
|
+
const reason = err instanceof MissingApiKeyError ? err.message : msg(err);
|
|
135
|
+
warn(`result failed, continuing: ${reason}`);
|
|
136
|
+
return 0;
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
function msg(err) {
|
|
140
|
+
return err instanceof Error ? err.message : String(err);
|
|
141
|
+
}
|
|
142
|
+
function warn(line) {
|
|
143
|
+
process.stderr.write(`maind ci-result: ${line}\n`);
|
|
144
|
+
}
|
|
145
|
+
//# sourceMappingURL=ci-result.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ci-result.js","sourceRoot":"","sources":["../src/ci-result.ts"],"names":[],"mappings":"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"}
|
package/build/index.js
CHANGED
|
@@ -10,14 +10,36 @@
|
|
|
10
10
|
// STDOUT is reserved for command results that are meant to be piped; all
|
|
11
11
|
// diagnostics go to STDERR. Today only the `context` command exists.
|
|
12
12
|
import { parseContextArgs, runContext } from "./context.js";
|
|
13
|
-
|
|
13
|
+
import { parseCiReportArgs, runCiReport } from "./ci-report.js";
|
|
14
|
+
import { parseCiResultArgs, runCiResult } from "./ci-result.js";
|
|
15
|
+
const VERSION = "0.3.0";
|
|
14
16
|
const HELP = `maind ${VERSION} — headless maind CLI for CI
|
|
15
17
|
|
|
16
18
|
Usage:
|
|
17
19
|
maind context [options] Pull stack-relevant lessons + conventions into the runner
|
|
20
|
+
maind ci-report [options] Report a CI failure as a draft-lesson candidate (on failure)
|
|
21
|
+
maind ci-result [options] Report a build outcome + served manifest (end of job)
|
|
18
22
|
maind --help Show this help
|
|
19
23
|
maind --version Print version
|
|
20
24
|
|
|
25
|
+
ci-report options (run in an on-failure step; the agent authors the lesson first):
|
|
26
|
+
--lesson <path> Agent-authored lesson markdown (frontmatter + body) — required
|
|
27
|
+
--rationale <text> Why this is a reusable lesson — required
|
|
28
|
+
--failure-type <t> e.g. tsc-error, test-failure, lint-error — required
|
|
29
|
+
--signature <s> Stable dedup key (default: failure-type + lesson hash)
|
|
30
|
+
--summary <text> Short failure description
|
|
31
|
+
--remediation <text> Suggested fix
|
|
32
|
+
--logs <path> Optional failure-log file (truncated to 8k)
|
|
33
|
+
--provider <p> CI provider (default github-actions)
|
|
34
|
+
--report-url <url> Override ingest endpoint (default \$MAIND_REPORT_URL or prod)
|
|
35
|
+
|
|
36
|
+
ci-result options (run at the end of a job, success OR failure):
|
|
37
|
+
--outcome <s> Build outcome, e.g. \${{ job.status }} (success|failure) — required
|
|
38
|
+
--manifest <path> Served-lesson manifest from 'maind context' (default .maind/manifest.json)
|
|
39
|
+
--run-ref <id> Build identifier (default manifest.run_ref or \$GITHUB_RUN_ID)
|
|
40
|
+
--run-url <url> Link to the CI run (default derived from GitHub env)
|
|
41
|
+
--result-url <url> Override ingest endpoint (default \$MAIND_RESULT_URL or prod)
|
|
42
|
+
|
|
21
43
|
context options:
|
|
22
44
|
--stack <a,b> Stack hint (seeds the platforms filter), e.g. nextjs,supabase
|
|
23
45
|
--platforms <a,b> Explicit platforms filter (overrides --stack)
|
|
@@ -49,6 +71,12 @@ async function main() {
|
|
|
49
71
|
if (cmd === "context") {
|
|
50
72
|
return runContext(parseContextArgs(argv.slice(1)));
|
|
51
73
|
}
|
|
74
|
+
if (cmd === "ci-report") {
|
|
75
|
+
return runCiReport(parseCiReportArgs(argv.slice(1)));
|
|
76
|
+
}
|
|
77
|
+
if (cmd === "ci-result") {
|
|
78
|
+
return runCiResult(parseCiResultArgs(argv.slice(1)));
|
|
79
|
+
}
|
|
52
80
|
process.stderr.write(`maind: unknown command '${cmd}'. Run 'maind --help'.\n`);
|
|
53
81
|
return 2;
|
|
54
82
|
}
|
package/build/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,EAAE;AACF,gCAAgC;AAChC,EAAE;AACF,uEAAuE;AACvE,2EAA2E;AAC3E,gFAAgF;AAChF,sEAAsE;AACtE,EAAE;AACF,yEAAyE;AACzE,qEAAqE;AAErE,OAAO,EAAE,gBAAgB,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,EAAE;AACF,gCAAgC;AAChC,EAAE;AACF,uEAAuE;AACvE,2EAA2E;AAC3E,gFAAgF;AAChF,sEAAsE;AACtE,EAAE;AACF,yEAAyE;AACzE,qEAAqE;AAErE,OAAO,EAAE,gBAAgB,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC5D,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAEhE,MAAM,OAAO,GAAG,OAAO,CAAC;AAExB,MAAM,IAAI,GAAG,SAAS,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA2C5B,CAAC;AAEF,KAAK,UAAU,IAAI;IACjB,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IACnC,MAAM,GAAG,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;IAEpB,IAAI,CAAC,GAAG,IAAI,GAAG,KAAK,QAAQ,IAAI,GAAG,KAAK,IAAI,IAAI,GAAG,KAAK,MAAM,EAAE,CAAC;QAC/D,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC3B,OAAO,CAAC,CAAC;IACX,CAAC;IACD,IAAI,GAAG,KAAK,WAAW,IAAI,GAAG,KAAK,IAAI,EAAE,CAAC;QACxC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC;QACrC,OAAO,CAAC,CAAC;IACX,CAAC;IACD,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC;QACtB,OAAO,UAAU,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACrD,CAAC;IACD,IAAI,GAAG,KAAK,WAAW,EAAE,CAAC;QACxB,OAAO,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACvD,CAAC;IACD,IAAI,GAAG,KAAK,WAAW,EAAE,CAAC;QACxB,OAAO,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACvD,CAAC;IAED,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,2BAA2B,GAAG,0BAA0B,CAAC,CAAC;IAC/E,OAAO,CAAC,CAAC;AACX,CAAC;AAED,IAAI,EAAE;KACH,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;KAClC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACb,+EAA+E;IAC/E,2EAA2E;IAC3E,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,iBAAiB,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,IAAI,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACzG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@maind-dev/cli",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.3.0",
|
|
4
4
|
"description": "Headless maind CLI — pull stack-relevant lessons into CI runners and emit a context manifest for build-outcome attribution. No device flow; authenticates from MAIND_API_KEY.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./build/index.js",
|