@sundaeswap/sprinkles 0.5.0 → 0.6.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.
Files changed (112) hide show
  1. package/dist/cjs/Sprinkle/__tests__/encryption.test.js +3 -1
  2. package/dist/cjs/Sprinkle/__tests__/encryption.test.js.map +1 -1
  3. package/dist/cjs/Sprinkle/__tests__/enhancements.test.js +3 -37
  4. package/dist/cjs/Sprinkle/__tests__/enhancements.test.js.map +1 -1
  5. package/dist/cjs/Sprinkle/__tests__/field-utils.test.js +170 -0
  6. package/dist/cjs/Sprinkle/__tests__/field-utils.test.js.map +1 -0
  7. package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js +242 -87
  8. package/dist/cjs/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -1
  9. package/dist/cjs/Sprinkle/__tests__/formatting.test.js +97 -0
  10. package/dist/cjs/Sprinkle/__tests__/formatting.test.js.map +1 -0
  11. package/dist/cjs/Sprinkle/__tests__/show-menu.test.js +9 -5
  12. package/dist/cjs/Sprinkle/__tests__/show-menu.test.js.map +1 -1
  13. package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js +9 -0
  14. package/dist/cjs/Sprinkle/__tests__/tx-dialog.test.js.map +1 -1
  15. package/dist/cjs/Sprinkle/index.js +135 -91
  16. package/dist/cjs/Sprinkle/index.js.map +1 -1
  17. package/dist/cjs/Sprinkle/menus/array-menu.js +195 -0
  18. package/dist/cjs/Sprinkle/menus/array-menu.js.map +1 -0
  19. package/dist/cjs/Sprinkle/menus/field-menu.js +161 -0
  20. package/dist/cjs/Sprinkle/menus/field-menu.js.map +1 -0
  21. package/dist/cjs/Sprinkle/menus/index.js +33 -0
  22. package/dist/cjs/Sprinkle/menus/index.js.map +1 -0
  23. package/dist/cjs/Sprinkle/menus/object-menu.js +324 -0
  24. package/dist/cjs/Sprinkle/menus/object-menu.js.map +1 -0
  25. package/dist/cjs/Sprinkle/prompts.js +68 -2
  26. package/dist/cjs/Sprinkle/prompts.js.map +1 -1
  27. package/dist/cjs/Sprinkle/type-guards.js +48 -1
  28. package/dist/cjs/Sprinkle/type-guards.js.map +1 -1
  29. package/dist/cjs/Sprinkle/types.js +24 -0
  30. package/dist/cjs/Sprinkle/types.js.map +1 -1
  31. package/dist/cjs/Sprinkle/utils/field-utils.js +154 -0
  32. package/dist/cjs/Sprinkle/utils/field-utils.js.map +1 -0
  33. package/dist/cjs/Sprinkle/utils/formatting.js +126 -0
  34. package/dist/cjs/Sprinkle/utils/formatting.js.map +1 -0
  35. package/dist/cjs/Sprinkle/utils/index.js +56 -0
  36. package/dist/cjs/Sprinkle/utils/index.js.map +1 -0
  37. package/dist/esm/Sprinkle/__tests__/encryption.test.js +3 -1
  38. package/dist/esm/Sprinkle/__tests__/encryption.test.js.map +1 -1
  39. package/dist/esm/Sprinkle/__tests__/enhancements.test.js +3 -37
  40. package/dist/esm/Sprinkle/__tests__/enhancements.test.js.map +1 -1
  41. package/dist/esm/Sprinkle/__tests__/field-utils.test.js +168 -0
  42. package/dist/esm/Sprinkle/__tests__/field-utils.test.js.map +1 -0
  43. package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js +243 -88
  44. package/dist/esm/Sprinkle/__tests__/fill-in-struct.test.js.map +1 -1
  45. package/dist/esm/Sprinkle/__tests__/formatting.test.js +95 -0
  46. package/dist/esm/Sprinkle/__tests__/formatting.test.js.map +1 -0
  47. package/dist/esm/Sprinkle/__tests__/show-menu.test.js +9 -5
  48. package/dist/esm/Sprinkle/__tests__/show-menu.test.js.map +1 -1
  49. package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js +9 -0
  50. package/dist/esm/Sprinkle/__tests__/tx-dialog.test.js.map +1 -1
  51. package/dist/esm/Sprinkle/index.js +102 -93
  52. package/dist/esm/Sprinkle/index.js.map +1 -1
  53. package/dist/esm/Sprinkle/menus/array-menu.js +190 -0
  54. package/dist/esm/Sprinkle/menus/array-menu.js.map +1 -0
  55. package/dist/esm/Sprinkle/menus/field-menu.js +155 -0
  56. package/dist/esm/Sprinkle/menus/field-menu.js.map +1 -0
  57. package/dist/esm/Sprinkle/menus/index.js +8 -0
  58. package/dist/esm/Sprinkle/menus/index.js.map +1 -0
  59. package/dist/esm/Sprinkle/menus/object-menu.js +318 -0
  60. package/dist/esm/Sprinkle/menus/object-menu.js.map +1 -0
  61. package/dist/esm/Sprinkle/prompts.js +59 -1
  62. package/dist/esm/Sprinkle/prompts.js.map +1 -1
  63. package/dist/esm/Sprinkle/type-guards.js +42 -0
  64. package/dist/esm/Sprinkle/type-guards.js.map +1 -1
  65. package/dist/esm/Sprinkle/types.js +24 -0
  66. package/dist/esm/Sprinkle/types.js.map +1 -1
  67. package/dist/esm/Sprinkle/utils/field-utils.js +145 -0
  68. package/dist/esm/Sprinkle/utils/field-utils.js.map +1 -0
  69. package/dist/esm/Sprinkle/utils/formatting.js +118 -0
  70. package/dist/esm/Sprinkle/utils/formatting.js.map +1 -0
  71. package/dist/esm/Sprinkle/utils/index.js +7 -0
  72. package/dist/esm/Sprinkle/utils/index.js.map +1 -0
  73. package/dist/types/Sprinkle/index.d.ts +9 -3
  74. package/dist/types/Sprinkle/index.d.ts.map +1 -1
  75. package/dist/types/Sprinkle/menus/array-menu.d.ts +31 -0
  76. package/dist/types/Sprinkle/menus/array-menu.d.ts.map +1 -0
  77. package/dist/types/Sprinkle/menus/field-menu.d.ts +34 -0
  78. package/dist/types/Sprinkle/menus/field-menu.d.ts.map +1 -0
  79. package/dist/types/Sprinkle/menus/index.d.ts +10 -0
  80. package/dist/types/Sprinkle/menus/index.d.ts.map +1 -0
  81. package/dist/types/Sprinkle/menus/object-menu.d.ts +34 -0
  82. package/dist/types/Sprinkle/menus/object-menu.d.ts.map +1 -0
  83. package/dist/types/Sprinkle/prompts.d.ts +25 -0
  84. package/dist/types/Sprinkle/prompts.d.ts.map +1 -1
  85. package/dist/types/Sprinkle/type-guards.d.ts +24 -1
  86. package/dist/types/Sprinkle/type-guards.d.ts.map +1 -1
  87. package/dist/types/Sprinkle/types.d.ts +53 -0
  88. package/dist/types/Sprinkle/types.d.ts.map +1 -1
  89. package/dist/types/Sprinkle/utils/field-utils.d.ts +47 -0
  90. package/dist/types/Sprinkle/utils/field-utils.d.ts.map +1 -0
  91. package/dist/types/Sprinkle/utils/formatting.d.ts +30 -0
  92. package/dist/types/Sprinkle/utils/formatting.d.ts.map +1 -0
  93. package/dist/types/tsconfig.build.tsbuildinfo +1 -1
  94. package/package.json +1 -1
  95. package/src/Sprinkle/__tests__/encryption.test.ts +2 -0
  96. package/src/Sprinkle/__tests__/enhancements.test.ts +3 -42
  97. package/src/Sprinkle/__tests__/field-utils.test.ts +191 -0
  98. package/src/Sprinkle/__tests__/fill-in-struct.test.ts +252 -103
  99. package/src/Sprinkle/__tests__/formatting.test.ts +115 -0
  100. package/src/Sprinkle/__tests__/show-menu.test.ts +14 -8
  101. package/src/Sprinkle/__tests__/tx-dialog.test.ts +9 -0
  102. package/src/Sprinkle/index.ts +131 -119
  103. package/src/Sprinkle/menus/array-menu.ts +191 -0
  104. package/src/Sprinkle/menus/field-menu.ts +145 -0
  105. package/src/Sprinkle/menus/index.ts +12 -0
  106. package/src/Sprinkle/menus/object-menu.ts +336 -0
  107. package/src/Sprinkle/prompts.ts +71 -1
  108. package/src/Sprinkle/type-guards.ts +42 -0
  109. package/src/Sprinkle/types.ts +43 -0
  110. package/src/Sprinkle/utils/field-utils.ts +158 -0
  111. package/src/Sprinkle/utils/formatting.ts +127 -0
  112. package/src/Sprinkle/utils/index.ts +17 -0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sundaeswap/sprinkles",
