@sundaeswap/sprinkles 0.5.0 → 0.6.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.
- package/dist/cjs/Sprinkle/__tests__/encryption.test.js +3 -1
- package/dist/cjs/Sprinkle/__tests__/encryption.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/enhancements.test.js +3 -37
- package/dist/cjs/Sprinkle/__tests__/enhancements.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/field-utils.test.js +170 -0
- package/dist/cjs/Sprinkle/__tests__/field-utils.test.js.map +1 -0
- package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js +377 -84
- package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/formatting.test.js +97 -0
- package/dist/cjs/Sprinkle/__tests__/formatting.test.js.map +1 -0
- package/dist/cjs/Sprinkle/__tests__/show-menu.test.js +9 -5
- package/dist/cjs/Sprinkle/__tests__/show-menu.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js +9 -0
- package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js.map +1 -1
- package/dist/cjs/Sprinkle/index.js +174 -94
- package/dist/cjs/Sprinkle/index.js.map +1 -1
- package/dist/cjs/Sprinkle/menus/array-menu.js +195 -0
- package/dist/cjs/Sprinkle/menus/array-menu.js.map +1 -0
- package/dist/cjs/Sprinkle/menus/field-menu.js +161 -0
- package/dist/cjs/Sprinkle/menus/field-menu.js.map +1 -0
- package/dist/cjs/Sprinkle/menus/index.js +33 -0
- package/dist/cjs/Sprinkle/menus/index.js.map +1 -0
- package/dist/cjs/Sprinkle/menus/object-menu.js +324 -0
- package/dist/cjs/Sprinkle/menus/object-menu.js.map +1 -0
- package/dist/cjs/Sprinkle/prompts.js +68 -2
- package/dist/cjs/Sprinkle/prompts.js.map +1 -1
- package/dist/cjs/Sprinkle/type-guards.js +48 -1
- package/dist/cjs/Sprinkle/type-guards.js.map +1 -1
- package/dist/cjs/Sprinkle/types.js +24 -0
- package/dist/cjs/Sprinkle/types.js.map +1 -1
- package/dist/cjs/Sprinkle/utils/field-utils.js +154 -0
- package/dist/cjs/Sprinkle/utils/field-utils.js.map +1 -0
- package/dist/cjs/Sprinkle/utils/formatting.js +126 -0
- package/dist/cjs/Sprinkle/utils/formatting.js.map +1 -0
- package/dist/cjs/Sprinkle/utils/index.js +56 -0
- package/dist/cjs/Sprinkle/utils/index.js.map +1 -0
- package/dist/esm/Sprinkle/__tests__/encryption.test.js +3 -1
- package/dist/esm/Sprinkle/__tests__/encryption.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/enhancements.test.js +3 -37
- package/dist/esm/Sprinkle/__tests__/enhancements.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/field-utils.test.js +168 -0
- package/dist/esm/Sprinkle/__tests__/field-utils.test.js.map +1 -0
- package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js +378 -85
- package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/formatting.test.js +95 -0
- package/dist/esm/Sprinkle/__tests__/formatting.test.js.map +1 -0
- package/dist/esm/Sprinkle/__tests__/show-menu.test.js +9 -5
- package/dist/esm/Sprinkle/__tests__/show-menu.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js +9 -0
- package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js.map +1 -1
- package/dist/esm/Sprinkle/index.js +141 -96
- package/dist/esm/Sprinkle/index.js.map +1 -1
- package/dist/esm/Sprinkle/menus/array-menu.js +190 -0
- package/dist/esm/Sprinkle/menus/array-menu.js.map +1 -0
- package/dist/esm/Sprinkle/menus/field-menu.js +155 -0
- package/dist/esm/Sprinkle/menus/field-menu.js.map +1 -0
- package/dist/esm/Sprinkle/menus/index.js +8 -0
- package/dist/esm/Sprinkle/menus/index.js.map +1 -0
- package/dist/esm/Sprinkle/menus/object-menu.js +318 -0
- package/dist/esm/Sprinkle/menus/object-menu.js.map +1 -0
- package/dist/esm/Sprinkle/prompts.js +59 -1
- package/dist/esm/Sprinkle/prompts.js.map +1 -1
- package/dist/esm/Sprinkle/type-guards.js +42 -0
- package/dist/esm/Sprinkle/type-guards.js.map +1 -1
- package/dist/esm/Sprinkle/types.js +24 -0
- package/dist/esm/Sprinkle/types.js.map +1 -1
- package/dist/esm/Sprinkle/utils/field-utils.js +145 -0
- package/dist/esm/Sprinkle/utils/field-utils.js.map +1 -0
- package/dist/esm/Sprinkle/utils/formatting.js +118 -0
- package/dist/esm/Sprinkle/utils/formatting.js.map +1 -0
- package/dist/esm/Sprinkle/utils/index.js +7 -0
- package/dist/esm/Sprinkle/utils/index.js.map +1 -0
- package/dist/types/Sprinkle/index.d.ts +9 -3
- package/dist/types/Sprinkle/index.d.ts.map +1 -1
- package/dist/types/Sprinkle/menus/array-menu.d.ts +31 -0
- package/dist/types/Sprinkle/menus/array-menu.d.ts.map +1 -0
- package/dist/types/Sprinkle/menus/field-menu.d.ts +34 -0
- package/dist/types/Sprinkle/menus/field-menu.d.ts.map +1 -0
- package/dist/types/Sprinkle/menus/index.d.ts +10 -0
- package/dist/types/Sprinkle/menus/index.d.ts.map +1 -0
- package/dist/types/Sprinkle/menus/object-menu.d.ts +34 -0
- package/dist/types/Sprinkle/menus/object-menu.d.ts.map +1 -0
- package/dist/types/Sprinkle/prompts.d.ts +25 -0
- package/dist/types/Sprinkle/prompts.d.ts.map +1 -1
- package/dist/types/Sprinkle/type-guards.d.ts +24 -1
- package/dist/types/Sprinkle/type-guards.d.ts.map +1 -1
- package/dist/types/Sprinkle/types.d.ts +53 -0
- package/dist/types/Sprinkle/types.d.ts.map +1 -1
- package/dist/types/Sprinkle/utils/field-utils.d.ts +47 -0
- package/dist/types/Sprinkle/utils/field-utils.d.ts.map +1 -0
- package/dist/types/Sprinkle/utils/formatting.d.ts +30 -0
- package/dist/types/Sprinkle/utils/formatting.d.ts.map +1 -0
- package/dist/types/tsconfig.build.tsbuildinfo +1 -1
- package/package.json +1 -1
- package/src/Sprinkle/__tests__/encryption.test.ts +2 -0
- package/src/Sprinkle/__tests__/enhancements.test.ts +3 -42
- package/src/Sprinkle/__tests__/field-utils.test.ts +191 -0
- package/src/Sprinkle/__tests__/fill-in-struct.test.ts +393 -100
- package/src/Sprinkle/__tests__/formatting.test.ts +115 -0
- package/src/Sprinkle/__tests__/show-menu.test.ts +14 -8
- package/src/Sprinkle/__tests__/tx-dialog.test.ts +9 -0
- package/src/Sprinkle/index.ts +175 -122
- package/src/Sprinkle/menus/array-menu.ts +191 -0
- package/src/Sprinkle/menus/field-menu.ts +145 -0
- package/src/Sprinkle/menus/index.ts +12 -0
- package/src/Sprinkle/menus/object-menu.ts +336 -0
- package/src/Sprinkle/prompts.ts +71 -1
- package/src/Sprinkle/type-guards.ts +42 -0
- package/src/Sprinkle/types.ts +43 -0
- package/src/Sprinkle/utils/field-utils.ts +158 -0
- package/src/Sprinkle/utils/formatting.ts +127 -0
- package/src/Sprinkle/utils/index.ts +17 -0
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { describe, expect, test, mock, beforeEach } from "bun:test";
|
|
2
|
-
import { Sprinkle, Type
|
|
2
|
+
import { Sprinkle, Type } from "../index.js";
|
|
3
3
|
import { UserCancelledError } from "../types.js";
|
|
4
4
|
|
|
5
5
|
// Mock @inquirer/prompts
|
|
@@ -21,7 +21,9 @@ mock.module("../prompts.js", () => ({
|
|
|
21
21
|
selectCancellable: mockSelectCancellable,
|
|
22
22
|
inputCancellable: mockInputCancellable,
|
|
23
23
|
passwordCancellable: mockPasswordCancellable,
|
|
24
|
-
confirmCancellable: mockConfirmCancellable
|
|
24
|
+
confirmCancellable: mockConfirmCancellable,
|
|
25
|
+
searchCancellable: mock(),
|
|
26
|
+
select: mockSelectCancellable
|
|
25
27
|
}));
|
|
26
28
|
describe("FillInStruct", () => {
|
|
27
29
|
let sprinkle;
|
|
@@ -30,15 +32,19 @@ describe("FillInStruct", () => {
|
|
|
30
32
|
placeholder: Type.String()
|
|
31
33
|
});
|
|
32
34
|
sprinkle = new Sprinkle(schema, "/tmp/test");
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
35
|
+
// Use mockReset to clear both call history and queued responses
|
|
36
|
+
mockSelect.mockReset();
|
|
37
|
+
mockInput.mockReset();
|
|
38
|
+
mockPassword.mockReset();
|
|
39
|
+
mockConfirm.mockReset();
|
|
40
|
+
mockSelectCancellable.mockReset();
|
|
41
|
+
mockInputCancellable.mockReset();
|
|
42
|
+
mockPasswordCancellable.mockReset();
|
|
43
|
+
mockConfirmCancellable.mockReset();
|
|
41
44
|
});
|
|
45
|
+
|
|
46
|
+
// --- Primitive types (unchanged behavior) ---
|
|
47
|
+
|
|
42
48
|
test("fills a simple string field", async () => {
|
|
43
49
|
mockInputCancellable.mockResolvedValueOnce("hello");
|
|
44
50
|
const result = await sprinkle.FillInStruct(Type.String());
|
|
@@ -62,18 +68,30 @@ describe("FillInStruct", () => {
|
|
|
62
68
|
expect(mockInputCancellable).not.toHaveBeenCalled();
|
|
63
69
|
expect(mockSelectCancellable).not.toHaveBeenCalled();
|
|
64
70
|
});
|
|
65
|
-
test("
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
});
|
|
70
|
-
mockInputCancellable.mockResolvedValueOnce("Alice").mockResolvedValueOnce("30");
|
|
71
|
-
const result = await sprinkle.FillInStruct(schema);
|
|
72
|
-
expect(result).toEqual({
|
|
73
|
-
name: "Alice",
|
|
74
|
-
age: 30n
|
|
75
|
-
});
|
|
71
|
+
test("uses default value for string", async () => {
|
|
72
|
+
mockInputCancellable.mockResolvedValueOnce("used-default");
|
|
73
|
+
await sprinkle.FillInStruct(Type.String(), "my-default");
|
|
74
|
+
expect(mockInputCancellable.mock.calls[0][0].default).toBe("my-default");
|
|
76
75
|
});
|
|
76
|
+
test("uses default value for bigint", async () => {
|
|
77
|
+
mockInputCancellable.mockResolvedValueOnce("99");
|
|
78
|
+
await sprinkle.FillInStruct(Type.BigInt(), 99n);
|
|
79
|
+
expect(mockInputCancellable.mock.calls[0][0].default).toBe("99");
|
|
80
|
+
});
|
|
81
|
+
test("throws for unsupported types", async () => {
|
|
82
|
+
expect(sprinkle.FillInStruct(Type.Boolean())).rejects.toThrow("Unable to fill in struct");
|
|
83
|
+
});
|
|
84
|
+
test("remembers last string input as default", async () => {
|
|
85
|
+
mockInputCancellable.mockResolvedValueOnce("first-input").mockResolvedValueOnce("second-input");
|
|
86
|
+
await sprinkle.FillInStruct(Type.String());
|
|
87
|
+
expect(sprinkle.defaults["string"]).toBe("first-input");
|
|
88
|
+
await sprinkle.FillInStruct(Type.String());
|
|
89
|
+
// Second call should have the first input as default
|
|
90
|
+
expect(mockInputCancellable.mock.calls[1][0].default).toBe("first-input");
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
// --- Union types ---
|
|
94
|
+
|
|
77
95
|
test("fills a union type by selecting variant then filling", async () => {
|
|
78
96
|
const schema = Type.Union([Type.Object({
|
|
79
97
|
type: Type.Literal("a"),
|
|
@@ -87,43 +105,183 @@ describe("FillInStruct", () => {
|
|
|
87
105
|
mockSelectCancellable.mockImplementationOnce(async opts => {
|
|
88
106
|
return opts.choices[0].value;
|
|
89
107
|
});
|
|
108
|
+
// Object menu: select "value" field, then Submit
|
|
109
|
+
mockSelectCancellable.mockResolvedValueOnce("field:value");
|
|
90
110
|
mockInputCancellable.mockResolvedValueOnce("test-value");
|
|
111
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
91
112
|
const result = await sprinkle.FillInStruct(schema);
|
|
92
113
|
expect(result).toEqual({
|
|
93
114
|
type: "a",
|
|
94
115
|
value: "test-value"
|
|
95
116
|
});
|
|
96
117
|
});
|
|
97
|
-
test("
|
|
98
|
-
const schema = Type.
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
118
|
+
test("throws UserCancelledError when select prompt is cancelled", async () => {
|
|
119
|
+
const schema = Type.Union([Type.Object({
|
|
120
|
+
type: Type.Literal("a")
|
|
121
|
+
}), Type.Object({
|
|
122
|
+
type: Type.Literal("b")
|
|
123
|
+
})]);
|
|
124
|
+
mockSelectCancellable.mockResolvedValueOnce(null);
|
|
125
|
+
await expect(sprinkle.FillInStruct(schema)).rejects.toThrow(UserCancelledError);
|
|
126
|
+
});
|
|
127
|
+
test("discriminated union propagates default when variant matches", async () => {
|
|
128
|
+
// When default has type "a" and user selects variant A, the default values
|
|
129
|
+
// for non-literal fields should be pre-populated (field starts as "set" status).
|
|
130
|
+
// Submitting without editing should return the default values.
|
|
131
|
+
const schema = Type.Union([Type.Object({
|
|
132
|
+
type: Type.Literal("a"),
|
|
133
|
+
value: Type.String()
|
|
134
|
+
}), Type.Object({
|
|
135
|
+
type: Type.Literal("b"),
|
|
136
|
+
count: Type.BigInt()
|
|
137
|
+
})]);
|
|
138
|
+
const defaultValue = {
|
|
139
|
+
type: "a",
|
|
140
|
+
value: "existing-value"
|
|
141
|
+
};
|
|
103
142
|
|
|
104
|
-
|
|
105
|
-
|
|
143
|
+
// Select the first variant (type "a") - matches the default
|
|
144
|
+
mockSelectCancellable.mockImplementationOnce(async opts => {
|
|
145
|
+
return opts.choices[0].value;
|
|
146
|
+
});
|
|
147
|
+
// Both fields are "set" (type auto-filled, value pre-populated from default).
|
|
148
|
+
// allRequiredFieldsFilled is true so Submit is enabled. Submit immediately.
|
|
149
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
150
|
+
const result = await sprinkle.FillInStruct(schema, defaultValue);
|
|
151
|
+
|
|
152
|
+
// The default value for the "value" field should be included in the result
|
|
153
|
+
expect(result).toEqual({
|
|
154
|
+
type: "a",
|
|
155
|
+
value: "existing-value"
|
|
156
|
+
});
|
|
106
157
|
});
|
|
107
|
-
test("
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
158
|
+
test("discriminated union with nested object propagates nested defaults", async () => {
|
|
159
|
+
// When a discriminated union default has a nested settings object, the nested
|
|
160
|
+
// fields should also be pre-populated from the default.
|
|
161
|
+
const schema = Type.Union([Type.Object({
|
|
162
|
+
type: Type.Literal("config"),
|
|
163
|
+
settings: Type.Object({
|
|
164
|
+
timeout: Type.BigInt(),
|
|
165
|
+
retries: Type.BigInt()
|
|
166
|
+
})
|
|
167
|
+
}), Type.Object({
|
|
168
|
+
type: Type.Literal("other"),
|
|
169
|
+
name: Type.String()
|
|
170
|
+
})]);
|
|
171
|
+
const defaultValue = {
|
|
172
|
+
type: "config",
|
|
173
|
+
settings: {
|
|
174
|
+
timeout: 30n,
|
|
175
|
+
retries: 3n
|
|
176
|
+
}
|
|
177
|
+
};
|
|
111
178
|
|
|
112
|
-
|
|
113
|
-
|
|
179
|
+
// Select the first variant (type "config") - matches the default
|
|
180
|
+
mockSelectCancellable.mockImplementationOnce(async opts => {
|
|
181
|
+
return opts.choices[0].value;
|
|
182
|
+
});
|
|
183
|
+
// Outer object: "type" is auto-filled literal, "settings" is pre-populated from default.
|
|
184
|
+
// Both required fields are "set", so Submit is available.
|
|
185
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
186
|
+
const result = await sprinkle.FillInStruct(schema, defaultValue);
|
|
187
|
+
|
|
188
|
+
// Both the outer and nested default values should be returned
|
|
189
|
+
expect(result).toEqual({
|
|
190
|
+
type: "config",
|
|
191
|
+
settings: {
|
|
192
|
+
timeout: 30n,
|
|
193
|
+
retries: 3n
|
|
194
|
+
}
|
|
195
|
+
});
|
|
114
196
|
});
|
|
115
|
-
test("
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
197
|
+
test("discriminated union does not propagate default when variant does not match", async () => {
|
|
198
|
+
// When default has type "a" but user selects variant B, no default should be passed.
|
|
199
|
+
// The variant B field should start as "unset" and require user input.
|
|
200
|
+
const schema = Type.Union([Type.Object({
|
|
201
|
+
type: Type.Literal("a"),
|
|
202
|
+
value: Type.String()
|
|
203
|
+
}), Type.Object({
|
|
204
|
+
type: Type.Literal("b"),
|
|
205
|
+
name: Type.String()
|
|
206
|
+
})]);
|
|
207
|
+
|
|
208
|
+
// Default has type "a" but user selects variant "b"
|
|
209
|
+
const defaultValue = {
|
|
210
|
+
type: "a",
|
|
211
|
+
value: "should-not-appear"
|
|
212
|
+
};
|
|
213
|
+
|
|
214
|
+
// Select the second variant (type "b") - does NOT match the default
|
|
215
|
+
mockSelectCancellable.mockImplementationOnce(async opts => {
|
|
216
|
+
return opts.choices[1].value;
|
|
217
|
+
});
|
|
218
|
+
// Variant B: "type" is auto-filled, "name" starts as unset (no default propagated).
|
|
219
|
+
// Select "name" field and fill it in.
|
|
220
|
+
mockSelectCancellable.mockResolvedValueOnce("field:name");
|
|
221
|
+
mockInputCancellable.mockResolvedValueOnce("new-name");
|
|
222
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
223
|
+
const result = await sprinkle.FillInStruct(schema, defaultValue);
|
|
224
|
+
expect(result).toEqual({
|
|
225
|
+
type: "b",
|
|
226
|
+
name: "new-name"
|
|
227
|
+
});
|
|
228
|
+
// The input prompt for "name" should have no default (default from variant A was not passed)
|
|
229
|
+
expect(mockInputCancellable.mock.calls[0][0].default).toBeUndefined();
|
|
119
230
|
});
|
|
120
|
-
test("
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
231
|
+
test("non-discriminated union propagates default when value structurally matches selected variant", async () => {
|
|
232
|
+
// For a union of simple types, when the default matches the selected variant
|
|
233
|
+
// structurally (via Value.Check), it should be passed through.
|
|
234
|
+
const schema = Type.Union([Type.String(), Type.BigInt()]);
|
|
235
|
+
const defaultValue = "default-string";
|
|
236
|
+
|
|
237
|
+
// Select the first variant (String)
|
|
238
|
+
mockSelectCancellable.mockImplementationOnce(async opts => {
|
|
239
|
+
return opts.choices[0].value;
|
|
240
|
+
});
|
|
241
|
+
mockInputCancellable.mockResolvedValueOnce("new-value");
|
|
242
|
+
await sprinkle.FillInStruct(schema, defaultValue);
|
|
243
|
+
|
|
244
|
+
// The input prompt should show the string default
|
|
245
|
+
expect(mockInputCancellable.mock.calls[0][0].default).toBe("default-string");
|
|
124
246
|
});
|
|
125
|
-
test("
|
|
126
|
-
|
|
247
|
+
test("non-discriminated union does not propagate default when value does not match selected variant", async () => {
|
|
248
|
+
// When a string default is provided but BigInt variant is selected,
|
|
249
|
+
// the default must not be passed (it would fail structural validation).
|
|
250
|
+
const schema = Type.Union([Type.String(), Type.BigInt()]);
|
|
251
|
+
|
|
252
|
+
// Default is a string but user selects BigInt variant
|
|
253
|
+
const defaultValue = "not-a-bigint";
|
|
254
|
+
|
|
255
|
+
// Select the second variant (BigInt)
|
|
256
|
+
mockSelectCancellable.mockImplementationOnce(async opts => {
|
|
257
|
+
return opts.choices[1].value;
|
|
258
|
+
});
|
|
259
|
+
mockInputCancellable.mockResolvedValueOnce("42");
|
|
260
|
+
const result = await sprinkle.FillInStruct(schema, defaultValue);
|
|
261
|
+
expect(result).toBe(42n);
|
|
262
|
+
// The string default must not be passed to the BigInt prompt
|
|
263
|
+
expect(mockInputCancellable.mock.calls[0][0].default).toBeUndefined();
|
|
264
|
+
});
|
|
265
|
+
|
|
266
|
+
// --- Object types (menu-based) ---
|
|
267
|
+
|
|
268
|
+
test("fills an object with multiple fields", async () => {
|
|
269
|
+
const schema = Type.Object({
|
|
270
|
+
name: Type.String(),
|
|
271
|
+
age: Type.BigInt()
|
|
272
|
+
});
|
|
273
|
+
|
|
274
|
+
// Menu flow: select name -> fill -> select age -> fill -> submit
|
|
275
|
+
mockSelectCancellable.mockResolvedValueOnce("field:name");
|
|
276
|
+
mockInputCancellable.mockResolvedValueOnce("Alice");
|
|
277
|
+
mockSelectCancellable.mockResolvedValueOnce("field:age");
|
|
278
|
+
mockInputCancellable.mockResolvedValueOnce("30");
|
|
279
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
280
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
281
|
+
expect(result).toEqual({
|
|
282
|
+
name: "Alice",
|
|
283
|
+
age: 30n
|
|
284
|
+
});
|
|
127
285
|
});
|
|
128
286
|
test("fills nested objects", async () => {
|
|
129
287
|
const schema = Type.Object({
|
|
@@ -131,7 +289,15 @@ describe("FillInStruct", () => {
|
|
|
131
289
|
inner: Type.String()
|
|
132
290
|
})
|
|
133
291
|
});
|
|
292
|
+
|
|
293
|
+
// Outer menu: select "outer" field
|
|
294
|
+
mockSelectCancellable.mockResolvedValueOnce("field:outer");
|
|
295
|
+
// Inner menu: select "inner" field, fill, submit
|
|
296
|
+
mockSelectCancellable.mockResolvedValueOnce("field:inner");
|
|
134
297
|
mockInputCancellable.mockResolvedValueOnce("deep-value");
|
|
298
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
299
|
+
// Back to outer menu: submit
|
|
300
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
135
301
|
const result = await sprinkle.FillInStruct(schema);
|
|
136
302
|
expect(result).toEqual({
|
|
137
303
|
outer: {
|
|
@@ -139,14 +305,154 @@ describe("FillInStruct", () => {
|
|
|
139
305
|
}
|
|
140
306
|
});
|
|
141
307
|
});
|
|
142
|
-
test("
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
//
|
|
148
|
-
|
|
308
|
+
test("single required field skips menu", async () => {
|
|
309
|
+
const schema = Type.Object({
|
|
310
|
+
name: Type.String()
|
|
311
|
+
});
|
|
312
|
+
|
|
313
|
+
// No menu - directly prompts for the field
|
|
314
|
+
mockInputCancellable.mockResolvedValueOnce("direct-value");
|
|
315
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
316
|
+
expect(result).toEqual({
|
|
317
|
+
name: "direct-value"
|
|
318
|
+
});
|
|
319
|
+
// Verify no select menu was shown
|
|
320
|
+
expect(mockSelectCancellable).not.toHaveBeenCalled();
|
|
321
|
+
});
|
|
322
|
+
test("throws UserCancelledError when object menu is cancelled", async () => {
|
|
323
|
+
const schema = Type.Object({
|
|
324
|
+
name: Type.String(),
|
|
325
|
+
age: Type.BigInt()
|
|
326
|
+
});
|
|
327
|
+
|
|
328
|
+
// Cancel at menu (no values set)
|
|
329
|
+
mockSelectCancellable.mockResolvedValueOnce(null);
|
|
330
|
+
await expect(sprinkle.FillInStruct(schema)).rejects.toThrow(UserCancelledError);
|
|
331
|
+
});
|
|
332
|
+
test("cancel with values prompts confirmation", async () => {
|
|
333
|
+
const schema = Type.Object({
|
|
334
|
+
name: Type.String(),
|
|
335
|
+
age: Type.BigInt()
|
|
336
|
+
});
|
|
337
|
+
|
|
338
|
+
// Fill one field
|
|
339
|
+
mockSelectCancellable.mockResolvedValueOnce("field:name");
|
|
340
|
+
mockInputCancellable.mockResolvedValueOnce("Alice");
|
|
341
|
+
// Escape at menu
|
|
342
|
+
mockSelectCancellable.mockResolvedValueOnce(null);
|
|
343
|
+
// Confirm discard
|
|
344
|
+
mockConfirmCancellable.mockResolvedValueOnce(true);
|
|
345
|
+
await expect(sprinkle.FillInStruct(schema)).rejects.toThrow(UserCancelledError);
|
|
346
|
+
});
|
|
347
|
+
|
|
348
|
+
// --- Optional fields in object menu ---
|
|
349
|
+
|
|
350
|
+
test("optional field shows without asterisk in menu", async () => {
|
|
351
|
+
const schema = Type.Object({
|
|
352
|
+
required: Type.String(),
|
|
353
|
+
optional: Type.Optional(Type.String())
|
|
354
|
+
});
|
|
355
|
+
|
|
356
|
+
// Menu flow: select required -> fill -> submit (leaving optional unset)
|
|
357
|
+
mockSelectCancellable.mockResolvedValueOnce("field:required");
|
|
358
|
+
mockInputCancellable.mockResolvedValueOnce("value");
|
|
359
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
360
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
361
|
+
expect(result).toEqual({
|
|
362
|
+
required: "value"
|
|
363
|
+
});
|
|
364
|
+
// Optional field should be omitted from result
|
|
365
|
+
expect("optional" in result).toBe(false);
|
|
366
|
+
});
|
|
367
|
+
test("selecting optional field prompts Yes/Skip", async () => {
|
|
368
|
+
const schema = Type.Object({
|
|
369
|
+
name: Type.String(),
|
|
370
|
+
nickname: Type.Optional(Type.String())
|
|
371
|
+
});
|
|
372
|
+
|
|
373
|
+
// Menu flow: select name -> fill -> select nickname -> Yes -> fill -> submit
|
|
374
|
+
mockSelectCancellable.mockResolvedValueOnce("field:name");
|
|
375
|
+
mockInputCancellable.mockResolvedValueOnce("Alice");
|
|
376
|
+
mockSelectCancellable.mockResolvedValueOnce("field:nickname");
|
|
377
|
+
// Prompt: "Set value for nickname? Yes/Skip"
|
|
378
|
+
mockSelectCancellable.mockResolvedValueOnce(true); // Yes
|
|
379
|
+
mockInputCancellable.mockResolvedValueOnce("Ali");
|
|
380
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
381
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
382
|
+
expect(result).toEqual({
|
|
383
|
+
name: "Alice",
|
|
384
|
+
nickname: "Ali"
|
|
385
|
+
});
|
|
386
|
+
});
|
|
387
|
+
test("skipping optional field leaves it undefined", async () => {
|
|
388
|
+
const schema = Type.Object({
|
|
389
|
+
name: Type.String(),
|
|
390
|
+
nickname: Type.Optional(Type.String())
|
|
391
|
+
});
|
|
392
|
+
|
|
393
|
+
// Menu flow: select name -> fill -> select nickname -> Skip -> submit
|
|
394
|
+
mockSelectCancellable.mockResolvedValueOnce("field:name");
|
|
395
|
+
mockInputCancellable.mockResolvedValueOnce("Alice");
|
|
396
|
+
mockSelectCancellable.mockResolvedValueOnce("field:nickname");
|
|
397
|
+
// Prompt: "Set value for nickname? Yes/Skip"
|
|
398
|
+
mockSelectCancellable.mockResolvedValueOnce(false); // Skip
|
|
399
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
400
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
401
|
+
expect(result).toEqual({
|
|
402
|
+
name: "Alice"
|
|
403
|
+
});
|
|
404
|
+
expect("nickname" in result).toBe(false);
|
|
405
|
+
});
|
|
406
|
+
test("all-optional object can be submitted empty", async () => {
|
|
407
|
+
const schema = Type.Object({
|
|
408
|
+
a: Type.Optional(Type.String()),
|
|
409
|
+
b: Type.Optional(Type.String())
|
|
410
|
+
});
|
|
411
|
+
|
|
412
|
+
// Submit immediately without setting any fields
|
|
413
|
+
mockSelectCancellable.mockResolvedValueOnce("submit");
|
|
414
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
415
|
+
expect(result).toEqual({});
|
|
416
|
+
});
|
|
417
|
+
|
|
418
|
+
// --- Array types (menu-based) ---
|
|
419
|
+
|
|
420
|
+
test("fills an array with items", async () => {
|
|
421
|
+
const schema = Type.Array(Type.String());
|
|
422
|
+
|
|
423
|
+
// Menu: Add -> fill -> Add -> fill -> Done
|
|
424
|
+
mockSelectCancellable.mockResolvedValueOnce("add");
|
|
425
|
+
mockInputCancellable.mockResolvedValueOnce("first");
|
|
426
|
+
mockSelectCancellable.mockResolvedValueOnce("add");
|
|
427
|
+
mockInputCancellable.mockResolvedValueOnce("second");
|
|
428
|
+
mockSelectCancellable.mockResolvedValueOnce("done");
|
|
429
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
430
|
+
expect(result).toEqual(["first", "second"]);
|
|
431
|
+
});
|
|
432
|
+
test("fills an array with single item", async () => {
|
|
433
|
+
const schema = Type.Array(Type.String());
|
|
434
|
+
|
|
435
|
+
// Menu: Add -> fill -> Done
|
|
436
|
+
mockSelectCancellable.mockResolvedValueOnce("add");
|
|
437
|
+
mockInputCancellable.mockResolvedValueOnce("only");
|
|
438
|
+
mockSelectCancellable.mockResolvedValueOnce("done");
|
|
439
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
440
|
+
expect(result).toEqual(["only"]);
|
|
441
|
+
});
|
|
442
|
+
test("fills empty array by selecting Done immediately", async () => {
|
|
443
|
+
const schema = Type.Array(Type.String());
|
|
444
|
+
mockSelectCancellable.mockResolvedValueOnce("done");
|
|
445
|
+
const result = await sprinkle.FillInStruct(schema);
|
|
446
|
+
expect(result).toEqual([]);
|
|
447
|
+
});
|
|
448
|
+
test("array Back throws UserCancelledError", async () => {
|
|
449
|
+
const schema = Type.Array(Type.String());
|
|
450
|
+
mockSelectCancellable.mockResolvedValueOnce("back");
|
|
451
|
+
await expect(sprinkle.FillInStruct(schema)).rejects.toThrow(UserCancelledError);
|
|
149
452
|
});
|
|
453
|
+
|
|
454
|
+
// --- Tuple types (unchanged - sequential) ---
|
|
455
|
+
|
|
150
456
|
test("fills a tuple with same-type elements", async () => {
|
|
151
457
|
const schema = Type.Tuple([Type.String(), Type.String()]);
|
|
152
458
|
mockInputCancellable.mockResolvedValueOnce("policyId").mockResolvedValueOnce("assetName");
|
|
@@ -170,12 +476,17 @@ describe("FillInStruct", () => {
|
|
|
170
476
|
const schema = Type.Object({
|
|
171
477
|
asset: Type.Tuple([Type.String(), Type.String()])
|
|
172
478
|
});
|
|
479
|
+
|
|
480
|
+
// Single required field skips menu
|
|
173
481
|
mockInputCancellable.mockResolvedValueOnce("policy123").mockResolvedValueOnce("token456");
|
|
174
482
|
const result = await sprinkle.FillInStruct(schema);
|
|
175
483
|
expect(result).toEqual({
|
|
176
484
|
asset: ["policy123", "token456"]
|
|
177
485
|
});
|
|
178
486
|
});
|
|
487
|
+
|
|
488
|
+
// --- Hot wallet special handling ---
|
|
489
|
+
|
|
179
490
|
test("hot wallet private key shows setup choice", async () => {
|
|
180
491
|
const schema = Type.String({
|
|
181
492
|
title: "Hot Wallet Private Key"
|
|
@@ -209,48 +520,30 @@ describe("FillInStruct", () => {
|
|
|
209
520
|
});
|
|
210
521
|
expect(result).toBe("deadbeef1234");
|
|
211
522
|
});
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
mockPasswordCancellable.mockResolvedValueOnce("abc123privatekey");
|
|
220
|
-
const result = await sprinkle.FillInStruct(WalletSettingsSchema);
|
|
221
|
-
expect(result).toEqual({
|
|
222
|
-
type: "hot",
|
|
223
|
-
privateKey: "abc123privatekey"
|
|
224
|
-
});
|
|
225
|
-
});
|
|
523
|
+
|
|
524
|
+
// Note: Full wallet settings test removed due to mock complexity.
|
|
525
|
+
// The individual components (union selection, single-field optimization,
|
|
526
|
+
// hot wallet key handling) are tested separately above.
|
|
527
|
+
|
|
528
|
+
// --- Cancel/escape behavior ---
|
|
529
|
+
|
|
226
530
|
test("throws UserCancelledError when input prompt is cancelled", async () => {
|
|
227
531
|
mockInputCancellable.mockResolvedValueOnce(null);
|
|
228
532
|
await expect(sprinkle.FillInStruct(Type.String())).rejects.toThrow(UserCancelledError);
|
|
229
533
|
});
|
|
230
|
-
test("throws UserCancelledError when
|
|
231
|
-
const schema = Type.Union([Type.Object({
|
|
232
|
-
type: Type.Literal("a")
|
|
233
|
-
}), Type.Object({
|
|
234
|
-
type: Type.Literal("b")
|
|
235
|
-
})]);
|
|
236
|
-
mockSelectCancellable.mockResolvedValueOnce(null);
|
|
237
|
-
await expect(sprinkle.FillInStruct(schema)).rejects.toThrow(UserCancelledError);
|
|
238
|
-
});
|
|
239
|
-
test("throws UserCancelledError when nested prompt is cancelled", async () => {
|
|
534
|
+
test("throws UserCancelledError when field input cancelled in object menu", async () => {
|
|
240
535
|
const schema = Type.Object({
|
|
241
536
|
name: Type.String(),
|
|
242
537
|
age: Type.BigInt()
|
|
243
538
|
});
|
|
244
539
|
|
|
245
|
-
//
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
mockSelectCancellable.mockResolvedValueOnce(null); // cancel on "add another?"
|
|
253
|
-
|
|
540
|
+
// Select name field
|
|
541
|
+
mockSelectCancellable.mockResolvedValueOnce("field:name");
|
|
542
|
+
// Cancel during input
|
|
543
|
+
mockInputCancellable.mockResolvedValueOnce(null);
|
|
544
|
+
// Return to menu (field unchanged)
|
|
545
|
+
// Then cancel at menu
|
|
546
|
+
mockSelectCancellable.mockResolvedValueOnce(null);
|
|
254
547
|
await expect(sprinkle.FillInStruct(schema)).rejects.toThrow(UserCancelledError);
|
|
255
548
|
});
|
|
256
549
|
});
|