@sundaeswap/sprinkles 0.1.1

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.
Files changed (69) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +260 -0
  3. package/dist/cjs/Sprinkle/__tests__/bigint-reviver.test.js +40 -0
  4. package/dist/cjs/Sprinkle/__tests__/bigint-reviver.test.js.map +1 -0
  5. package/dist/cjs/Sprinkle/__tests__/encryption.test.js +267 -0
  6. package/dist/cjs/Sprinkle/__tests__/encryption.test.js.map +1 -0
  7. package/dist/cjs/Sprinkle/__tests__/enhancements.test.js +147 -0
  8. package/dist/cjs/Sprinkle/__tests__/enhancements.test.js.map +1 -0
  9. package/dist/cjs/Sprinkle/__tests__/extract-message.test.js +60 -0
  10. package/dist/cjs/Sprinkle/__tests__/extract-message.test.js.map +1 -0
  11. package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js +131 -0
  12. package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -0
  13. package/dist/cjs/Sprinkle/__tests__/schemas.test.js +184 -0
  14. package/dist/cjs/Sprinkle/__tests__/schemas.test.js.map +1 -0
  15. package/dist/cjs/Sprinkle/__tests__/settings-persistence.test.js +199 -0
  16. package/dist/cjs/Sprinkle/__tests__/settings-persistence.test.js.map +1 -0
  17. package/dist/cjs/Sprinkle/__tests__/show-menu.test.js +108 -0
  18. package/dist/cjs/Sprinkle/__tests__/show-menu.test.js.map +1 -0
  19. package/dist/cjs/Sprinkle/__tests__/test-helpers.js +16 -0
  20. package/dist/cjs/Sprinkle/__tests__/test-helpers.js.map +1 -0
  21. package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js +271 -0
  22. package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js.map +1 -0
  23. package/dist/cjs/Sprinkle/index.js +954 -0
  24. package/dist/cjs/Sprinkle/index.js.map +1 -0
  25. package/dist/cjs/index.js +17 -0
  26. package/dist/cjs/index.js.map +1 -0
  27. package/dist/cjs/package.json +1 -0
  28. package/dist/esm/Sprinkle/__tests__/bigint-reviver.test.js +38 -0
  29. package/dist/esm/Sprinkle/__tests__/bigint-reviver.test.js.map +1 -0
  30. package/dist/esm/Sprinkle/__tests__/encryption.test.js +264 -0
  31. package/dist/esm/Sprinkle/__tests__/encryption.test.js.map +1 -0
  32. package/dist/esm/Sprinkle/__tests__/enhancements.test.js +145 -0
  33. package/dist/esm/Sprinkle/__tests__/enhancements.test.js.map +1 -0
  34. package/dist/esm/Sprinkle/__tests__/extract-message.test.js +58 -0
  35. package/dist/esm/Sprinkle/__tests__/extract-message.test.js.map +1 -0
  36. package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js +130 -0
  37. package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -0
  38. package/dist/esm/Sprinkle/__tests__/schemas.test.js +182 -0
  39. package/dist/esm/Sprinkle/__tests__/schemas.test.js.map +1 -0
  40. package/dist/esm/Sprinkle/__tests__/settings-persistence.test.js +196 -0
  41. package/dist/esm/Sprinkle/__tests__/settings-persistence.test.js.map +1 -0
  42. package/dist/esm/Sprinkle/__tests__/show-menu.test.js +106 -0
  43. package/dist/esm/Sprinkle/__tests__/show-menu.test.js.map +1 -0
  44. package/dist/esm/Sprinkle/__tests__/test-helpers.js +10 -0
  45. package/dist/esm/Sprinkle/__tests__/test-helpers.js.map +1 -0
  46. package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js +269 -0
  47. package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js.map +1 -0
  48. package/dist/esm/Sprinkle/index.js +928 -0
  49. package/dist/esm/Sprinkle/index.js.map +1 -0
  50. package/dist/esm/index.js +2 -0
  51. package/dist/esm/index.js.map +1 -0
  52. package/dist/types/Sprinkle/index.d.ts +205 -0
  53. package/dist/types/Sprinkle/index.d.ts.map +1 -0
  54. package/dist/types/index.d.ts +2 -0
  55. package/dist/types/index.d.ts.map +1 -0
  56. package/dist/types/tsconfig.build.tsbuildinfo +1 -0
  57. package/package.json +85 -0
  58. package/src/Sprinkle/__tests__/bigint-reviver.test.ts +49 -0
  59. package/src/Sprinkle/__tests__/encryption.test.ts +266 -0
  60. package/src/Sprinkle/__tests__/enhancements.test.ts +154 -0
  61. package/src/Sprinkle/__tests__/extract-message.test.ts +60 -0
  62. package/src/Sprinkle/__tests__/fill-in-struct.test.ts +159 -0
  63. package/src/Sprinkle/__tests__/schemas.test.ts +215 -0
  64. package/src/Sprinkle/__tests__/settings-persistence.test.ts +181 -0
  65. package/src/Sprinkle/__tests__/show-menu.test.ts +123 -0
  66. package/src/Sprinkle/__tests__/test-helpers.ts +14 -0
  67. package/src/Sprinkle/__tests__/tx-dialog.test.ts +293 -0
  68. package/src/Sprinkle/index.ts +1215 -0
  69. package/src/index.ts +1 -0