3
- "version": "0.5.0",
3
+ "version": "0.6.0",
4
4
  "description": "A TypeScript library for building interactive CLI menus and TUI applications with TypeBox schema validation",
5
5
  "type": "module",
6
6
  "main": "./dist/cjs/index.js",
@@ -23,6 +23,8 @@ mock.module("../prompts.js", () => ({
23
23
  inputCancellable: mockInputCancellable,
24
24
  passwordCancellable: mockPasswordCancellable,
25
25
  confirmCancellable: mock(),
26
+ searchCancellable: mock(),
27
+ select: mockSelectCancellable,
26
28
  }));
27
29
 
28
30
  describe("Encryption & Sensitive Fields", () => {
@@ -136,45 +136,6 @@ describe("SearchSelect (2.3)", () => {
136
136
  });
137
137
  });
138
138
 
139
- describe("Optional type support (2.4)", () => {
140
- let sprinkle: Sprinkle<any>;
141
-
142
- beforeEach(() => {
143
- const schema = Type.Object({ placeholder: Type.String() });
144
- sprinkle = new Sprinkle(schema, "/tmp/test");
145
- mockSelect.mockClear();
146
- mockInput.mockClear();
147
- mockSelectCancellable.mockClear();
148
- mockInputCancellable.mockClear();
149
- });
150
-
151
- test("skips optional field when user selects Skip", async () => {
152
- const schema = Type.Object({
153
- name: Type.String(),
154
- nickname: Type.Optional(Type.String()),
155
- });
156
-
157
- mockInputCancellable.mockResolvedValueOnce("Alice"); // name
158
- mockSelectCancellable.mockResolvedValueOnce(false); // skip nickname
159
-
160
- const result = await sprinkle.FillInStruct(schema);
161
- expect(result.name).toBe("Alice");
162
- expect(result.nickname).toBeUndefined();
163
- });
164
-
165
- test("fills optional field when user selects Yes", async () => {
166
- const schema = Type.Object({
167
- name: Type.String(),
168
- nickname: Type.Optional(Type.String()),
169
- });
170
-
171
- mockInputCancellable
172
- .mockResolvedValueOnce("Alice") // name
173
- .mockResolvedValueOnce("Ali"); // nickname
174
- mockSelectCancellable.mockResolvedValueOnce(true); // fill nickname
175
-
176
- const result = await sprinkle.FillInStruct(schema);
177
- expect(result.name).toBe("Alice");
178
- expect(result.nickname).toBe("Ali");
179
- });
180
- });
139
+ // Note: "Optional type support (2.4)" tests removed - they tested the old sequential
140
+ // FillInStruct behavior. The new menu-based FillInStruct handles optional fields
141
+ // differently (via menu selection). See fill-in-struct.test.ts for comprehensive tests.
@@ -0,0 +1,191 @@
1
+ import { describe, expect, test } from "bun:test";
2
+ import { Type } from "@sinclair/typebox";
3
+ import {
4
+ countRequiredFields,
5
+ isFieldRequired,
6
+ buildFieldLabel,
7
+ getInitialFieldState,
8
+ allRequiredFieldsFilled,
9
+ } from "../utils/field-utils.js";
10
+ import type { FieldState } from "../types.js";
11
+
12
+ describe("isFieldRequired", () => {
13
+ test("returns true for plain required field", () => {
14
+ expect(isFieldRequired(Type.String())).toBe(true);
15
+ });
16
+
17
+ test("returns false for optional field", () => {
18
+ expect(isFieldRequired(Type.Optional(Type.String()))).toBe(false);
19
+ });
20
+
21
+ test("returns false for field with default", () => {
22
+ expect(isFieldRequired(Type.String({ default: "hello" }))).toBe(false);
23
+ });
24
+
25
+ test("returns true for nullable field with default", () => {
26
+ // Nullable fields with defaults are required because the user should
27
+ // explicitly choose whether to use the value or set null
28
+ const schema = Type.Union([Type.String({ default: "hello" }), Type.Null()]);
29
+ expect(isFieldRequired(schema)).toBe(true);
30
+ });
31
+ });
32
+
33
+ describe("countRequiredFields", () => {
34
+ test("counts fields in simple object", () => {
35
+ const schema = Type.Object({
36
+ name: Type.String(),
37
+ age: Type.BigInt(),
38
+ });
39
+ const state = new Map<string, FieldState>();
40
+
41
+ const result = countRequiredFields(schema, state);
42
+ expect(result.total).toBe(2);
43
+ expect(result.filled).toBe(0);
44
+ });
45
+
46
+ test("skips optional fields", () => {
47
+ const schema = Type.Object({
48
+ name: Type.String(),
49
+ nickname: Type.Optional(Type.String()),
50
+ });
51
+ const state = new Map<string, FieldState>();
52
+
53
+ const result = countRequiredFields(schema, state);
54
+ expect(result.total).toBe(1);
55
+ });
56
+
57
+ test("skips fields with defaults", () => {
58
+ const schema = Type.Object({
59
+ name: Type.String(),
60
+ timeout: Type.BigInt({ default: 30n }),
61
+ });
62
+ const state = new Map<string, FieldState>();
63
+
64
+ const result = countRequiredFields(schema, state);
65
+ expect(result.total).toBe(1);
66
+ });
67
+
68
+ test("counts filled fields", () => {
69
+ const schema = Type.Object({
70
+ name: Type.String(),
71
+ age: Type.BigInt(),
72
+ });
73
+ const state = new Map<string, FieldState>([
74
+ ["name", { status: "set", value: "Alice" }],
75
+ ]);
76
+
77
+ const result = countRequiredFields(schema, state);
78
+ expect(result.total).toBe(2);
79
+ expect(result.filled).toBe(1);
80
+ });
81
+
82
+ test("counts null as filled for nullable fields", () => {
83
+ const schema = Type.Object({
84
+ name: Type.Union([Type.String(), Type.Null()]),
85
+ });
86
+ const state = new Map<string, FieldState>([["name", { status: "null" }]]);
87
+
88
+ const result = countRequiredFields(schema, state);
89
+ expect(result.filled).toBe(1);
90
+ });
91
+ });
92
+
93
+ describe("buildFieldLabel", () => {
94
+ test("shows [not set] with asterisk for required unset field", () => {
95
+ const label = buildFieldLabel("name", Type.String(), { status: "unset" });
96
+ expect(label).toContain("name");
97
+ expect(label).toContain("[not set]");
98
+ expect(label).toContain("*");
99
+ });
100
+
101
+ test("shows [not set] without asterisk for optional field", () => {
102
+ const label = buildFieldLabel("nickname", Type.Optional(Type.String()), {
103
+ status: "unset",
104
+ });
105
+ expect(label).toContain("[not set]");
106
+ expect(label).not.toContain("*");
107
+ });
108
+
109
+ test("shows default value for unset field with default", () => {
110
+ const label = buildFieldLabel(
111
+ "timeout",
112
+ Type.BigInt({ default: 30n }),
113
+ { status: "unset" },
114
+ );
115
+ expect(label).toContain("default");
116
+ expect(label).toContain("30");
117
+ });
118
+
119
+ test("shows null for null state", () => {
120
+ const label = buildFieldLabel(
121
+ "value",
122
+ Type.Union([Type.String(), Type.Null()]),
123
+ { status: "null" },
124
+ );
125
+ expect(label).toContain("null");
126
+ });
127
+
128
+ test("shows value preview for set field", () => {
129
+ const label = buildFieldLabel("name", Type.String(), {
130
+ status: "set",
131
+ value: "Alice",
132
+ });
133
+ expect(label).toContain("Alice");
134
+ });
135
+ });
136
+
137
+ describe("getInitialFieldState", () => {
138
+ test("returns unset for undefined", () => {
139
+ expect(getInitialFieldState(Type.String(), undefined)).toEqual({
140
+ status: "unset",
141
+ });
142
+ });
143
+
144
+ test("returns null for null value", () => {
145
+ expect(getInitialFieldState(Type.String(), null)).toEqual({
146
+ status: "null",
147
+ });
148
+ });
149
+
150
+ test("returns set with value", () => {
151
+ expect(getInitialFieldState(Type.String(), "hello")).toEqual({
152
+ status: "set",
153
+ value: "hello",
154
+ });
155
+ });
156
+ });
157
+
158
+ describe("allRequiredFieldsFilled", () => {
159
+ test("returns true when all required fields filled", () => {
160
+ const schema = Type.Object({
161
+ name: Type.String(),
162
+ nickname: Type.Optional(Type.String()),
163
+ });
164
+ const state = new Map<string, FieldState>([
165
+ ["name", { status: "set", value: "Alice" }],
166
+ ]);
167
+
168
+ expect(allRequiredFieldsFilled(schema, state)).toBe(true);
169
+ });
170
+
171
+ test("returns false when required field missing", () => {
172
+ const schema = Type.Object({
173
+ name: Type.String(),
174
+ age: Type.BigInt(),
175
+ });
176
+ const state = new Map<string, FieldState>([
177
+ ["name", { status: "set", value: "Alice" }],
178
+ ]);
179
+
180
+ expect(allRequiredFieldsFilled(schema, state)).toBe(false);
181
+ });
182
+
183
+ test("returns true for all-optional schema", () => {
184
+ const schema = Type.Object({
185
+ nickname: Type.Optional(Type.String()),
186
+ });
187
+ const state = new Map<string, FieldState>();
188
+
189
+ expect(allRequiredFieldsFilled(schema, state)).toBe(true);
190
+ });
191
+ });