opengstack 0.13.7 → 0.13.9
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/bin/opengstack.js +35 -90
- package/package.json +2 -3
- package/scripts/install-skills.js +47 -58
- package/skills/browse/bin/find-browse +21 -0
- package/skills/browse/bin/remote-slug +14 -0
- package/skills/browse/scripts/build-node-server.sh +48 -0
- package/skills/browse/src/activity.ts +208 -0
- package/skills/browse/src/browser-manager.ts +959 -0
- package/skills/browse/src/buffers.ts +137 -0
- package/skills/browse/src/bun-polyfill.cjs +109 -0
- package/skills/browse/src/cli.ts +678 -0
- package/skills/browse/src/commands.ts +128 -0
- package/skills/browse/src/config.ts +150 -0
- package/skills/browse/src/cookie-import-browser.ts +625 -0
- package/skills/browse/src/cookie-picker-routes.ts +230 -0
- package/skills/browse/src/cookie-picker-ui.ts +688 -0
- package/skills/browse/src/find-browse.ts +61 -0
- package/skills/browse/src/meta-commands.ts +550 -0
- package/skills/browse/src/platform.ts +17 -0
- package/skills/browse/src/read-commands.ts +358 -0
- package/skills/browse/src/server.ts +1192 -0
- package/skills/browse/src/sidebar-agent.ts +280 -0
- package/skills/browse/src/sidebar-utils.ts +21 -0
- package/skills/browse/src/snapshot.ts +407 -0
- package/skills/browse/src/url-validation.ts +95 -0
- package/skills/browse/src/write-commands.ts +364 -0
- package/skills/browse/test/activity.test.ts +120 -0
- package/skills/browse/test/adversarial-security.test.ts +32 -0
- package/skills/browse/test/browser-manager-unit.test.ts +17 -0
- package/skills/browse/test/bun-polyfill.test.ts +72 -0
- package/skills/browse/test/commands.test.ts +2075 -0
- package/skills/browse/test/compare-board.test.ts +342 -0
- package/skills/browse/test/config.test.ts +316 -0
- package/skills/browse/test/cookie-import-browser.test.ts +519 -0
- package/skills/browse/test/cookie-picker-routes.test.ts +260 -0
- package/skills/browse/test/file-drop.test.ts +271 -0
- package/skills/browse/test/find-browse.test.ts +50 -0
- package/skills/browse/test/findport.test.ts +191 -0
- package/skills/browse/test/fixtures/basic.html +33 -0
- package/skills/browse/test/fixtures/cursor-interactive.html +22 -0
- package/skills/browse/test/fixtures/dialog.html +15 -0
- package/skills/browse/test/fixtures/empty.html +2 -0
- package/skills/browse/test/fixtures/forms.html +55 -0
- package/skills/browse/test/fixtures/iframe.html +30 -0
- package/skills/browse/test/fixtures/network-idle.html +30 -0
- package/skills/browse/test/fixtures/qa-eval-checkout.html +108 -0
- package/skills/browse/test/fixtures/qa-eval-spa.html +98 -0
- package/skills/browse/test/fixtures/qa-eval.html +51 -0
- package/skills/browse/test/fixtures/responsive.html +49 -0
- package/skills/browse/test/fixtures/snapshot.html +55 -0
- package/skills/browse/test/fixtures/spa.html +24 -0
- package/skills/browse/test/fixtures/states.html +17 -0
- package/skills/browse/test/fixtures/upload.html +25 -0
- package/skills/browse/test/gstack-config.test.ts +138 -0
- package/skills/browse/test/gstack-update-check.test.ts +514 -0
- package/skills/browse/test/handoff.test.ts +235 -0
- package/skills/browse/test/path-validation.test.ts +91 -0
- package/skills/browse/test/platform.test.ts +37 -0
- package/skills/browse/test/server-auth.test.ts +65 -0
- package/skills/browse/test/sidebar-agent-roundtrip.test.ts +226 -0
- package/skills/browse/test/sidebar-agent.test.ts +199 -0
- package/skills/browse/test/sidebar-integration.test.ts +320 -0
- package/skills/browse/test/sidebar-unit.test.ts +96 -0
- package/skills/browse/test/snapshot.test.ts +467 -0
- package/skills/browse/test/state-ttl.test.ts +35 -0
- package/skills/browse/test/test-server.ts +57 -0
- package/skills/browse/test/url-validation.test.ts +72 -0
- package/skills/browse/test/watch.test.ts +129 -0
- package/skills/careful/bin/check-careful.sh +112 -0
- package/skills/cso/ACKNOWLEDGEMENTS.md +14 -0
- package/skills/freeze/bin/check-freeze.sh +79 -0
- package/skills/qa/references/issue-taxonomy.md +85 -0
- package/skills/qa/templates/qa-report-template.md +126 -0
- package/skills/review/TODOS-format.md +62 -0
- package/skills/review/checklist.md +220 -0
- package/skills/review/design-checklist.md +132 -0
- package/skills/review/greptile-triage.md +220 -0
- /package/{autoplan → skills/autoplan}/SKILL.md +0 -0
- /package/{autoplan → skills/autoplan}/SKILL.md.tmpl +0 -0
- /package/{benchmark → skills/benchmark}/SKILL.md +0 -0
- /package/{benchmark → skills/benchmark}/SKILL.md.tmpl +0 -0
- /package/{browse → skills/browse}/SKILL.md +0 -0
- /package/{browse → skills/browse}/SKILL.md.tmpl +0 -0
- /package/{canary → skills/canary}/SKILL.md +0 -0
- /package/{canary → skills/canary}/SKILL.md.tmpl +0 -0
- /package/{careful → skills/careful}/SKILL.md +0 -0
- /package/{careful → skills/careful}/SKILL.md.tmpl +0 -0
- /package/{codex → skills/codex}/SKILL.md +0 -0
- /package/{codex → skills/codex}/SKILL.md.tmpl +0 -0
- /package/{connect-chrome → skills/connect-chrome}/SKILL.md +0 -0
- /package/{connect-chrome → skills/connect-chrome}/SKILL.md.tmpl +0 -0
- /package/{cso → skills/cso}/SKILL.md +0 -0
- /package/{cso → skills/cso}/SKILL.md.tmpl +0 -0
- /package/{design-consultation → skills/design-consultation}/SKILL.md +0 -0
- /package/{design-consultation → skills/design-consultation}/SKILL.md.tmpl +0 -0
- /package/{design-review → skills/design-review}/SKILL.md +0 -0
- /package/{design-review → skills/design-review}/SKILL.md.tmpl +0 -0
- /package/{design-shotgun → skills/design-shotgun}/SKILL.md +0 -0
- /package/{design-shotgun → skills/design-shotgun}/SKILL.md.tmpl +0 -0
- /package/{document-release → skills/document-release}/SKILL.md +0 -0
- /package/{document-release → skills/document-release}/SKILL.md.tmpl +0 -0
- /package/{freeze → skills/freeze}/SKILL.md +0 -0
- /package/{freeze → skills/freeze}/SKILL.md.tmpl +0 -0
- /package/{gstack-upgrade → skills/gstack-upgrade}/SKILL.md +0 -0
- /package/{gstack-upgrade → skills/gstack-upgrade}/SKILL.md.tmpl +0 -0
- /package/{guard → skills/guard}/SKILL.md +0 -0
- /package/{guard → skills/guard}/SKILL.md.tmpl +0 -0
- /package/{investigate → skills/investigate}/SKILL.md +0 -0
- /package/{investigate → skills/investigate}/SKILL.md.tmpl +0 -0
- /package/{land-and-deploy → skills/land-and-deploy}/SKILL.md +0 -0
- /package/{land-and-deploy → skills/land-and-deploy}/SKILL.md.tmpl +0 -0
- /package/{office-hours → skills/office-hours}/SKILL.md +0 -0
- /package/{office-hours → skills/office-hours}/SKILL.md.tmpl +0 -0
- /package/{plan-ceo-review → skills/plan-ceo-review}/SKILL.md +0 -0
- /package/{plan-ceo-review → skills/plan-ceo-review}/SKILL.md.tmpl +0 -0
- /package/{plan-design-review → skills/plan-design-review}/SKILL.md +0 -0
- /package/{plan-design-review → skills/plan-design-review}/SKILL.md.tmpl +0 -0
- /package/{plan-eng-review → skills/plan-eng-review}/SKILL.md +0 -0
- /package/{plan-eng-review → skills/plan-eng-review}/SKILL.md.tmpl +0 -0
- /package/{qa → skills/qa}/SKILL.md +0 -0
- /package/{qa → skills/qa}/SKILL.md.tmpl +0 -0
- /package/{qa-only → skills/qa-only}/SKILL.md +0 -0
- /package/{qa-only → skills/qa-only}/SKILL.md.tmpl +0 -0
- /package/{retro → skills/retro}/SKILL.md +0 -0
- /package/{retro → skills/retro}/SKILL.md.tmpl +0 -0
- /package/{review → skills/review}/SKILL.md +0 -0
- /package/{review → skills/review}/SKILL.md.tmpl +0 -0
- /package/{setup-browser-cookies → skills/setup-browser-cookies}/SKILL.md +0 -0
- /package/{setup-browser-cookies → skills/setup-browser-cookies}/SKILL.md.tmpl +0 -0
- /package/{setup-deploy → skills/setup-deploy}/SKILL.md +0 -0
- /package/{setup-deploy → skills/setup-deploy}/SKILL.md.tmpl +0 -0
- /package/{ship → skills/ship}/SKILL.md +0 -0
- /package/{ship → skills/ship}/SKILL.md.tmpl +0 -0
- /package/{unfreeze → skills/unfreeze}/SKILL.md +0 -0
- /package/{unfreeze → skills/unfreeze}/SKILL.md.tmpl +0 -0
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8">
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
6
|
+
<title>Test Page - Responsive</title>
|
|
7
|
+
<style>
|
|
8
|
+
body { font-family: sans-serif; margin: 0; padding: 20px; }
|
|
9
|
+
.container { max-width: 1200px; margin: 0 auto; }
|
|
10
|
+
.grid { display: grid; gap: 16px; }
|
|
11
|
+
.card { padding: 16px; border: 1px solid #ddd; border-radius: 8px; }
|
|
12
|
+
|
|
13
|
+
/* Mobile: single column */
|
|
14
|
+
.grid { grid-template-columns: 1fr; }
|
|
15
|
+
|
|
16
|
+
/* Tablet: two columns */
|
|
17
|
+
@media (min-width: 768px) {
|
|
18
|
+
.grid { grid-template-columns: 1fr 1fr; }
|
|
19
|
+
.mobile-only { display: none; }
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/* Desktop: three columns */
|
|
23
|
+
@media (min-width: 1024px) {
|
|
24
|
+
.grid { grid-template-columns: 1fr 1fr 1fr; }
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
.mobile-only { color: red; }
|
|
28
|
+
.desktop-indicator { display: none; }
|
|
29
|
+
@media (min-width: 1024px) {
|
|
30
|
+
.desktop-indicator { display: block; color: green; }
|
|
31
|
+
}
|
|
32
|
+
</style>
|
|
33
|
+
</head>
|
|
34
|
+
<body>
|
|
35
|
+
<div class="container">
|
|
36
|
+
<h1>Responsive Layout Test</h1>
|
|
37
|
+
<p class="mobile-only">You are on mobile</p>
|
|
38
|
+
<p class="desktop-indicator">You are on desktop</p>
|
|
39
|
+
<div class="grid">
|
|
40
|
+
<div class="card">Card 1</div>
|
|
41
|
+
<div class="card">Card 2</div>
|
|
42
|
+
<div class="card">Card 3</div>
|
|
43
|
+
<div class="card">Card 4</div>
|
|
44
|
+
<div class="card">Card 5</div>
|
|
45
|
+
<div class="card">Card 6</div>
|
|
46
|
+
</div>
|
|
47
|
+
</div>
|
|
48
|
+
</body>
|
|
49
|
+
</html>
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8">
|
|
5
|
+
<title>Snapshot Test Page</title>
|
|
6
|
+
<style>
|
|
7
|
+
body { font-family: sans-serif; padding: 20px; }
|
|
8
|
+
form { margin: 10px 0; }
|
|
9
|
+
input, select, button { margin: 5px; padding: 5px; }
|
|
10
|
+
#main { border: 1px solid #ccc; padding: 10px; }
|
|
11
|
+
.empty-div { }
|
|
12
|
+
.hidden { display: none; }
|
|
13
|
+
</style>
|
|
14
|
+
</head>
|
|
15
|
+
<body>
|
|
16
|
+
<h1>Snapshot Test</h1>
|
|
17
|
+
<h2>Subheading</h2>
|
|
18
|
+
|
|
19
|
+
<nav>
|
|
20
|
+
<a href="/page1">Internal Link</a>
|
|
21
|
+
<a href="https://external.com">External Link</a>
|
|
22
|
+
</nav>
|
|
23
|
+
|
|
24
|
+
<div id="main">
|
|
25
|
+
<h3>Form Section</h3>
|
|
26
|
+
<form id="test-form">
|
|
27
|
+
<input type="text" id="username" placeholder="Username" aria-label="Username">
|
|
28
|
+
<input type="email" id="email" placeholder="Email" aria-label="Email">
|
|
29
|
+
<input type="password" id="pass" placeholder="Password" aria-label="Password">
|
|
30
|
+
<label><input type="checkbox" id="agree"> I agree</label>
|
|
31
|
+
<select id="role" aria-label="Role">
|
|
32
|
+
<option value="">Choose...</option>
|
|
33
|
+
<option value="admin">Admin</option>
|
|
34
|
+
<option value="user">User</option>
|
|
35
|
+
</select>
|
|
36
|
+
<button type="submit" id="submit-btn">Submit</button>
|
|
37
|
+
<button type="button" id="cancel-btn">Cancel</button>
|
|
38
|
+
</form>
|
|
39
|
+
</div>
|
|
40
|
+
|
|
41
|
+
<div class="empty-div">
|
|
42
|
+
<div class="empty-div">
|
|
43
|
+
<button id="nested-btn">Nested Button</button>
|
|
44
|
+
</div>
|
|
45
|
+
</div>
|
|
46
|
+
|
|
47
|
+
<p>Some paragraph text that is not interactive.</p>
|
|
48
|
+
|
|
49
|
+
<script>
|
|
50
|
+
document.getElementById('test-form').addEventListener('submit', (e) => {
|
|
51
|
+
e.preventDefault();
|
|
52
|
+
});
|
|
53
|
+
</script>
|
|
54
|
+
</body>
|
|
55
|
+
</html>
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8">
|
|
5
|
+
<title>Test Page - SPA</title>
|
|
6
|
+
<style>
|
|
7
|
+
body { font-family: sans-serif; }
|
|
8
|
+
#app { padding: 20px; }
|
|
9
|
+
.loaded { color: green; }
|
|
10
|
+
</style>
|
|
11
|
+
</head>
|
|
12
|
+
<body>
|
|
13
|
+
<div id="app">Loading...</div>
|
|
14
|
+
<script>
|
|
15
|
+
console.log('[SPA] Starting render');
|
|
16
|
+
console.warn('[SPA] This is a warning');
|
|
17
|
+
console.error('[SPA] This is an error');
|
|
18
|
+
setTimeout(() => {
|
|
19
|
+
document.getElementById('app').innerHTML = '<h1 class="loaded">SPA Content Loaded</h1><p>Rendered by JavaScript</p>';
|
|
20
|
+
console.log('[SPA] Render complete');
|
|
21
|
+
}, 500);
|
|
22
|
+
</script>
|
|
23
|
+
</body>
|
|
24
|
+
</html>
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8">
|
|
5
|
+
<title>Test Page - Element States</title>
|
|
6
|
+
</head>
|
|
7
|
+
<body>
|
|
8
|
+
<h1>Element States Test</h1>
|
|
9
|
+
<input type="text" id="enabled-input" value="enabled" />
|
|
10
|
+
<input type="text" id="disabled-input" value="disabled" disabled />
|
|
11
|
+
<input type="checkbox" id="checked-box" checked />
|
|
12
|
+
<input type="checkbox" id="unchecked-box" />
|
|
13
|
+
<div id="visible-div">Visible</div>
|
|
14
|
+
<div id="hidden-div" style="display: none;">Hidden</div>
|
|
15
|
+
<input type="text" id="readonly-input" readonly value="readonly" />
|
|
16
|
+
</body>
|
|
17
|
+
</html>
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8">
|
|
5
|
+
<title>Test Page - Upload</title>
|
|
6
|
+
</head>
|
|
7
|
+
<body>
|
|
8
|
+
<h1>Upload Test</h1>
|
|
9
|
+
<input type="file" id="file-input" />
|
|
10
|
+
<input type="file" id="multi-input" multiple />
|
|
11
|
+
<p id="upload-result"></p>
|
|
12
|
+
<script>
|
|
13
|
+
document.getElementById('file-input').addEventListener('change', function(e) {
|
|
14
|
+
const files = e.target.files;
|
|
15
|
+
const names = Array.from(files).map(f => f.name).join(', ');
|
|
16
|
+
document.getElementById('upload-result').textContent = 'Uploaded: ' + names;
|
|
17
|
+
});
|
|
18
|
+
document.getElementById('multi-input').addEventListener('change', function(e) {
|
|
19
|
+
const files = e.target.files;
|
|
20
|
+
const names = Array.from(files).map(f => f.name).join(', ');
|
|
21
|
+
document.getElementById('upload-result').textContent = 'Multi: ' + names;
|
|
22
|
+
});
|
|
23
|
+
</script>
|
|
24
|
+
</body>
|
|
25
|
+
</html>
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tests for bin/gstack-config bash script.
|
|
3
|
+
*
|
|
4
|
+
* Uses Bun.spawnSync to invoke the script with temp dirs and
|
|
5
|
+
* GSTACK_STATE_DIR env override for full isolation.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, test, expect, beforeEach, afterEach } from 'bun:test';
|
|
9
|
+
import { mkdtempSync, writeFileSync, rmSync, readFileSync, existsSync } from 'fs';
|
|
10
|
+
import { join } from 'path';
|
|
11
|
+
import { tmpdir } from 'os';
|
|
12
|
+
|
|
13
|
+
const SCRIPT = join(import.meta.dir, '..', '..', 'bin', 'gstack-config');
|
|
14
|
+
|
|
15
|
+
let stateDir: string;
|
|
16
|
+
|
|
17
|
+
function run(args: string[] = [], extraEnv: Record<string, string> = {}) {
|
|
18
|
+
const result = Bun.spawnSync(['bash', SCRIPT, ...args], {
|
|
19
|
+
env: {
|
|
20
|
+
...process.env,
|
|
21
|
+
GSTACK_STATE_DIR: stateDir,
|
|
22
|
+
...extraEnv,
|
|
23
|
+
},
|
|
24
|
+
stdout: 'pipe',
|
|
25
|
+
stderr: 'pipe',
|
|
26
|
+
});
|
|
27
|
+
return {
|
|
28
|
+
exitCode: result.exitCode,
|
|
29
|
+
stdout: result.stdout.toString().trim(),
|
|
30
|
+
stderr: result.stderr.toString().trim(),
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
beforeEach(() => {
|
|
35
|
+
stateDir = mkdtempSync(join(tmpdir(), 'gstack-config-test-'));
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
afterEach(() => {
|
|
39
|
+
rmSync(stateDir, { recursive: true, force: true });
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
describe('gstack-config', () => {
|
|
43
|
+
// ─── get ──────────────────────────────────────────────────
|
|
44
|
+
test('get on missing file returns empty, exit 0', () => {
|
|
45
|
+
const { exitCode, stdout } = run(['get', 'auto_upgrade']);
|
|
46
|
+
expect(exitCode).toBe(0);
|
|
47
|
+
expect(stdout).toBe('');
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
test('get existing key returns value', () => {
|
|
51
|
+
writeFileSync(join(stateDir, 'config.yaml'), 'auto_upgrade: true\n');
|
|
52
|
+
const { exitCode, stdout } = run(['get', 'auto_upgrade']);
|
|
53
|
+
expect(exitCode).toBe(0);
|
|
54
|
+
expect(stdout).toBe('true');
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
test('get missing key returns empty', () => {
|
|
58
|
+
writeFileSync(join(stateDir, 'config.yaml'), 'auto_upgrade: true\n');
|
|
59
|
+
const { exitCode, stdout } = run(['get', 'nonexistent']);
|
|
60
|
+
expect(exitCode).toBe(0);
|
|
61
|
+
expect(stdout).toBe('');
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
test('get returns last value when key appears multiple times', () => {
|
|
65
|
+
writeFileSync(join(stateDir, 'config.yaml'), 'foo: bar\nfoo: baz\n');
|
|
66
|
+
const { exitCode, stdout } = run(['get', 'foo']);
|
|
67
|
+
expect(exitCode).toBe(0);
|
|
68
|
+
expect(stdout).toBe('baz');
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
// ─── set ──────────────────────────────────────────────────
|
|
72
|
+
test('set creates file and writes key on missing file', () => {
|
|
73
|
+
const { exitCode } = run(['set', 'auto_upgrade', 'true']);
|
|
74
|
+
expect(exitCode).toBe(0);
|
|
75
|
+
const content = readFileSync(join(stateDir, 'config.yaml'), 'utf-8');
|
|
76
|
+
expect(content).toContain('auto_upgrade: true');
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
test('set appends new key to existing file', () => {
|
|
80
|
+
writeFileSync(join(stateDir, 'config.yaml'), 'foo: bar\n');
|
|
81
|
+
const { exitCode } = run(['set', 'auto_upgrade', 'true']);
|
|
82
|
+
expect(exitCode).toBe(0);
|
|
83
|
+
const content = readFileSync(join(stateDir, 'config.yaml'), 'utf-8');
|
|
84
|
+
expect(content).toContain('foo: bar');
|
|
85
|
+
expect(content).toContain('auto_upgrade: true');
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
test('set replaces existing key in-place', () => {
|
|
89
|
+
writeFileSync(join(stateDir, 'config.yaml'), 'auto_upgrade: false\n');
|
|
90
|
+
const { exitCode } = run(['set', 'auto_upgrade', 'true']);
|
|
91
|
+
expect(exitCode).toBe(0);
|
|
92
|
+
const content = readFileSync(join(stateDir, 'config.yaml'), 'utf-8');
|
|
93
|
+
expect(content).toContain('auto_upgrade: true');
|
|
94
|
+
expect(content).not.toContain('auto_upgrade: false');
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
test('set creates state dir if missing', () => {
|
|
98
|
+
const nestedDir = join(stateDir, 'nested', 'dir');
|
|
99
|
+
const { exitCode } = run(['set', 'foo', 'bar'], { GSTACK_STATE_DIR: nestedDir });
|
|
100
|
+
expect(exitCode).toBe(0);
|
|
101
|
+
expect(existsSync(join(nestedDir, 'config.yaml'))).toBe(true);
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
// ─── list ─────────────────────────────────────────────────
|
|
105
|
+
test('list shows all keys', () => {
|
|
106
|
+
writeFileSync(join(stateDir, 'config.yaml'), 'auto_upgrade: true\nupdate_check: false\n');
|
|
107
|
+
const { exitCode, stdout } = run(['list']);
|
|
108
|
+
expect(exitCode).toBe(0);
|
|
109
|
+
expect(stdout).toContain('auto_upgrade: true');
|
|
110
|
+
expect(stdout).toContain('update_check: false');
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
test('list on missing file returns empty, exit 0', () => {
|
|
114
|
+
const { exitCode, stdout } = run(['list']);
|
|
115
|
+
expect(exitCode).toBe(0);
|
|
116
|
+
expect(stdout).toBe('');
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
// ─── usage ────────────────────────────────────────────────
|
|
120
|
+
test('no args shows usage and exits 1', () => {
|
|
121
|
+
const { exitCode, stdout } = run([]);
|
|
122
|
+
expect(exitCode).toBe(1);
|
|
123
|
+
expect(stdout).toContain('Usage');
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
// ─── security: input validation ─────────────────────────
|
|
127
|
+
test('set rejects key with regex metacharacters', () => {
|
|
128
|
+
const { exitCode, stderr } = run(['set', '.*', 'value']);
|
|
129
|
+
expect(exitCode).toBe(1);
|
|
130
|
+
expect(stderr).toContain('alphanumeric');
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
test('set preserves value with sed special chars', () => {
|
|
134
|
+
run(['set', 'test_special', 'a/b&c\\d']);
|
|
135
|
+
const { stdout } = run(['get', 'test_special']);
|
|
136
|
+
expect(stdout).toBe('a/b&c\\d');
|
|
137
|
+
});
|
|
138
|
+
});
|