@sundaeswap/sprinkles 0.3.0 → 0.5.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/dist/cjs/Sprinkle/__tests__/encryption.test.js +20 -8
- package/dist/cjs/Sprinkle/__tests__/encryption.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/enhancements.test.js +41 -16
- package/dist/cjs/Sprinkle/__tests__/enhancements.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js +85 -38
- package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/settings-persistence.test.js +120 -0
- package/dist/cjs/Sprinkle/__tests__/settings-persistence.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/show-menu.test.js +93 -7
- package/dist/cjs/Sprinkle/__tests__/show-menu.test.js.map +1 -1
- package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js +21 -0
- package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js.map +1 -1
- package/dist/cjs/Sprinkle/encryption.js +131 -0
- package/dist/cjs/Sprinkle/encryption.js.map +1 -0
- package/dist/cjs/Sprinkle/index.js +318 -352
- package/dist/cjs/Sprinkle/index.js.map +1 -1
- package/dist/cjs/Sprinkle/prompts.js +393 -0
- package/dist/cjs/Sprinkle/prompts.js.map +1 -0
- package/dist/cjs/Sprinkle/schemas.js +97 -0
- package/dist/cjs/Sprinkle/schemas.js.map +1 -0
- package/dist/cjs/Sprinkle/tx-dialog.js +101 -0
- package/dist/cjs/Sprinkle/tx-dialog.js.map +1 -0
- package/dist/cjs/Sprinkle/type-guards.js +42 -0
- package/dist/cjs/Sprinkle/type-guards.js.map +1 -0
- package/dist/cjs/Sprinkle/types.js +49 -0
- package/dist/cjs/Sprinkle/types.js.map +1 -0
- package/dist/cjs/Sprinkle/wallet.js +98 -0
- package/dist/cjs/Sprinkle/wallet.js.map +1 -0
- package/dist/esm/Sprinkle/__tests__/encryption.test.js +20 -8
- package/dist/esm/Sprinkle/__tests__/encryption.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/enhancements.test.js +41 -16
- package/dist/esm/Sprinkle/__tests__/enhancements.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js +85 -38
- package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/settings-persistence.test.js +120 -0
- package/dist/esm/Sprinkle/__tests__/settings-persistence.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/show-menu.test.js +94 -8
- package/dist/esm/Sprinkle/__tests__/show-menu.test.js.map +1 -1
- package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js +21 -0
- package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js.map +1 -1
- package/dist/esm/Sprinkle/encryption.js +117 -0
- package/dist/esm/Sprinkle/encryption.js.map +1 -0
- package/dist/esm/Sprinkle/index.js +172 -337
- package/dist/esm/Sprinkle/index.js.map +1 -1
- package/dist/esm/Sprinkle/prompts.js +385 -0
- package/dist/esm/Sprinkle/prompts.js.map +1 -0
- package/dist/esm/Sprinkle/schemas.js +91 -0
- package/dist/esm/Sprinkle/schemas.js.map +1 -0
- package/dist/esm/Sprinkle/tx-dialog.js +90 -0
- package/dist/esm/Sprinkle/tx-dialog.js.map +1 -0
- package/dist/esm/Sprinkle/type-guards.js +24 -0
- package/dist/esm/Sprinkle/type-guards.js.map +1 -0
- package/dist/esm/Sprinkle/types.js +42 -0
- package/dist/esm/Sprinkle/types.js.map +1 -0
- package/dist/esm/Sprinkle/wallet.js +90 -0
- package/dist/esm/Sprinkle/wallet.js.map +1 -0
- package/dist/types/Sprinkle/encryption.d.ts +43 -0
- package/dist/types/Sprinkle/encryption.d.ts.map +1 -0
- package/dist/types/Sprinkle/index.d.ts +13 -174
- package/dist/types/Sprinkle/index.d.ts.map +1 -1
- package/dist/types/Sprinkle/prompts.d.ts +94 -0
- package/dist/types/Sprinkle/prompts.d.ts.map +1 -0
- package/dist/types/Sprinkle/schemas.d.ts +125 -0
- package/dist/types/Sprinkle/schemas.d.ts.map +1 -0
- package/dist/types/Sprinkle/tx-dialog.d.ts +37 -0
- package/dist/types/Sprinkle/tx-dialog.d.ts.map +1 -0
- package/dist/types/Sprinkle/type-guards.d.ts +22 -0
- package/dist/types/Sprinkle/type-guards.d.ts.map +1 -0
- package/dist/types/Sprinkle/types.d.ts +62 -0
- package/dist/types/Sprinkle/types.d.ts.map +1 -0
- package/dist/types/Sprinkle/wallet.d.ts +27 -0
- package/dist/types/Sprinkle/wallet.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 +21 -8
- package/src/Sprinkle/__tests__/enhancements.test.ts +41 -15
- package/src/Sprinkle/__tests__/fill-in-struct.test.ts +104 -38
- package/src/Sprinkle/__tests__/settings-persistence.test.ts +108 -0
- package/src/Sprinkle/__tests__/show-menu.test.ts +96 -8
- package/src/Sprinkle/__tests__/tx-dialog.test.ts +21 -0
- package/src/Sprinkle/encryption.ts +130 -0
- package/src/Sprinkle/index.ts +265 -478
- package/src/Sprinkle/prompts.ts +481 -0
- package/src/Sprinkle/schemas.ts +111 -0
- package/src/Sprinkle/tx-dialog.ts +100 -0
- package/src/Sprinkle/type-guards.ts +51 -0
- package/src/Sprinkle/types.ts +73 -0
- package/src/Sprinkle/wallet.ts +133 -0
|
@@ -0,0 +1,481 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cancellable prompt wrappers with escape key support.
|
|
3
|
+
* Uses @inquirer/core's createPrompt and useKeypress for proper escape handling.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import {
|
|
7
|
+
createPrompt,
|
|
8
|
+
useState,
|
|
9
|
+
useKeypress,
|
|
10
|
+
usePrefix,
|
|
11
|
+
usePagination,
|
|
12
|
+
useMemo,
|
|
13
|
+
makeTheme,
|
|
14
|
+
isEnterKey,
|
|
15
|
+
isUpKey,
|
|
16
|
+
isDownKey,
|
|
17
|
+
isBackspaceKey,
|
|
18
|
+
Separator,
|
|
19
|
+
ValidationError,
|
|
20
|
+
type Theme,
|
|
21
|
+
} from "@inquirer/core";
|
|
22
|
+
import type { PartialDeep } from "@inquirer/type";
|
|
23
|
+
import colors from "yoctocolors-cjs";
|
|
24
|
+
import figures from "@inquirer/figures";
|
|
25
|
+
|
|
26
|
+
// Check if key is escape
|
|
27
|
+
const isEscapeKey = (key: { name: string }) => key.name === "escape";
|
|
28
|
+
|
|
29
|
+
// Theme for select prompt
|
|
30
|
+
const selectTheme = {
|
|
31
|
+
icon: { cursor: figures.pointer },
|
|
32
|
+
style: {
|
|
33
|
+
disabled: (text: string) => colors.dim(`- ${text}`),
|
|
34
|
+
description: (text: string) => colors.cyan(text),
|
|
35
|
+
helpTip: colors.dim("(Use arrow keys, Enter to select, Esc to cancel)"),
|
|
36
|
+
},
|
|
37
|
+
helpMode: "always" as const,
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
// Theme for input prompt
|
|
41
|
+
const inputTheme = {
|
|
42
|
+
style: {
|
|
43
|
+
helpTip: colors.dim("(Esc to cancel)"),
|
|
44
|
+
},
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
interface SelectChoice<T> {
|
|
48
|
+
name?: string;
|
|
49
|
+
value: T;
|
|
50
|
+
description?: string;
|
|
51
|
+
short?: string;
|
|
52
|
+
disabled?: boolean | string;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
interface SelectConfig<T> {
|
|
56
|
+
message: string;
|
|
57
|
+
choices: ReadonlyArray<SelectChoice<T> | Separator | string>;
|
|
58
|
+
default?: T;
|
|
59
|
+
pageSize?: number;
|
|
60
|
+
loop?: boolean;
|
|
61
|
+
theme?: PartialDeep<Theme<typeof selectTheme>>;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
function isSelectable<T>(item: SelectChoice<T> | Separator): item is SelectChoice<T> {
|
|
65
|
+
return !Separator.isSeparator(item) && !item.disabled;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function normalizeChoices<T>(
|
|
69
|
+
choices: ReadonlyArray<SelectChoice<T> | Separator | string>,
|
|
70
|
+
): Array<SelectChoice<T> | Separator> {
|
|
71
|
+
return choices.map((choice) => {
|
|
72
|
+
if (Separator.isSeparator(choice)) return choice;
|
|
73
|
+
if (typeof choice === "string") {
|
|
74
|
+
return {
|
|
75
|
+
value: choice as unknown as T,
|
|
76
|
+
name: choice,
|
|
77
|
+
short: choice,
|
|
78
|
+
disabled: false,
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
const name = choice.name ?? String(choice.value);
|
|
82
|
+
return {
|
|
83
|
+
value: choice.value,
|
|
84
|
+
name,
|
|
85
|
+
short: choice.short ?? name,
|
|
86
|
+
disabled: choice.disabled ?? false,
|
|
87
|
+
description: choice.description,
|
|
88
|
+
};
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Select prompt with escape key support.
|
|
94
|
+
* Returns null if user presses Escape.
|
|
95
|
+
*/
|
|
96
|
+
export const selectCancellable = createPrompt<
|
|
97
|
+
unknown | null,
|
|
98
|
+
SelectConfig<unknown>
|
|
99
|
+
>((config, done) => {
|
|
100
|
+
const { loop = true, pageSize = 7 } = config;
|
|
101
|
+
const theme = makeTheme(selectTheme, config.theme);
|
|
102
|
+
const [status, setStatus] = useState<"idle" | "done" | "cancelled">("idle");
|
|
103
|
+
const prefix = usePrefix({ status: status === "cancelled" ? "done" : status, theme });
|
|
104
|
+
|
|
105
|
+
const items = useMemo(
|
|
106
|
+
() => normalizeChoices(config.choices),
|
|
107
|
+
[config.choices],
|
|
108
|
+
);
|
|
109
|
+
|
|
110
|
+
const bounds = useMemo(() => {
|
|
111
|
+
const first = items.findIndex(isSelectable);
|
|
112
|
+
// Find last selectable item (compatible with es2022)
|
|
113
|
+
let last = -1;
|
|
114
|
+
for (let i = items.length - 1; i >= 0; i--) {
|
|
115
|
+
if (isSelectable(items[i]!)) {
|
|
116
|
+
last = i;
|
|
117
|
+
break;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
if (first === -1) {
|
|
121
|
+
throw new ValidationError(
|
|
122
|
+
"[select prompt] No selectable choices. All choices are disabled.",
|
|
123
|
+
);
|
|
124
|
+
}
|
|
125
|
+
return { first, last };
|
|
126
|
+
}, [items]);
|
|
127
|
+
|
|
128
|
+
const defaultItemIndex = useMemo(() => {
|
|
129
|
+
if (!("default" in config)) return -1;
|
|
130
|
+
return items.findIndex(
|
|
131
|
+
(item) => isSelectable(item) && item.value === config.default,
|
|
132
|
+
);
|
|
133
|
+
}, [config.default, items]);
|
|
134
|
+
|
|
135
|
+
const [active, setActive] = useState(
|
|
136
|
+
defaultItemIndex === -1 ? bounds.first : defaultItemIndex,
|
|
137
|
+
);
|
|
138
|
+
|
|
139
|
+
const selectedChoice = items[active] as SelectChoice<unknown>;
|
|
140
|
+
|
|
141
|
+
useKeypress((key, rl) => {
|
|
142
|
+
if (isEscapeKey(key)) {
|
|
143
|
+
setStatus("cancelled");
|
|
144
|
+
done(null);
|
|
145
|
+
} else if (isEnterKey(key)) {
|
|
146
|
+
setStatus("done");
|
|
147
|
+
done(selectedChoice.value);
|
|
148
|
+
} else if (isUpKey(key)) {
|
|
149
|
+
rl.clearLine(0);
|
|
150
|
+
if (active === bounds.first && loop) {
|
|
151
|
+
setActive(bounds.last);
|
|
152
|
+
} else {
|
|
153
|
+
let newActive = active - 1;
|
|
154
|
+
while (newActive >= bounds.first && !isSelectable(items[newActive]!)) {
|
|
155
|
+
newActive--;
|
|
156
|
+
}
|
|
157
|
+
if (newActive >= bounds.first) setActive(newActive);
|
|
158
|
+
}
|
|
159
|
+
} else if (isDownKey(key)) {
|
|
160
|
+
rl.clearLine(0);
|
|
161
|
+
if (active === bounds.last && loop) {
|
|
162
|
+
setActive(bounds.first);
|
|
163
|
+
} else {
|
|
164
|
+
let newActive = active + 1;
|
|
165
|
+
while (newActive <= bounds.last && !isSelectable(items[newActive]!)) {
|
|
166
|
+
newActive++;
|
|
167
|
+
}
|
|
168
|
+
if (newActive <= bounds.last) setActive(newActive);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
const page = usePagination({
|
|
174
|
+
items,
|
|
175
|
+
active,
|
|
176
|
+
renderItem: ({ item, isActive }) => {
|
|
177
|
+
if (Separator.isSeparator(item)) {
|
|
178
|
+
return ` ${item.separator}`;
|
|
179
|
+
}
|
|
180
|
+
const line = item.name;
|
|
181
|
+
if (item.disabled) {
|
|
182
|
+
const disabledLabel =
|
|
183
|
+
typeof item.disabled === "string" ? item.disabled : "(disabled)";
|
|
184
|
+
return colors.dim(`- ${line} ${disabledLabel}`);
|
|
185
|
+
}
|
|
186
|
+
const cursor = isActive ? figures.pointer : " ";
|
|
187
|
+
const color = isActive ? colors.cyan : (x: string) => x;
|
|
188
|
+
return color(`${cursor} ${line}`);
|
|
189
|
+
},
|
|
190
|
+
pageSize,
|
|
191
|
+
loop,
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
if (status === "cancelled") {
|
|
195
|
+
return `${prefix} ${config.message} ${colors.dim("(cancelled)")}`;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
if (status === "done") {
|
|
199
|
+
return `${prefix} ${config.message} ${colors.cyan(selectedChoice?.short ?? selectedChoice?.name ?? "")}`;
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
const helpTip = colors.dim("(Use arrow keys, Enter to select, Esc to cancel)");
|
|
203
|
+
return `${prefix} ${config.message} ${helpTip}\n${page}`;
|
|
204
|
+
});
|
|
205
|
+
|
|
206
|
+
interface InputConfig {
|
|
207
|
+
message: string;
|
|
208
|
+
default?: string;
|
|
209
|
+
transformer?: (value: string, options: { isFinal: boolean }) => string;
|
|
210
|
+
validate?: (value: string) => boolean | string | Promise<boolean | string>;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
/**
|
|
214
|
+
* Input prompt with escape key support.
|
|
215
|
+
* Returns null if user presses Escape.
|
|
216
|
+
*/
|
|
217
|
+
export const inputCancellable = createPrompt<string | null, InputConfig>(
|
|
218
|
+
(config, done) => {
|
|
219
|
+
const [status, setStatus] = useState<"idle" | "done" | "cancelled">("idle");
|
|
220
|
+
const [value, setValue] = useState(config.default ?? "");
|
|
221
|
+
const [touched, setTouched] = useState(false);
|
|
222
|
+
const [error, setError] = useState<string | null>(null);
|
|
223
|
+
const theme = makeTheme(inputTheme);
|
|
224
|
+
const prefix = usePrefix({ status: status === "cancelled" ? "done" : status, theme });
|
|
225
|
+
|
|
226
|
+
useKeypress(async (key, rl) => {
|
|
227
|
+
if (isEscapeKey(key)) {
|
|
228
|
+
setStatus("cancelled");
|
|
229
|
+
done(null);
|
|
230
|
+
} else if (isEnterKey(key)) {
|
|
231
|
+
// If user has edited, use their value (even if empty); otherwise use default
|
|
232
|
+
const answer = touched ? value : (config.default ?? "");
|
|
233
|
+
if (config.validate) {
|
|
234
|
+
const result = await config.validate(answer);
|
|
235
|
+
if (result !== true) {
|
|
236
|
+
setError(typeof result === "string" ? result : "Invalid input");
|
|
237
|
+
return;
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
setStatus("done");
|
|
241
|
+
done(answer);
|
|
242
|
+
} else if (isBackspaceKey(key)) {
|
|
243
|
+
// Sync with readline's buffer after backspace
|
|
244
|
+
setValue(rl.line);
|
|
245
|
+
setTouched(true);
|
|
246
|
+
setError(null);
|
|
247
|
+
} else if (key.name !== "tab" && !key.ctrl && !(key as any).meta) {
|
|
248
|
+
// Sync state with readline's current line buffer
|
|
249
|
+
setValue(rl.line);
|
|
250
|
+
setTouched(true);
|
|
251
|
+
setError(null);
|
|
252
|
+
}
|
|
253
|
+
});
|
|
254
|
+
|
|
255
|
+
const displayValue = config.transformer
|
|
256
|
+
? config.transformer(value, { isFinal: status === "done" })
|
|
257
|
+
: value;
|
|
258
|
+
|
|
259
|
+
if (status === "cancelled") {
|
|
260
|
+
return `${prefix} ${config.message} ${colors.dim("(cancelled)")}`;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
if (status === "done") {
|
|
264
|
+
return `${prefix} ${config.message} ${colors.cyan(displayValue)}`;
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
const helpTip = colors.dim("(Esc to cancel)");
|
|
268
|
+
const errorMsg = error ? colors.red(`\n> ${error}`) : "";
|
|
269
|
+
return `${prefix} ${config.message} ${helpTip} ${displayValue}${errorMsg}`;
|
|
270
|
+
},
|
|
271
|
+
);
|
|
272
|
+
|
|
273
|
+
interface PasswordConfig {
|
|
274
|
+
message: string;
|
|
275
|
+
mask?: string;
|
|
276
|
+
validate?: (value: string) => boolean | string | Promise<boolean | string>;
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
/**
|
|
280
|
+
* Password prompt with escape key support.
|
|
281
|
+
* Returns null if user presses Escape.
|
|
282
|
+
*/
|
|
283
|
+
export const passwordCancellable = createPrompt<string | null, PasswordConfig>(
|
|
284
|
+
(config, done) => {
|
|
285
|
+
const [status, setStatus] = useState<"idle" | "done" | "cancelled">("idle");
|
|
286
|
+
const [value, setValue] = useState("");
|
|
287
|
+
const [error, setError] = useState<string | null>(null);
|
|
288
|
+
const theme = makeTheme(inputTheme);
|
|
289
|
+
const prefix = usePrefix({ status: status === "cancelled" ? "done" : status, theme });
|
|
290
|
+
const mask = config.mask ?? "*";
|
|
291
|
+
|
|
292
|
+
useKeypress(async (key, rl) => {
|
|
293
|
+
if (isEscapeKey(key)) {
|
|
294
|
+
setStatus("cancelled");
|
|
295
|
+
done(null);
|
|
296
|
+
} else if (isEnterKey(key)) {
|
|
297
|
+
if (config.validate) {
|
|
298
|
+
const result = await config.validate(value);
|
|
299
|
+
if (result !== true) {
|
|
300
|
+
setError(typeof result === "string" ? result : "Invalid input");
|
|
301
|
+
return;
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
setStatus("done");
|
|
305
|
+
done(value);
|
|
306
|
+
} else if (isBackspaceKey(key)) {
|
|
307
|
+
setValue(value.slice(0, -1));
|
|
308
|
+
setError(null);
|
|
309
|
+
} else if (key.name !== "tab" && !key.ctrl && !(key as any).meta) {
|
|
310
|
+
// Use key.sequence for single character to avoid rl.line buffer issues
|
|
311
|
+
const char = (key as any).sequence ?? "";
|
|
312
|
+
if (char) {
|
|
313
|
+
rl.clearLine(0);
|
|
314
|
+
setValue(value + char);
|
|
315
|
+
setError(null);
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
});
|
|
319
|
+
|
|
320
|
+
const masked = mask.repeat(value.length);
|
|
321
|
+
|
|
322
|
+
if (status === "cancelled") {
|
|
323
|
+
return `${prefix} ${config.message} ${colors.dim("(cancelled)")}`;
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
if (status === "done") {
|
|
327
|
+
return `${prefix} ${config.message} ${colors.dim(masked)}`;
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
const helpTip = colors.dim("(Esc to cancel)");
|
|
331
|
+
const errorMsg = error ? colors.red(`\n> ${error}`) : "";
|
|
332
|
+
return `${prefix} ${config.message} ${helpTip} ${masked}${errorMsg}`;
|
|
333
|
+
},
|
|
334
|
+
);
|
|
335
|
+
|
|
336
|
+
interface ConfirmConfig {
|
|
337
|
+
message: string;
|
|
338
|
+
default?: boolean;
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
/**
|
|
342
|
+
* Confirm prompt with escape key support.
|
|
343
|
+
* Returns null if user presses Escape.
|
|
344
|
+
*/
|
|
345
|
+
export const confirmCancellable = createPrompt<boolean | null, ConfirmConfig>(
|
|
346
|
+
(config, done) => {
|
|
347
|
+
const [status, setStatus] = useState<"idle" | "done" | "cancelled">("idle");
|
|
348
|
+
// Track value as undefined when no default, requiring explicit y/n
|
|
349
|
+
const [value, setValue] = useState<boolean | undefined>(config.default);
|
|
350
|
+
const theme = makeTheme(inputTheme);
|
|
351
|
+
const prefix = usePrefix({ status: status === "cancelled" ? "done" : status, theme });
|
|
352
|
+
|
|
353
|
+
useKeypress((key) => {
|
|
354
|
+
if (isEscapeKey(key)) {
|
|
355
|
+
setStatus("cancelled");
|
|
356
|
+
done(null);
|
|
357
|
+
} else if (isEnterKey(key)) {
|
|
358
|
+
// Only accept Enter if a value has been chosen (explicit or default)
|
|
359
|
+
if (value !== undefined) {
|
|
360
|
+
setStatus("done");
|
|
361
|
+
done(value);
|
|
362
|
+
}
|
|
363
|
+
} else if (key.name === "y" || key.name === "Y") {
|
|
364
|
+
setValue(true);
|
|
365
|
+
} else if (key.name === "n" || key.name === "N") {
|
|
366
|
+
setValue(false);
|
|
367
|
+
}
|
|
368
|
+
});
|
|
369
|
+
|
|
370
|
+
const hint = config.default === true ? "(Y/n)" : config.default === false ? "(y/N)" : "(y/n)";
|
|
371
|
+
const displayValue = value === true ? "Yes" : value === false ? "No" : "";
|
|
372
|
+
|
|
373
|
+
if (status === "cancelled") {
|
|
374
|
+
return `${prefix} ${config.message} ${colors.dim("(cancelled)")}`;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
if (status === "done") {
|
|
378
|
+
return `${prefix} ${config.message} ${colors.cyan(displayValue)}`;
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
const helpTip = colors.dim("(Esc to cancel)");
|
|
382
|
+
return `${prefix} ${config.message} ${helpTip} ${hint} ${displayValue}`;
|
|
383
|
+
},
|
|
384
|
+
);
|
|
385
|
+
|
|
386
|
+
/**
|
|
387
|
+
* Type-safe wrapper for selectCancellable.
|
|
388
|
+
* Returns the selected value with proper typing, or null if cancelled.
|
|
389
|
+
*/
|
|
390
|
+
export async function select<T>(config: SelectConfig<T>): Promise<T | null> {
|
|
391
|
+
return selectCancellable(config as SelectConfig<unknown>) as Promise<T | null>;
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
interface SearchConfig<T> {
|
|
395
|
+
message: string;
|
|
396
|
+
source: (
|
|
397
|
+
term: string | undefined,
|
|
398
|
+
) => Promise<{ name: string; value: T }[]> | { name: string; value: T }[];
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
/**
|
|
402
|
+
* Search prompt with escape key support.
|
|
403
|
+
* Returns null if user presses Escape.
|
|
404
|
+
*
|
|
405
|
+
* Note: This wraps @inquirer/search which has built-in escape handling,
|
|
406
|
+
* but we provide a consistent API with other cancellable prompts.
|
|
407
|
+
*/
|
|
408
|
+
export const searchCancellable = createPrompt<unknown | null, SearchConfig<unknown>>(
|
|
409
|
+
(config, done) => {
|
|
410
|
+
const [status, setStatus] = useState<"idle" | "done" | "cancelled">("idle");
|
|
411
|
+
const [searchTerm, setSearchTerm] = useState("");
|
|
412
|
+
const [results, setResults] = useState<{ name: string; value: unknown }[]>([]);
|
|
413
|
+
const [active, setActive] = useState(0);
|
|
414
|
+
const [loading, setLoading] = useState(false);
|
|
415
|
+
const theme = makeTheme(selectTheme);
|
|
416
|
+
const prefix = usePrefix({ status: status === "cancelled" ? "done" : status, theme });
|
|
417
|
+
|
|
418
|
+
// Fetch results when search term changes
|
|
419
|
+
useMemo(async () => {
|
|
420
|
+
setLoading(true);
|
|
421
|
+
try {
|
|
422
|
+
const items = await config.source(searchTerm || undefined);
|
|
423
|
+
setResults(items);
|
|
424
|
+
setActive(0);
|
|
425
|
+
} finally {
|
|
426
|
+
setLoading(false);
|
|
427
|
+
}
|
|
428
|
+
}, [searchTerm]);
|
|
429
|
+
|
|
430
|
+
useKeypress((key, rl) => {
|
|
431
|
+
if (isEscapeKey(key)) {
|
|
432
|
+
setStatus("cancelled");
|
|
433
|
+
done(null);
|
|
434
|
+
} else if (isEnterKey(key) && results.length > 0) {
|
|
435
|
+
setStatus("done");
|
|
436
|
+
done(results[active]?.value ?? null);
|
|
437
|
+
} else if (isUpKey(key) && results.length > 0) {
|
|
438
|
+
rl.clearLine(0);
|
|
439
|
+
setActive(active > 0 ? active - 1 : results.length - 1);
|
|
440
|
+
} else if (isDownKey(key) && results.length > 0) {
|
|
441
|
+
rl.clearLine(0);
|
|
442
|
+
setActive(active < results.length - 1 ? active + 1 : 0);
|
|
443
|
+
} else if (isBackspaceKey(key)) {
|
|
444
|
+
setSearchTerm(rl.line);
|
|
445
|
+
} else if (key.name !== "tab" && !key.ctrl && !(key as any).meta) {
|
|
446
|
+
setSearchTerm(rl.line);
|
|
447
|
+
}
|
|
448
|
+
});
|
|
449
|
+
|
|
450
|
+
if (status === "cancelled") {
|
|
451
|
+
return `${prefix} ${config.message} ${colors.dim("(cancelled)")}`;
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
if (status === "done") {
|
|
455
|
+
const selected = results[active];
|
|
456
|
+
return `${prefix} ${config.message} ${colors.cyan(selected?.name ?? "")}`;
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
const helpTip = colors.dim("(Type to search, Esc to cancel)");
|
|
460
|
+
const searchLine = `${prefix} ${config.message} ${helpTip} ${searchTerm}`;
|
|
461
|
+
|
|
462
|
+
if (loading) {
|
|
463
|
+
return `${searchLine}\n${colors.dim(" Searching...")}`;
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
if (results.length === 0) {
|
|
467
|
+
return `${searchLine}\n${colors.dim(" No results")}`;
|
|
468
|
+
}
|
|
469
|
+
|
|
470
|
+
const resultLines = results
|
|
471
|
+
.slice(0, 7)
|
|
472
|
+
.map((item, i) => {
|
|
473
|
+
const cursor = i === active ? figures.pointer : " ";
|
|
474
|
+
const color = i === active ? colors.cyan : (x: string) => x;
|
|
475
|
+
return color(`${cursor} ${item.name}`);
|
|
476
|
+
})
|
|
477
|
+
.join("\n");
|
|
478
|
+
|
|
479
|
+
return `${searchLine}\n${resultLines}`;
|
|
480
|
+
},
|
|
481
|
+
);
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
import { Type } from "@sinclair/typebox";
|
|
2
|
+
import type { TExact } from "./types.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Network selection schema (mainnet, preview, preprod).
|
|
6
|
+
*/
|
|
7
|
+
export const NetworkSchema = Type.Union([
|
|
8
|
+
Type.Literal("mainnet"),
|
|
9
|
+
Type.Literal("preview"),
|
|
10
|
+
Type.Literal("preprod"),
|
|
11
|
+
]);
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Recursive multisig script module for Cardano native scripts.
|
|
15
|
+
*/
|
|
16
|
+
export const MultisigScriptModule = Type.Module({
|
|
17
|
+
MultisigScript: Type.Union([
|
|
18
|
+
Type.Object({
|
|
19
|
+
Signature: Type.Object(
|
|
20
|
+
{
|
|
21
|
+
key_hash: Type.String(),
|
|
22
|
+
},
|
|
23
|
+
{ ctor: 0n },
|
|
24
|
+
),
|
|
25
|
+
}),
|
|
26
|
+
Type.Object({
|
|
27
|
+
AllOf: Type.Object(
|
|
28
|
+
{
|
|
29
|
+
scripts: Type.Array(Type.Ref("MultisigScript")),
|
|
30
|
+
},
|
|
31
|
+
{ ctor: 1n },
|
|
32
|
+
),
|
|
33
|
+
}),
|
|
34
|
+
Type.Object({
|
|
35
|
+
AnyOf: Type.Object(
|
|
36
|
+
{
|
|
37
|
+
scripts: Type.Array(Type.Ref("MultisigScript")),
|
|
38
|
+
},
|
|
39
|
+
{ ctor: 2n },
|
|
40
|
+
),
|
|
41
|
+
}),
|
|
42
|
+
Type.Object({
|
|
43
|
+
AtLeast: Type.Object(
|
|
44
|
+
{
|
|
45
|
+
required: Type.BigInt(),
|
|
46
|
+
scripts: Type.Array(Type.Ref("MultisigScript")),
|
|
47
|
+
},
|
|
48
|
+
{ ctor: 3n },
|
|
49
|
+
),
|
|
50
|
+
}),
|
|
51
|
+
Type.Object({
|
|
52
|
+
Before: Type.Object(
|
|
53
|
+
{
|
|
54
|
+
time: Type.BigInt(),
|
|
55
|
+
},
|
|
56
|
+
{ ctor: 4n },
|
|
57
|
+
),
|
|
58
|
+
}),
|
|
59
|
+
Type.Object({
|
|
60
|
+
After: Type.Object(
|
|
61
|
+
{
|
|
62
|
+
time: Type.BigInt(),
|
|
63
|
+
},
|
|
64
|
+
{ ctor: 5n },
|
|
65
|
+
),
|
|
66
|
+
}),
|
|
67
|
+
Type.Object({
|
|
68
|
+
Script: Type.Object(
|
|
69
|
+
{
|
|
70
|
+
script_hash: Type.String(),
|
|
71
|
+
},
|
|
72
|
+
{ ctor: 6n },
|
|
73
|
+
),
|
|
74
|
+
}),
|
|
75
|
+
]),
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
export const MultisigScript = MultisigScriptModule.Import("MultisigScript");
|
|
79
|
+
export type TMultisigScript = TExact<typeof MultisigScript>;
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Provider settings schema for blockchain data providers.
|
|
83
|
+
*/
|
|
84
|
+
export const ProviderSettingsSchema = Type.Union([
|
|
85
|
+
Type.Object({
|
|
86
|
+
type: Type.Literal("blockfrost"),
|
|
87
|
+
projectId: Type.String({ minLength: 1, title: "Blockfrost Project ID" }),
|
|
88
|
+
}),
|
|
89
|
+
Type.Object({
|
|
90
|
+
type: Type.Literal("maestro"),
|
|
91
|
+
apiKey: Type.String({ minLength: 1, title: "Maestro API Key" }),
|
|
92
|
+
}),
|
|
93
|
+
]);
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Wallet settings schema for hot and cold wallets.
|
|
97
|
+
*/
|
|
98
|
+
export const WalletSettingsSchema = Type.Union([
|
|
99
|
+
Type.Object({
|
|
100
|
+
type: Type.Literal("hot"),
|
|
101
|
+
privateKey: Type.String({
|
|
102
|
+
minLength: 1,
|
|
103
|
+
title: "Hot Wallet Private Key",
|
|
104
|
+
sensitive: true,
|
|
105
|
+
}),
|
|
106
|
+
}),
|
|
107
|
+
Type.Object({
|
|
108
|
+
type: Type.Literal("cold"),
|
|
109
|
+
address: Type.String({ minLength: 1, title: "Cold Wallet Address" }),
|
|
110
|
+
}),
|
|
111
|
+
]);
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Transaction dialog helper utilities.
|
|
3
|
+
* Functions for working with transaction signatures and display.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { Core, HotWallet } from "@blaze-cardano/sdk";
|
|
7
|
+
import { CborSet, VkeyWitness, blake2b_256 } from "@blaze-cardano/core";
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Get the payment key hash from a HotWallet's first address.
|
|
11
|
+
*/
|
|
12
|
+
export async function getWalletPaymentKeyHash(
|
|
13
|
+
wallet: HotWallet,
|
|
14
|
+
): Promise<string | null> {
|
|
15
|
+
try {
|
|
16
|
+
const addresses = await wallet.getUsedAddresses();
|
|
17
|
+
const address = addresses[0];
|
|
18
|
+
if (!address) return null;
|
|
19
|
+
const paymentCredential = address.asBase()?.getPaymentCredential();
|
|
20
|
+
return paymentCredential?.hash?.toString() ?? null;
|
|
21
|
+
} catch {
|
|
22
|
+
return null;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Count the number of vkey signatures in a transaction's witness set.
|
|
28
|
+
*/
|
|
29
|
+
export function countSignatures(tx: Core.Transaction): number {
|
|
30
|
+
const vkeys = tx.witnessSet().vkeys();
|
|
31
|
+
return vkeys ? vkeys.size() : 0;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Check if a specific public key has already signed the transaction.
|
|
36
|
+
* Compares by vkey (public key bytes).
|
|
37
|
+
*/
|
|
38
|
+
export function hasVkeySigned(tx: Core.Transaction, vkeyHex: string): boolean {
|
|
39
|
+
const vkeys = tx.witnessSet().vkeys();
|
|
40
|
+
if (!vkeys) return false;
|
|
41
|
+
const vkeyArray = vkeys.toCore();
|
|
42
|
+
return vkeyArray.some(([vkey]) => vkey === vkeyHex);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Get the list of required signer key hashes from the transaction body.
|
|
47
|
+
*/
|
|
48
|
+
export function getRequiredSigners(tx: Core.Transaction): string[] {
|
|
49
|
+
const requiredSigners = tx.body().requiredSigners();
|
|
50
|
+
if (!requiredSigners) return [];
|
|
51
|
+
return Array.from(requiredSigners.values()).map((s) => s.toString());
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Compute the transaction body hash for display.
|
|
56
|
+
*/
|
|
57
|
+
export function getTxBodyHash(tx: Core.Transaction): string {
|
|
58
|
+
const bodyCbor = tx.body().toCbor();
|
|
59
|
+
return blake2b_256(bodyCbor);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Format a hash for display: first 8 chars + ... + last 8 chars.
|
|
64
|
+
*/
|
|
65
|
+
export function formatHash(hash: string): string {
|
|
66
|
+
if (hash.length <= 20) return hash;
|
|
67
|
+
return `${hash.slice(0, 8)}...${hash.slice(-8)}`;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Merge signatures from source transaction into target transaction.
|
|
72
|
+
* Prevents duplicate signatures by comparing vkey (public key).
|
|
73
|
+
* Returns the count of newly added signatures.
|
|
74
|
+
*/
|
|
75
|
+
export function mergeSignatures(
|
|
76
|
+
target: Core.Transaction,
|
|
77
|
+
source: Core.Transaction,
|
|
78
|
+
): number {
|
|
79
|
+
const targetWs = target.witnessSet();
|
|
80
|
+
const sourceWs = source.witnessSet();
|
|
81
|
+
|
|
82
|
+
const targetVkeys = targetWs.vkeys()?.toCore() ?? [];
|
|
83
|
+
const sourceVkeys = sourceWs.vkeys()?.toCore() ?? [];
|
|
84
|
+
|
|
85
|
+
// Find vkeys in source that aren't in target (by comparing public key)
|
|
86
|
+
const existingPubKeys = new Set(targetVkeys.map(([vkey]) => vkey));
|
|
87
|
+
const newVkeys = sourceVkeys.filter(([vkey]) => !existingPubKeys.has(vkey));
|
|
88
|
+
|
|
89
|
+
if (newVkeys.length === 0) {
|
|
90
|
+
return 0;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// Merge the new vkeys into target
|
|
94
|
+
targetWs.setVkeys(
|
|
95
|
+
CborSet.fromCore([...targetVkeys, ...newVkeys], VkeyWitness.fromCore),
|
|
96
|
+
);
|
|
97
|
+
target.setWitnessSet(targetWs);
|
|
98
|
+
|
|
99
|
+
return newVkeys.length;
|
|
100
|
+
}
|