@@ -0,0 +1,196 @@
1
+ import { describe, expect, test, beforeEach, afterEach } from "bun:test";
2
+ import { Sprinkle, Type } from "../index.js";
3
+ import * as fs from "fs";
4
+ import * as path from "path";
5
+ import * as os from "os";
6
+ import { withProfile } from "./test-helpers.js";
7
+ describe("Settings Persistence", () => {
8
+ let tmpDir;
9
+ beforeEach(() => {
10
+ tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), "sprinkles-test-"));
11
+ });
12
+ afterEach(() => {
13
+ fs.rmSync(tmpDir, {
14
+ recursive: true,
15
+ force: true
16
+ });
17
+ });
18
+ test("SettingsPath returns correct path", () => {
19
+ const result = Sprinkle.SettingsPath("/some/path");
20
+ expect(result).toBe(`/some/path${path.sep}settings.json`);
21
+ });
22
+ test("saveSettings creates directory if missing", () => {
23
+ const nestedDir = path.join(tmpDir, "nested", "deep");
24
+ const schema = Type.Object({
25
+ name: Type.String()
26
+ });
27
+ const sprinkle = withProfile(new Sprinkle(schema, nestedDir));
28
+ sprinkle.settings = {
29
+ name: "test"
30
+ };
31
+ sprinkle.saveSettings();
32
+ expect(fs.existsSync(path.join(nestedDir, "profiles", "test.json"))).toBe(true);
33
+ });
34
+ test("saveSettings writes valid JSON with profile format", () => {
35
+ const schema = Type.Object({
36
+ name: Type.String()
37
+ });
38
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir));
39
+ sprinkle.settings = {
40
+ name: "hello"
41
+ };
42
+ sprinkle.saveSettings();
43
+ const content = fs.readFileSync(path.join(tmpDir, "profiles", "test.json"), "utf-8");
44
+ const parsed = JSON.parse(content);
45
+ expect(parsed.settings.name).toBe("hello");
46
+ expect(parsed.defaults).toEqual({});
47
+ expect(parsed.meta.name).toBe("Test");
48
+ });
49
+ test("saveSettings serializes BigInt values", () => {
50
+ const schema = Type.Object({
51
+ amount: Type.BigInt()
52
+ });
53
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir));
54
+ sprinkle.settings = {
55
+ amount: 42n
56
+ };
57
+ sprinkle.saveSettings();
58
+ const content = fs.readFileSync(path.join(tmpDir, "profiles", "test.json"), "utf-8");
59
+ const raw = JSON.parse(content);
60
+ expect(raw.settings.amount).toBe("42n");
61
+ });
62
+ test("saveSettings and loadProfile round-trip with BigInt", async () => {
63
+ const schema = Type.Object({
64
+ amount: Type.BigInt()
65
+ });
66
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir));
67
+ sprinkle.settings = {
68
+ amount: 99n
69
+ };
70
+ sprinkle.saveSettings();
71
+ const sprinkle2 = new Sprinkle(schema, tmpDir);
72
+ await sprinkle2.loadProfile("test");
73
+ expect(sprinkle2.settings).toEqual({
74
+ amount: 99n
75
+ });
76
+ });
77
+ test("saveSettings preserves defaults", () => {
78
+ const schema = Type.Object({
79
+ name: Type.String()
80
+ });
81
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir));
82
+ sprinkle.settings = {
83
+ name: "test"
84
+ };
85
+ sprinkle.defaults = {
86
+ string: "last-input"
87
+ };
88
+ sprinkle.saveSettings();
89
+ const content = fs.readFileSync(path.join(tmpDir, "profiles", "test.json"), "utf-8");
90
+ const parsed = JSON.parse(content);
91
+ expect(parsed.defaults.string).toBe("last-input");
92
+ });
93
+ test("loadProfile restores defaults", async () => {
94
+ const schema = Type.Object({
95
+ name: Type.String()
96
+ });
97
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir));
98
+ sprinkle.settings = {
99
+ name: "test"
100
+ };
101
+ sprinkle.defaults = {
102
+ string: "remembered"
103
+ };
104
+ sprinkle.saveSettings();
105
+ const sprinkle2 = new Sprinkle(schema, tmpDir);
106
+ await sprinkle2.loadProfile("test");
107
+ expect(sprinkle2.defaults).toEqual({
108
+ string: "remembered"
109
+ });
110
+ });
111
+ test("round-trip with nested objects", async () => {
112
+ const schema = Type.Object({
113
+ user: Type.Object({
114
+ name: Type.String(),
115
+ score: Type.BigInt()
116
+ })
117
+ });
118
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir));
119
+ sprinkle.settings = {
120
+ user: {
121
+ name: "alice",
122
+ score: 100n
123
+ }
124
+ };
125
+ sprinkle.saveSettings();
126
+ const sprinkle2 = new Sprinkle(schema, tmpDir);
127
+ await sprinkle2.loadProfile("test");
128
+ expect(sprinkle2.settings).toEqual({
129
+ user: {
130
+ name: "alice",
131
+ score: 100n
132
+ }
133
+ });
134
+ });
135
+ test("scanProfiles returns profile entries", () => {
136
+ const schema = Type.Object({
137
+ name: Type.String()
138
+ });
139
+ const sprinkle = withProfile(new Sprinkle(schema, tmpDir), "alice");
140
+ sprinkle.profileMeta.name = "Alice";
141
+ sprinkle.settings = {
142
+ name: "test"
143
+ };
144
+ sprinkle.saveSettings();
145
+ const profiles = sprinkle.scanProfiles();
146
+ expect(profiles.length).toBe(1);
147
+ expect(profiles[0].id).toBe("alice");
148
+ expect(profiles[0].meta.name).toBe("Alice");
149
+ });
150
+ test("sanitizeProfileId handles various inputs", () => {
151
+ expect(Sprinkle.sanitizeProfileId("Mainnet Alice")).toBe("mainnet-alice");
152
+ expect(Sprinkle.sanitizeProfileId("test--profile")).toBe("test-profile");
153
+ expect(Sprinkle.sanitizeProfileId("Hello World!@#$")).toBe("hello-world");
154
+ expect(Sprinkle.sanitizeProfileId(" ")).toBe("profile");
155
+ expect(Sprinkle.sanitizeProfileId("simple")).toBe("simple");
156
+ });
157
+ test("migration converts legacy settings.json to profile", async () => {
158
+ const schema = Type.Object({
159
+ name: Type.String()
160
+ });
161
+
162
+ // Write legacy settings.json
163
+ fs.mkdirSync(tmpDir, {
164
+ recursive: true
165
+ });
166
+ fs.writeFileSync(path.join(tmpDir, "settings.json"), JSON.stringify({
167
+ settings: {
168
+ name: "legacy"
169
+ },
170
+ defaults: {
171
+ string: "old"
172
+ }
173
+ }));
174
+
175
+ // Create sprinkle and trigger migration
176
+ const sprinkle = new Sprinkle(schema, tmpDir);
177
+ await sprinkle.migrateIfNeeded();
178
+
179
+ // Check profile was created
180
+ expect(fs.existsSync(path.join(tmpDir, "profiles", "default.json"))).toBe(true);
181
+ // Check legacy file was backed up
182
+ expect(fs.existsSync(path.join(tmpDir, "settings.json.bak"))).toBe(true);
183
+ expect(fs.existsSync(path.join(tmpDir, "settings.json"))).toBe(false);
184
+
185
+ // Load the migrated profile
186
+ await sprinkle.loadProfile("default");
187
+ expect(sprinkle.settings).toEqual({
188
+ name: "legacy"
189
+ });
190
+ expect(sprinkle.defaults).toEqual({
191
+ string: "old"
192
+ });
193
+ expect(sprinkle.profileMeta.name).toBe("Default");
194
+ });
195
+ });
196
+ //# sourceMappingURL=settings-persistence.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"settings-persistence.test.js","names":["describe","expect","test","beforeEach","afterEach","Sprinkle","Type","fs","path","os","withProfile","tmpDir","mkdtempSync","join","tmpdir","rmSync","recursive","force","result","SettingsPath","toBe","sep","nestedDir","schema","Object","name","String","sprinkle","settings","saveSettings","existsSync","content","readFileSync","parsed","JSON","parse","defaults","toEqual","meta","amount","BigInt","raw","sprinkle2","loadProfile","string","user","score","profileMeta","profiles","scanProfiles","length","id","sanitizeProfileId","mkdirSync","writeFileSync","stringify","migrateIfNeeded"],"sources":["../../../../src/Sprinkle/__tests__/settings-persistence.test.ts"],"sourcesContent":["import { describe, expect, test, beforeEach, afterEach } from \"bun:test\";\nimport { Sprinkle, Type } from \"../index.js\";\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\nimport { withProfile } from \"./test-helpers.js\";\n\ndescribe(\"Settings Persistence\", () => {\n let tmpDir: string;\n\n beforeEach(() => {\n tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), \"sprinkles-test-\"));\n });\n\n afterEach(() => {\n fs.rmSync(tmpDir, { recursive: true, force: true });\n });\n\n test(\"SettingsPath returns correct path\", () => {\n const result = Sprinkle.SettingsPath(\"/some/path\");\n expect(result).toBe(`/some/path${path.sep}settings.json`);\n });\n\n test(\"saveSettings creates directory if missing\", () => {\n const nestedDir = path.join(tmpDir, \"nested\", \"deep\");\n const schema = Type.Object({ name: Type.String() });\n const sprinkle = withProfile(new Sprinkle(schema, nestedDir));\n sprinkle.settings = { name: \"test\" } as any;\n\n sprinkle.saveSettings();\n\n expect(fs.existsSync(path.join(nestedDir, \"profiles\", \"test.json\"))).toBe(\n true,\n );\n });\n\n test(\"saveSettings writes valid JSON with profile format\", () => {\n const schema = Type.Object({ name: Type.String() });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir));\n sprinkle.settings = { name: \"hello\" } as any;\n\n sprinkle.saveSettings();\n\n const content = fs.readFileSync(\n path.join(tmpDir, \"profiles\", \"test.json\"),\n \"utf-8\",\n );\n const parsed = JSON.parse(content);\n expect(parsed.settings.name).toBe(\"hello\");\n expect(parsed.defaults).toEqual({});\n expect(parsed.meta.name).toBe(\"Test\");\n });\n\n test(\"saveSettings serializes BigInt values\", () => {\n const schema = Type.Object({ amount: Type.BigInt() });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir));\n sprinkle.settings = { amount: 42n } as any;\n\n sprinkle.saveSettings();\n\n const content = fs.readFileSync(\n path.join(tmpDir, \"profiles\", \"test.json\"),\n \"utf-8\",\n );\n const raw = JSON.parse(content);\n expect(raw.settings.amount).toBe(\"42n\");\n });\n\n test(\"saveSettings and loadProfile round-trip with BigInt\", async () => {\n const schema = Type.Object({ amount: Type.BigInt() });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir));\n sprinkle.settings = { amount: 99n } as any;\n sprinkle.saveSettings();\n\n const sprinkle2 = new Sprinkle(schema, tmpDir);\n await sprinkle2.loadProfile(\"test\");\n\n expect(sprinkle2.settings).toEqual({ amount: 99n });\n });\n\n test(\"saveSettings preserves defaults\", () => {\n const schema = Type.Object({ name: Type.String() });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir));\n sprinkle.settings = { name: \"test\" } as any;\n sprinkle.defaults = { string: \"last-input\" };\n\n sprinkle.saveSettings();\n\n const content = fs.readFileSync(\n path.join(tmpDir, \"profiles\", \"test.json\"),\n \"utf-8\",\n );\n const parsed = JSON.parse(content);\n expect(parsed.defaults.string).toBe(\"last-input\");\n });\n\n test(\"loadProfile restores defaults\", async () => {\n const schema = Type.Object({ name: Type.String() });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir));\n sprinkle.settings = { name: \"test\" } as any;\n sprinkle.defaults = { string: \"remembered\" };\n sprinkle.saveSettings();\n\n const sprinkle2 = new Sprinkle(schema, tmpDir);\n await sprinkle2.loadProfile(\"test\");\n\n expect(sprinkle2.defaults).toEqual({ string: \"remembered\" });\n });\n\n test(\"round-trip with nested objects\", async () => {\n const schema = Type.Object({\n user: Type.Object({\n name: Type.String(),\n score: Type.BigInt(),\n }),\n });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir));\n sprinkle.settings = { user: { name: \"alice\", score: 100n } } as any;\n sprinkle.saveSettings();\n\n const sprinkle2 = new Sprinkle(schema, tmpDir);\n await sprinkle2.loadProfile(\"test\");\n\n expect(sprinkle2.settings).toEqual({\n user: { name: \"alice\", score: 100n },\n });\n });\n\n test(\"scanProfiles returns profile entries\", () => {\n const schema = Type.Object({ name: Type.String() });\n const sprinkle = withProfile(new Sprinkle(schema, tmpDir), \"alice\");\n sprinkle.profileMeta.name = \"Alice\";\n sprinkle.settings = { name: \"test\" } as any;\n sprinkle.saveSettings();\n\n const profiles = sprinkle.scanProfiles();\n expect(profiles.length).toBe(1);\n expect(profiles[0]!.id).toBe(\"alice\");\n expect(profiles[0]!.meta.name).toBe(\"Alice\");\n });\n\n test(\"sanitizeProfileId handles various inputs\", () => {\n expect(Sprinkle.sanitizeProfileId(\"Mainnet Alice\")).toBe(\"mainnet-alice\");\n expect(Sprinkle.sanitizeProfileId(\"test--profile\")).toBe(\"test-profile\");\n expect(Sprinkle.sanitizeProfileId(\"Hello World!@#$\")).toBe(\"hello-world\");\n expect(Sprinkle.sanitizeProfileId(\" \")).toBe(\"profile\");\n expect(Sprinkle.sanitizeProfileId(\"simple\")).toBe(\"simple\");\n });\n\n test(\"migration converts legacy settings.json to profile\", async () => {\n const schema = Type.Object({ name: Type.String() });\n\n // Write legacy settings.json\n fs.mkdirSync(tmpDir, { recursive: true });\n fs.writeFileSync(\n path.join(tmpDir, \"settings.json\"),\n JSON.stringify({\n settings: { name: \"legacy\" },\n defaults: { string: \"old\" },\n }),\n );\n\n // Create sprinkle and trigger migration\n const sprinkle = new Sprinkle(schema, tmpDir);\n await (sprinkle as any).migrateIfNeeded();\n\n // Check profile was created\n expect(fs.existsSync(path.join(tmpDir, \"profiles\", \"default.json\"))).toBe(\n true,\n );\n // Check legacy file was backed up\n expect(fs.existsSync(path.join(tmpDir, \"settings.json.bak\"))).toBe(true);\n expect(fs.existsSync(path.join(tmpDir, \"settings.json\"))).toBe(false);\n\n // Load the migrated profile\n await sprinkle.loadProfile(\"default\");\n expect(sprinkle.settings).toEqual({ name: \"legacy\" });\n expect(sprinkle.defaults).toEqual({ string: \"old\" });\n expect(sprinkle.profileMeta.name).toBe(\"Default\");\n });\n});\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,106 @@
1
+ import { describe, expect, test, mock, beforeEach } from "bun:test";
2
+ import { Sprinkle, Type } from "../index.js";
3
+ import { withProfile } from "./test-helpers.js";
4
+ const mockSelect = mock();
5
+ const mockInput = mock();
6
+ mock.module("@inquirer/prompts", () => ({
7
+ select: mockSelect,
8
+ input: mockInput
9
+ }));
10
+ describe("showMenu", () => {
11
+ let sprinkle;
12
+ beforeEach(() => {
13
+ const schema = Type.Object({
14
+ name: Type.String()
15
+ });
16
+ sprinkle = withProfile(new Sprinkle(schema, "/tmp/test"));
17
+ sprinkle.settings = {
18
+ name: "test"
19
+ };
20
+ mockSelect.mockClear();
21
+ mockInput.mockClear();
22
+ });
23
+ test("exits when Exit is selected on main menu", async () => {
24
+ mockSelect.mockResolvedValueOnce(-1);
25
+ const menu = {
26
+ title: "Test Menu",
27
+ items: [{
28
+ title: "Action 1",
29
+ action: async () => {}
30
+ }]
31
+ };
32
+ await sprinkle.showMenu(menu);
33
+ });
34
+ test("executes action and re-shows menu", async () => {
35
+ const actionFn = mock(async () => {});
36
+ mockSelect.mockResolvedValueOnce(0).mockResolvedValueOnce(-1);
37
+ const menu = {
38
+ title: "Test Menu",
39
+ items: [{
40
+ title: "My Action",
41
+ action: actionFn
42
+ }]
43
+ };
44
+ await sprinkle.showMenu(menu);
45
+ expect(actionFn).toHaveBeenCalledTimes(1);
46
+ });
47
+ test("main menu includes Settings & Profiles submenu and Exit", async () => {
48
+ mockSelect.mockResolvedValueOnce(-1);
49
+ const menu = {
50
+ title: "Test",
51
+ items: [{
52
+ title: "Action",
53
+ action: async () => {}
54
+ }]
55
+ };
56
+ await sprinkle.showMenu(menu);
57
+ const choices = mockSelect.mock.calls[0][0].choices;
58
+ const names = choices.map(c => c.name);
59
+ expect(names).toContain("Settings & Profiles");
60
+ expect(names).toContain("Exit");
61
+ expect(names).not.toContain("Switch profile");
62
+ expect(names).not.toContain("Manage profiles");
63
+ expect(names).not.toContain("Edit settings");
64
+ });
65
+ test("submenu includes Back instead of Exit", async () => {
66
+ mockSelect.mockResolvedValueOnce(0).mockResolvedValueOnce(-1).mockResolvedValueOnce(-1);
67
+ const menu = {
68
+ title: "Main",
69
+ items: [{
70
+ title: "Sub",
71
+ items: [{
72
+ title: "Sub Action",
73
+ action: async () => {}
74
+ }]
75
+ }]
76
+ };
77
+ await sprinkle.showMenu(menu);
78
+ const subChoices = mockSelect.mock.calls[1][0].choices;
79
+ const subNames = subChoices.map(c => c.name);
80
+ expect(subNames).toContain("Back");
81
+ expect(subNames).not.toContain("Exit");
82
+ });
83
+ test("action returning sprinkle instance saves settings", async () => {
84
+ sprinkle.settings = {
85
+ name: "original"
86
+ };
87
+ mockSelect.mockResolvedValueOnce(0).mockResolvedValueOnce(-1);
88
+ const menu = {
89
+ title: "Test",
90
+ items: [{
91
+ title: "Update",
92
+ action: async s => {
93
+ s.settings = {
94
+ name: "updated"
95
+ };
96
+ return s;
97
+ }
98
+ }]
99
+ };
100
+ await sprinkle.showMenu(menu);
101
+ expect(sprinkle.settings).toEqual({
102
+ name: "updated"
103
+ });
104
+ });
105
+ });
106
+ //# sourceMappingURL=show-menu.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"show-menu.test.js","names":["describe","expect","test","mock","beforeEach","Sprinkle","Type","withProfile","mockSelect","mockInput","module","select","input","sprinkle","schema","Object","name","String","settings","mockClear","mockResolvedValueOnce","menu","title","items","action","showMenu","actionFn","toHaveBeenCalledTimes","choices","calls","names","map","c","toContain","not","subChoices","subNames","s","toEqual"],"sources":["../../../../src/Sprinkle/__tests__/show-menu.test.ts"],"sourcesContent":["import { describe, expect, test, mock, beforeEach } from \"bun:test\";\nimport { Sprinkle, Type, type IMenu } from \"../index.js\";\nimport { withProfile } from \"./test-helpers.js\";\n\nconst mockSelect = mock();\nconst mockInput = mock();\n\nmock.module(\"@inquirer/prompts\", () => ({\n select: mockSelect,\n input: mockInput,\n}));\n\ndescribe(\"showMenu\", () => {\n let sprinkle: Sprinkle<any>;\n\n beforeEach(() => {\n const schema = Type.Object({ name: Type.String() });\n sprinkle = withProfile(new Sprinkle(schema, \"/tmp/test\"));\n sprinkle.settings = { name: \"test\" } as any;\n mockSelect.mockClear();\n mockInput.mockClear();\n });\n\n test(\"exits when Exit is selected on main menu\", async () => {\n mockSelect.mockResolvedValueOnce(-1);\n\n const menu: IMenu<any> = {\n title: \"Test Menu\",\n items: [\n {\n title: \"Action 1\",\n action: async () => {},\n },\n ],\n };\n\n await sprinkle.showMenu(menu);\n });\n\n test(\"executes action and re-shows menu\", async () => {\n const actionFn = mock(async () => {});\n\n mockSelect.mockResolvedValueOnce(0).mockResolvedValueOnce(-1);\n\n const menu: IMenu<any> = {\n title: \"Test Menu\",\n items: [\n {\n title: \"My Action\",\n action: actionFn,\n },\n ],\n };\n\n await sprinkle.showMenu(menu);\n expect(actionFn).toHaveBeenCalledTimes(1);\n });\n\n test(\"main menu includes Settings & Profiles submenu and Exit\", async () => {\n mockSelect.mockResolvedValueOnce(-1);\n\n const menu: IMenu<any> = {\n title: \"Test\",\n items: [{ title: \"Action\", action: async () => {} }],\n };\n\n await sprinkle.showMenu(menu);\n\n const choices = mockSelect.mock.calls[0][0].choices;\n const names = choices.map((c: any) => c.name);\n expect(names).toContain(\"Settings & Profiles\");\n expect(names).toContain(\"Exit\");\n expect(names).not.toContain(\"Switch profile\");\n expect(names).not.toContain(\"Manage profiles\");\n expect(names).not.toContain(\"Edit settings\");\n });\n\n test(\"submenu includes Back instead of Exit\", async () => {\n mockSelect\n .mockResolvedValueOnce(0)\n .mockResolvedValueOnce(-1)\n .mockResolvedValueOnce(-1);\n\n const menu: IMenu<any> = {\n title: \"Main\",\n items: [\n {\n title: \"Sub\",\n items: [{ title: \"Sub Action\", action: async () => {} }],\n },\n ],\n };\n\n await sprinkle.showMenu(menu);\n\n const subChoices = mockSelect.mock.calls[1][0].choices;\n const subNames = subChoices.map((c: any) => c.name);\n expect(subNames).toContain(\"Back\");\n expect(subNames).not.toContain(\"Exit\");\n });\n\n test(\"action returning sprinkle instance saves settings\", async () => {\n sprinkle.settings = { name: \"original\" } as any;\n\n mockSelect.mockResolvedValueOnce(0).mockResolvedValueOnce(-1);\n\n const menu: IMenu<any> = {\n title: \"Test\",\n items: [\n {\n title: \"Update\",\n action: async (s: Sprinkle<any>) => {\n s.settings = { name: \"updated\" } as any;\n return s;\n },\n },\n ],\n };\n\n await sprinkle.showMenu(menu);\n expect(sprinkle.settings).toEqual({ name: \"updated\" });\n });\n});\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,10 @@
1
+ export function withProfile(sprinkle, id = "test") {
2
+ sprinkle.profileId = id;
3
+ sprinkle.profileMeta = {
4
+ name: "Test",
5
+ createdAt: new Date().toISOString(),
6
+ updatedAt: new Date().toISOString()
7
+ };
8
+ return sprinkle;
9
+ }
10
+ //# sourceMappingURL=test-helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test-helpers.js","names":["withProfile","sprinkle","id","profileId","profileMeta","name","createdAt","Date","toISOString","updatedAt"],"sources":["../../../../src/Sprinkle/__tests__/test-helpers.ts"],"sourcesContent":["import { Sprinkle } from \"../index.js\";\n\nexport function withProfile<S>(\n sprinkle: Sprinkle<S>,\n id: string = \"test\",\n): Sprinkle<S> {\n sprinkle.profileId = id;\n sprinkle.profileMeta = {\n name: \"Test\",\n createdAt: new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n };\n return sprinkle;\n}\n"],"mappings":"AAEA,OAAO,SAASA,WAAWA,CACzBC,QAAqB,EACrBC,EAAU,GAAG,MAAM,EACN;EACbD,QAAQ,CAACE,SAAS,GAAGD,EAAE;EACvBD,QAAQ,CAACG,WAAW,GAAG;IACrBC,IAAI,EAAE,MAAM;IACZC,SAAS,EAAE,IAAIC,IAAI,CAAC,CAAC,CAACC,WAAW,CAAC,CAAC;IACnCC,SAAS,EAAE,IAAIF,IAAI,CAAC,CAAC,CAACC,WAAW,CAAC;EACpC,CAAC;EACD,OAAOP,QAAQ;AACjB","ignoreList":[]}
@@ -0,0 +1,269 @@
1
+ import { describe, test, expect, mock, beforeEach } from "bun:test";
2
+ import { Type } from "@sinclair/typebox";
3
+
4
+ // Track select calls
5
+ let selectCalls = [];
6
+ let selectResponses = [];
7
+
8
+ // Mock @inquirer/prompts
9
+ mock.module("@inquirer/prompts", () => ({
10
+ input: mock(async () => "test"),
11
+ password: mock(async () => "secret"),
12
+ select: mock(async opts => {
13
+ selectCalls.push(opts);
14
+ const response = selectResponses.shift();
15
+ if (response === undefined) {
16
+ // Default: return -1 (Back/Exit)
17
+ return -1;
18
+ }
19
+ return response;
20
+ }),
21
+ search: mock(async () => "result")
22
+ }));
23
+
24
+ // Mock clipboardy
25
+ let clipboardContent = "";
26
+ let clipboardShouldFail = false;
27
+ mock.module("clipboardy", () => ({
28
+ default: {
29
+ writeSync: content => {
30
+ if (clipboardShouldFail) throw new Error("clipboard not available");
31
+ clipboardContent = content;
32
+ }
33
+ }
34
+ }));
35
+ const {
36
+ Sprinkle
37
+ } = await import("../../index.js");
38
+ const TestSchema = Type.Object({
39
+ name: Type.String()
40
+ });
41
+ describe("TxDialog (2.6)", () => {
42
+ beforeEach(() => {
43
+ selectCalls = [];
44
+ selectResponses = [];
45
+ clipboardContent = "";
46
+ clipboardShouldFail = false;
47
+ });
48
+ test("shows truncated CBOR by default", async () => {
49
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog");
50
+ const logs = [];
51
+ const origLog = console.log;
52
+ console.log = (...args) => logs.push(args.join(" "));
53
+
54
+ // Just exit immediately
55
+ selectResponses = [-1];
56
+ const mockTx = {
57
+ toCbor: () => "a".repeat(100),
58
+ body: () => ({
59
+ requiredSigners: () => null,
60
+ certs: () => null,
61
+ withdrawals: () => null
62
+ })
63
+ };
64
+ const mockBlaze = {
65
+ wallet: {
66
+ getUsedAddresses: async () => []
67
+ }
68
+ };
69
+ await sprinkle.TxDialog(mockBlaze, mockTx);
70
+ console.log = origLog;
71
+
72
+ // Should show truncated CBOR (50 chars + ...)
73
+ const cborLog = logs.find(l => l.includes("Transaction CBOR:"));
74
+ expect(cborLog).toBeDefined();
75
+ expect(cborLog.length).toBeLessThan(100);
76
+ expect(cborLog).toContain("...");
77
+ });
78
+ test("menu items for non-HotWallet include Expand, Copy, no Sign", async () => {
79
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog2");
80
+ const origLog = console.log;
81
+ console.log = () => {};
82
+ selectResponses = [-1];
83
+ const mockTx = {
84
+ toCbor: () => "deadbeef",
85
+ body: () => ({
86
+ requiredSigners: () => null,
87
+ certs: () => null,
88
+ withdrawals: () => null
89
+ })
90
+ };
91
+ const mockBlaze = {
92
+ wallet: {
93
+ getUsedAddresses: async () => []
94
+ }
95
+ };
96
+ await sprinkle.TxDialog(mockBlaze, mockTx);
97
+ console.log = origLog;
98
+
99
+ // Check the select call had Expand and Copy but not Sign
100
+ const menuCall = selectCalls[0];
101
+ expect(menuCall).toBeDefined();
102
+ const choiceNames = menuCall.choices.map(c => c.name);
103
+ expect(choiceNames).toContain("Expand CBOR");
104
+ expect(choiceNames).toContain("Copy CBOR to clipboard");
105
+ expect(choiceNames).not.toContain("Sign and submit transaction");
106
+ });
107
+ test("menu items for HotWallet include Sign", async () => {
108
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog3");
109
+ const origLog = console.log;
110
+ console.log = () => {};
111
+
112
+ // Need to import HotWallet to use instanceof
113
+ const {
114
+ HotWallet
115
+ } = await import("@blaze-cardano/sdk");
116
+ selectResponses = [-1];
117
+ const mockTx = {
118
+ toCbor: () => "deadbeef",
119
+ body: () => ({
120
+ requiredSigners: () => null,
121
+ certs: () => null,
122
+ withdrawals: () => null
123
+ })
124
+ };
125
+
126
+ // Create a mock that passes instanceof HotWallet check
127
+ // We need to use Object.create to set the prototype
128
+ const hotWalletProto = HotWallet.prototype;
129
+ const mockWallet = Object.create(hotWalletProto);
130
+ mockWallet.getUsedAddresses = async () => [];
131
+ const mockBlaze = {
132
+ wallet: mockWallet
133
+ };
134
+ await sprinkle.TxDialog(mockBlaze, mockTx);
135
+ console.log = origLog;
136
+ const menuCall = selectCalls[0];
137
+ expect(menuCall).toBeDefined();
138
+ const choiceNames = menuCall.choices.map(c => c.name);
139
+ expect(choiceNames).toContain("Sign and submit transaction");
140
+ });
141
+ test("copy CBOR to clipboard works", async () => {
142
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog4");
143
+ const origLog = console.log;
144
+ const logs = [];
145
+ console.log = (...args) => logs.push(args.join(" "));
146
+
147
+ // Select "Copy CBOR to clipboard" (index 1), then Back (-1)
148
+ selectResponses = [1, -1];
149
+ const txCbor = "deadbeefcafebabe";
150
+ const mockTx = {
151
+ toCbor: () => txCbor,
152
+ body: () => ({
153
+ requiredSigners: () => null,
154
+ certs: () => null,
155
+ withdrawals: () => null
156
+ })
157
+ };
158
+ const mockBlaze = {
159
+ wallet: {
160
+ getUsedAddresses: async () => []
161
+ }
162
+ };
163
+ await sprinkle.TxDialog(mockBlaze, mockTx);
164
+ console.log = origLog;
165
+ expect(clipboardContent).toBe(txCbor);
166
+ expect(logs.some(l => l.includes("copied to clipboard"))).toBe(true);
167
+ });
168
+ test("clipboard failure falls back to expanded view", async () => {
169
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog5");
170
+ const origLog = console.log;
171
+ const logs = [];
172
+ console.log = (...args) => logs.push(args.join(" "));
173
+ clipboardShouldFail = true;
174
+
175
+ // Select "Copy CBOR to clipboard" (index 1),
176
+ // It will fail and call showDialog again with expanded=true
177
+ // Then Back (-1) from the expanded menu, then Back from outer
178
+ selectResponses = [1, -1, -1];
179
+ const txCbor = "a".repeat(100);
180
+ const mockTx = {
181
+ toCbor: () => txCbor,
182
+ body: () => ({
183
+ requiredSigners: () => null,
184
+ certs: () => null,
185
+ withdrawals: () => null
186
+ })
187
+ };
188
+ const mockBlaze = {
189
+ wallet: {
190
+ getUsedAddresses: async () => []
191
+ }
192
+ };
193
+ await sprinkle.TxDialog(mockBlaze, mockTx);
194
+ console.log = origLog;
195
+ expect(logs.some(l => l.includes("Failed to copy"))).toBe(true);
196
+ // After failure it should show the full CBOR without truncation
197
+ expect(logs.some(l => l.includes(txCbor) && !l.includes("..."))).toBe(true);
198
+ });
199
+ test("expand CBOR shows full content", async () => {
200
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog6");
201
+ const origLog = console.log;
202
+ const logs = [];
203
+ console.log = (...args) => logs.push(args.join(" "));
204
+
205
+ // Select "Expand CBOR" (index 0), then Back (-1) from expanded, then Back from outer
206
+ selectResponses = [0, -1, -1];
207
+ const txCbor = "b".repeat(100);
208
+ const mockTx = {
209
+ toCbor: () => txCbor,
210
+ body: () => ({
211
+ requiredSigners: () => null,
212
+ certs: () => null,
213
+ withdrawals: () => null
214
+ })
215
+ };
216
+ const mockBlaze = {
217
+ wallet: {
218
+ getUsedAddresses: async () => []
219
+ }
220
+ };
221
+ await sprinkle.TxDialog(mockBlaze, mockTx);
222
+ console.log = origLog;
223
+
224
+ // First call shows truncated, second shows full
225
+ const truncatedLog = logs.find(l => l.includes("...") && l.includes("Transaction CBOR:"));
226
+ const fullLog = logs.find(l => l.includes(txCbor) && !l.includes("..."));
227
+ expect(truncatedLog).toBeDefined();
228
+ expect(fullLog).toBeDefined();
229
+ });
230
+ test("beforeSign hook is called before signing", async () => {
231
+ const sprinkle = new Sprinkle(TestSchema, "/tmp/test-txdialog7");
232
+ const origLog = console.log;
233
+ const origWarn = console.warn;
234
+ console.log = () => {};
235
+ console.warn = () => {};
236
+ const {
237
+ HotWallet
238
+ } = await import("@blaze-cardano/sdk");
239
+ let beforeSignCalled = false;
240
+
241
+ // Select "Sign and submit" (index 2 - after Expand and Copy), then Back
242
+ selectResponses = [2, -1];
243
+ const mockTx = {
244
+ toCbor: () => "deadbeef",
245
+ body: () => ({
246
+ requiredSigners: () => null,
247
+ certs: () => null,
248
+ withdrawals: () => null
249
+ })
250
+ };
251
+ const hotWalletProto = HotWallet.prototype;
252
+ const mockWallet = Object.create(hotWalletProto);
253
+ mockWallet.getUsedAddresses = async () => [];
254
+ const mockBlaze = {
255
+ wallet: mockWallet,
256
+ signTransaction: async tx => tx,
257
+ submitTransaction: async () => "txhash123"
258
+ };
259
+ await sprinkle.TxDialog(mockBlaze, mockTx, {
260
+ beforeSign: async () => {
261
+ beforeSignCalled = true;
262
+ }
263
+ });
264
+ console.log = origLog;
265
+ console.warn = origWarn;
266
+ expect(beforeSignCalled).toBe(true);
267
+ });
268
+ });
269
+ //# sourceMappingURL=tx-dialog.test.js.map