@wp-typia/project-tools 0.19.0 → 0.19.2

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 (39) hide show
  1. package/dist/runtime/block-generator-service-spec.js +206 -2
  2. package/dist/runtime/built-in-block-artifacts.js +3 -1
  3. package/dist/runtime/built-in-block-code-artifacts.js +3 -1
  4. package/dist/runtime/built-in-block-non-ts-artifacts.js +1 -1039
  5. package/dist/runtime/built-in-block-non-ts-family-artifacts.d.ts +30 -0
  6. package/dist/runtime/built-in-block-non-ts-family-artifacts.js +1035 -0
  7. package/dist/runtime/built-in-block-non-ts-render-utils.d.ts +27 -0
  8. package/dist/runtime/built-in-block-non-ts-render-utils.js +51 -0
  9. package/dist/runtime/cli-add-block.js +11 -5
  10. package/dist/runtime/cli-add-workspace-rest-anchors.d.ts +3 -0
  11. package/dist/runtime/cli-add-workspace-rest-anchors.js +188 -0
  12. package/dist/runtime/cli-add-workspace-rest-source-emitters.d.ts +7 -0
  13. package/dist/runtime/cli-add-workspace-rest-source-emitters.js +379 -0
  14. package/dist/runtime/cli-add-workspace-rest.js +5 -564
  15. package/dist/runtime/cli-diagnostics.d.ts +26 -0
  16. package/dist/runtime/cli-diagnostics.js +107 -0
  17. package/dist/runtime/cli-doctor-workspace.js +4 -4
  18. package/dist/runtime/cli-scaffold.js +3 -3
  19. package/dist/runtime/index.d.ts +3 -1
  20. package/dist/runtime/index.js +2 -1
  21. package/dist/runtime/scaffold-bootstrap.js +5 -1
  22. package/dist/runtime/scaffold-documents.js +11 -8
  23. package/dist/runtime/scaffold-template-variable-groups.d.ts +154 -0
  24. package/dist/runtime/scaffold-template-variable-groups.js +13 -0
  25. package/dist/runtime/scaffold-template-variables.js +58 -1
  26. package/dist/runtime/scaffold.d.ts +5 -1
  27. package/dist/runtime/scaffold.js +5 -1
  28. package/dist/runtime/temp-roots.d.ts +44 -0
  29. package/dist/runtime/temp-roots.js +129 -0
  30. package/dist/runtime/template-builtins.js +4 -6
  31. package/dist/runtime/template-registry.d.ts +8 -0
  32. package/dist/runtime/template-registry.js +34 -1
  33. package/dist/runtime/template-source-external.d.ts +1 -0
  34. package/dist/runtime/template-source-external.js +4 -7
  35. package/dist/runtime/template-source-remote.js +44 -23
  36. package/dist/runtime/template-source-seeds.js +3 -9
  37. package/dist/runtime/template-source.d.ts +2 -3
  38. package/dist/runtime/template-source.js +8 -2
  39. package/package.json +6 -1
@@ -0,0 +1,379 @@
1
+ import { normalizeBlockSlug, quoteTsString, } from "./cli-add-shared.js";
2
+ import { buildRestResourceEndpointManifest } from "./rest-resource-artifacts.js";
3
+ import { toTitleCase } from "./string-case.js";
4
+ export function toPascalCaseFromSlug(slug) {
5
+ return normalizeBlockSlug(slug)
6
+ .split("-")
7
+ .filter(Boolean)
8
+ .map((segment) => segment.charAt(0).toUpperCase() + segment.slice(1))
9
+ .join("");
10
+ }
11
+ function indentMultiline(source, prefix) {
12
+ return source
13
+ .split("\n")
14
+ .map((line) => `${prefix}${line}`)
15
+ .join("\n");
16
+ }
17
+ export function buildRestResourceConfigEntry(restResourceSlug, namespace, methods) {
18
+ const pascalCase = toPascalCaseFromSlug(restResourceSlug);
19
+ const title = toTitleCase(restResourceSlug);
20
+ const manifest = buildRestResourceEndpointManifest({
21
+ namespace,
22
+ pascalCase,
23
+ slugKebabCase: restResourceSlug,
24
+ title,
25
+ }, methods);
26
+ return [
27
+ "\t{",
28
+ `\t\tapiFile: ${quoteTsString(`src/rest/${restResourceSlug}/api.ts`)},`,
29
+ `\t\tclientFile: ${quoteTsString(`src/rest/${restResourceSlug}/api-client.ts`)},`,
30
+ `\t\tdataFile: ${quoteTsString(`src/rest/${restResourceSlug}/data.ts`)},`,
31
+ `\t\tmethods: [ ${methods.map((method) => quoteTsString(method)).join(", ")} ],`,
32
+ `\t\tnamespace: ${quoteTsString(namespace)},`,
33
+ `\t\topenApiFile: ${quoteTsString(`src/rest/${restResourceSlug}/api.openapi.json`)},`,
34
+ `\t\tphpFile: ${quoteTsString(`inc/rest/${restResourceSlug}.php`)},`,
35
+ "\t\trestManifest: defineEndpointManifest(",
36
+ indentMultiline(JSON.stringify(manifest, null, "\t"), "\t\t\t"),
37
+ "\t\t),",
38
+ `\t\tslug: ${quoteTsString(restResourceSlug)},`,
39
+ `\t\ttypesFile: ${quoteTsString(`src/rest/${restResourceSlug}/api-types.ts`)},`,
40
+ `\t\tvalidatorsFile: ${quoteTsString(`src/rest/${restResourceSlug}/api-validators.ts`)},`,
41
+ "\t},",
42
+ ].join("\n");
43
+ }
44
+ export function buildRestResourceTypesSource(restResourceSlug, methods) {
45
+ const pascalCase = toPascalCaseFromSlug(restResourceSlug);
46
+ const lines = [
47
+ "import { tags } from 'typia';",
48
+ "",
49
+ `export type ${pascalCase}Status = 'draft' | 'published';`,
50
+ "",
51
+ `export interface ${pascalCase}Record {`,
52
+ "\tid: number & tags.Type< 'uint32' >;",
53
+ "\ttitle: string & tags.MinLength< 1 > & tags.MaxLength< 120 >;",
54
+ "\tcontent?: string & tags.MaxLength< 2000 >;",
55
+ `\tstatus: ${pascalCase}Status;`,
56
+ "\tupdatedAt: string;",
57
+ "}",
58
+ ];
59
+ if (methods.includes("list")) {
60
+ lines.push("", `export interface ${pascalCase}ListQuery {`, "\tpage?: number & tags.Type< 'uint32' > & tags.Minimum< 1 > & tags.Default< 1 >;", "\tperPage?: number & tags.Type< 'uint32' > & tags.Minimum< 1 > & tags.Maximum< 50 > & tags.Default< 10 >;", "\tsearch?: string & tags.MaxLength< 120 >;", "}", "", `export interface ${pascalCase}ListResponse {`, `\titems: ${pascalCase}Record[];`, "\tpage: number & tags.Type< 'uint32' >;", "\tperPage: number & tags.Type< 'uint32' >;", "\ttotal: number & tags.Type< 'uint32' >;", "}");
61
+ }
62
+ if (methods.includes("read")) {
63
+ lines.push("", `export interface ${pascalCase}ReadQuery {`, "\tid: number & tags.Type< 'uint32' >;", "}", "", `export type ${pascalCase}ReadResponse = ${pascalCase}Record;`);
64
+ }
65
+ if (methods.includes("create")) {
66
+ lines.push("", `export interface ${pascalCase}CreateRequest {`, "\ttitle: string & tags.MinLength< 1 > & tags.MaxLength< 120 >;", "\tcontent?: string & tags.MaxLength< 2000 >;", `\tstatus?: ${pascalCase}Status;`, "}", "", `export type ${pascalCase}CreateResponse = ${pascalCase}Record;`);
67
+ }
68
+ if (methods.includes("update")) {
69
+ lines.push("", `export interface ${pascalCase}UpdateQuery {`, "\tid: number & tags.Type< 'uint32' >;", "}", "", `export interface ${pascalCase}UpdateRequest {`, "\ttitle?: string & tags.MinLength< 1 > & tags.MaxLength< 120 >;", "\tcontent?: string & tags.MaxLength< 2000 >;", `\tstatus?: ${pascalCase}Status;`, "}", "", `export type ${pascalCase}UpdateResponse = ${pascalCase}Record;`);
70
+ }
71
+ if (methods.includes("delete")) {
72
+ lines.push("", `export interface ${pascalCase}DeleteQuery {`, "\tid: number & tags.Type< 'uint32' >;", "}", "", `export interface ${pascalCase}DeleteResponse {`, "\tdeleted: true;", "\tid: number & tags.Type< 'uint32' >;", "}");
73
+ }
74
+ return `${lines.join("\n")}\n`;
75
+ }
76
+ export function buildRestResourceValidatorsSource(restResourceSlug, methods) {
77
+ const pascalCase = toPascalCaseFromSlug(restResourceSlug);
78
+ const importedTypes = new Set();
79
+ const validatorDeclarations = [];
80
+ const validatorEntries = [];
81
+ const addValidator = (propertyName, typeName, validateIdentifier) => {
82
+ importedTypes.add(typeName);
83
+ validatorDeclarations.push(`const ${validateIdentifier} = typia.createValidate< ${typeName} >();`);
84
+ validatorEntries.push(`\t${propertyName}: ( input: unknown ) => toValidationResult< ${typeName} >( ${validateIdentifier}( input ) ),`);
85
+ };
86
+ if (methods.includes("list")) {
87
+ addValidator("listQuery", `${pascalCase}ListQuery`, "validateListQuery");
88
+ addValidator("listResponse", `${pascalCase}ListResponse`, "validateListResponse");
89
+ }
90
+ if (methods.includes("read")) {
91
+ addValidator("readQuery", `${pascalCase}ReadQuery`, "validateReadQuery");
92
+ addValidator("readResponse", `${pascalCase}ReadResponse`, "validateReadResponse");
93
+ }
94
+ if (methods.includes("create")) {
95
+ addValidator("createRequest", `${pascalCase}CreateRequest`, "validateCreateRequest");
96
+ addValidator("createResponse", `${pascalCase}CreateResponse`, "validateCreateResponse");
97
+ }
98
+ if (methods.includes("update")) {
99
+ addValidator("updateQuery", `${pascalCase}UpdateQuery`, "validateUpdateQuery");
100
+ addValidator("updateRequest", `${pascalCase}UpdateRequest`, "validateUpdateRequest");
101
+ addValidator("updateResponse", `${pascalCase}UpdateResponse`, "validateUpdateResponse");
102
+ }
103
+ if (methods.includes("delete")) {
104
+ addValidator("deleteQuery", `${pascalCase}DeleteQuery`, "validateDeleteQuery");
105
+ addValidator("deleteResponse", `${pascalCase}DeleteResponse`, "validateDeleteResponse");
106
+ }
107
+ return `import typia from 'typia';
108
+
109
+ import { toValidationResult } from '@wp-typia/rest';
110
+ import type {
111
+ \t${Array.from(importedTypes).sort().join(",\n\t")},
112
+ } from './api-types';
113
+
114
+ ${validatorDeclarations.join("\n")}
115
+
116
+ export const apiValidators = {
117
+ ${validatorEntries.join("\n")}
118
+ };
119
+ `;
120
+ }
121
+ export function buildRestResourceApiSource(restResourceSlug, methods) {
122
+ const pascalCase = toPascalCaseFromSlug(restResourceSlug);
123
+ const typeImports = new Set();
124
+ const clientEndpointImports = [];
125
+ const exportedBindings = [];
126
+ const writeMethods = methods.filter((method) => ["create", "update", "delete"].includes(method));
127
+ if (methods.includes("list")) {
128
+ typeImports.add(`${pascalCase}ListQuery`);
129
+ clientEndpointImports.push(`list${pascalCase}ResourcesEndpoint`);
130
+ exportedBindings.push(`export const restResourceListEndpoint = {
131
+ \t...list${pascalCase}ResourcesEndpoint,
132
+ \tbuildRequestOptions: () => ( {
133
+ \t\turl: resolveRestRouteUrl( list${pascalCase}ResourcesEndpoint.path ),
134
+ \t} ),
135
+ };
136
+
137
+ export function listResource( request: ${pascalCase}ListQuery ) {
138
+ \treturn callEndpoint( restResourceListEndpoint, request );
139
+ }`);
140
+ }
141
+ if (methods.includes("read")) {
142
+ typeImports.add(`${pascalCase}ReadQuery`);
143
+ clientEndpointImports.push(`read${pascalCase}ResourceEndpoint`);
144
+ exportedBindings.push(`export const restResourceReadEndpoint = {
145
+ \t...read${pascalCase}ResourceEndpoint,
146
+ \tbuildRequestOptions: () => ( {
147
+ \t\turl: resolveRestRouteUrl( read${pascalCase}ResourceEndpoint.path ),
148
+ \t} ),
149
+ };
150
+
151
+ export function readResource( request: ${pascalCase}ReadQuery ) {
152
+ \treturn callEndpoint( restResourceReadEndpoint, request );
153
+ }`);
154
+ }
155
+ if (methods.includes("create")) {
156
+ typeImports.add(`${pascalCase}CreateRequest`);
157
+ clientEndpointImports.push(`create${pascalCase}ResourceEndpoint`);
158
+ exportedBindings.push(`export const restResourceCreateEndpoint = {
159
+ \t...create${pascalCase}ResourceEndpoint,
160
+ \tbuildRequestOptions: () => {
161
+ \t\tconst nonce = resolveRestNonce();
162
+ \t\treturn {
163
+ \t\t\theaders: nonce
164
+ \t\t\t\t? {
165
+ \t\t\t\t\t'X-WP-Nonce': nonce,
166
+ \t\t\t\t}
167
+ \t\t\t\t: undefined,
168
+ \t\t\turl: resolveRestRouteUrl( create${pascalCase}ResourceEndpoint.path ),
169
+ \t\t};
170
+ \t},
171
+ };
172
+
173
+ export function createResource( request: ${pascalCase}CreateRequest ) {
174
+ \treturn callEndpoint( restResourceCreateEndpoint, request );
175
+ }`);
176
+ }
177
+ if (methods.includes("update")) {
178
+ typeImports.add(`${pascalCase}UpdateQuery`);
179
+ typeImports.add(`${pascalCase}UpdateRequest`);
180
+ clientEndpointImports.push(`update${pascalCase}ResourceEndpoint`);
181
+ exportedBindings.push(`export const restResourceUpdateEndpoint = {
182
+ \t...update${pascalCase}ResourceEndpoint,
183
+ \tbuildRequestOptions: () => {
184
+ \t\tconst nonce = resolveRestNonce();
185
+ \t\treturn {
186
+ \t\t\theaders: nonce
187
+ \t\t\t\t? {
188
+ \t\t\t\t\t'X-WP-Nonce': nonce,
189
+ \t\t\t\t}
190
+ \t\t\t\t: undefined,
191
+ \t\t\turl: resolveRestRouteUrl( update${pascalCase}ResourceEndpoint.path ),
192
+ \t\t};
193
+ \t},
194
+ };
195
+
196
+ export function updateResource( request: {
197
+ \tbody: ${pascalCase}UpdateRequest;
198
+ \tquery: ${pascalCase}UpdateQuery;
199
+ } ) {
200
+ \treturn callEndpoint( restResourceUpdateEndpoint, request );
201
+ }`);
202
+ }
203
+ if (methods.includes("delete")) {
204
+ typeImports.add(`${pascalCase}DeleteQuery`);
205
+ clientEndpointImports.push(`delete${pascalCase}ResourceEndpoint`);
206
+ exportedBindings.push(`export const restResourceDeleteEndpoint = {
207
+ \t...delete${pascalCase}ResourceEndpoint,
208
+ \tbuildRequestOptions: () => {
209
+ \t\tconst nonce = resolveRestNonce();
210
+ \t\treturn {
211
+ \t\t\theaders: nonce
212
+ \t\t\t\t? {
213
+ \t\t\t\t\t'X-WP-Nonce': nonce,
214
+ \t\t\t\t}
215
+ \t\t\t\t: undefined,
216
+ \t\t\turl: resolveRestRouteUrl( delete${pascalCase}ResourceEndpoint.path ),
217
+ \t\t};
218
+ \t},
219
+ };
220
+
221
+ export function deleteResource( request: ${pascalCase}DeleteQuery ) {
222
+ \treturn callEndpoint( restResourceDeleteEndpoint, request );
223
+ }`);
224
+ }
225
+ const resolveRestNonceSource = writeMethods.length > 0
226
+ ? `function resolveRestNonce( fallback?: string ): string | undefined {
227
+ \tif ( typeof fallback === 'string' && fallback.length > 0 ) {
228
+ \t\treturn fallback;
229
+ \t}
230
+
231
+ \tif ( typeof window === 'undefined' ) {
232
+ \t\treturn undefined;
233
+ \t}
234
+
235
+ \tconst wpApiSettings = (
236
+ \t\twindow as typeof window & {
237
+ \t\t\twpApiSettings?: { nonce?: string };
238
+ \t\t}
239
+ \t).wpApiSettings;
240
+
241
+ \treturn typeof wpApiSettings?.nonce === 'string' &&
242
+ \t\twpApiSettings.nonce.length > 0
243
+ \t\t? wpApiSettings.nonce
244
+ \t\t: undefined;
245
+ }
246
+
247
+ `
248
+ : "";
249
+ return `import {
250
+ \tcallEndpoint,
251
+ \tresolveRestRouteUrl,
252
+ } from '@wp-typia/rest';
253
+
254
+ import type {
255
+ \t${Array.from(typeImports).sort().join(",\n\t")},
256
+ } from './api-types';
257
+ import {
258
+ \t${clientEndpointImports.sort().join(",\n\t")},
259
+ } from './api-client';
260
+ ${resolveRestNonceSource}
261
+ ${exportedBindings.join("\n\n")}
262
+ `;
263
+ }
264
+ export function buildRestResourceDataSource(restResourceSlug, methods) {
265
+ const pascalCase = toPascalCaseFromSlug(restResourceSlug);
266
+ const typeImports = new Set();
267
+ const endpointImports = [];
268
+ const exportedBindings = [];
269
+ if (methods.includes("list")) {
270
+ typeImports.add(`${pascalCase}ListQuery`);
271
+ typeImports.add(`${pascalCase}ListResponse`);
272
+ endpointImports.push("restResourceListEndpoint");
273
+ exportedBindings.push(`export type Use${pascalCase}ListQueryOptions<
274
+ \tSelected = ${pascalCase}ListResponse,
275
+ > = UseEndpointQueryOptions<
276
+ \t${pascalCase}ListQuery,
277
+ \t${pascalCase}ListResponse,
278
+ \tSelected
279
+ >;
280
+
281
+ export function use${pascalCase}ListQuery<
282
+ \tSelected = ${pascalCase}ListResponse,
283
+ >(
284
+ \trequest: ${pascalCase}ListQuery,
285
+ \toptions: Use${pascalCase}ListQueryOptions< Selected > = {}
286
+ ) {
287
+ \treturn useEndpointQuery( restResourceListEndpoint, request, options );
288
+ }`);
289
+ }
290
+ if (methods.includes("read")) {
291
+ typeImports.add(`${pascalCase}ReadQuery`);
292
+ typeImports.add(`${pascalCase}ReadResponse`);
293
+ endpointImports.push("restResourceReadEndpoint");
294
+ exportedBindings.push(`export type Use${pascalCase}ReadQueryOptions<
295
+ \tSelected = ${pascalCase}ReadResponse,
296
+ > = UseEndpointQueryOptions<
297
+ \t${pascalCase}ReadQuery,
298
+ \t${pascalCase}ReadResponse,
299
+ \tSelected
300
+ >;
301
+
302
+ export function use${pascalCase}ReadQuery<
303
+ \tSelected = ${pascalCase}ReadResponse,
304
+ >(
305
+ \trequest: ${pascalCase}ReadQuery,
306
+ \toptions: Use${pascalCase}ReadQueryOptions< Selected > = {}
307
+ ) {
308
+ \treturn useEndpointQuery( restResourceReadEndpoint, request, options );
309
+ }`);
310
+ }
311
+ if (methods.includes("create")) {
312
+ typeImports.add(`${pascalCase}CreateRequest`);
313
+ typeImports.add(`${pascalCase}CreateResponse`);
314
+ endpointImports.push("restResourceCreateEndpoint");
315
+ exportedBindings.push(`export type UseCreate${pascalCase}ResourceMutationOptions = UseEndpointMutationOptions<
316
+ \t${pascalCase}CreateRequest,
317
+ \t${pascalCase}CreateResponse,
318
+ \tunknown
319
+ >;
320
+
321
+ export function useCreate${pascalCase}ResourceMutation(
322
+ \toptions: UseCreate${pascalCase}ResourceMutationOptions = {}
323
+ ) {
324
+ \treturn useEndpointMutation( restResourceCreateEndpoint, options );
325
+ }`);
326
+ }
327
+ if (methods.includes("update")) {
328
+ typeImports.add(`${pascalCase}UpdateQuery`);
329
+ typeImports.add(`${pascalCase}UpdateRequest`);
330
+ typeImports.add(`${pascalCase}UpdateResponse`);
331
+ endpointImports.push("restResourceUpdateEndpoint");
332
+ exportedBindings.push(`export type UseUpdate${pascalCase}ResourceMutationOptions = UseEndpointMutationOptions<
333
+ \t{
334
+ \t\tbody: ${pascalCase}UpdateRequest;
335
+ \t\tquery: ${pascalCase}UpdateQuery;
336
+ \t},
337
+ \t${pascalCase}UpdateResponse,
338
+ \tunknown
339
+ >;
340
+
341
+ export function useUpdate${pascalCase}ResourceMutation(
342
+ \toptions: UseUpdate${pascalCase}ResourceMutationOptions = {}
343
+ ) {
344
+ \treturn useEndpointMutation( restResourceUpdateEndpoint, options );
345
+ }`);
346
+ }
347
+ if (methods.includes("delete")) {
348
+ typeImports.add(`${pascalCase}DeleteQuery`);
349
+ typeImports.add(`${pascalCase}DeleteResponse`);
350
+ endpointImports.push("restResourceDeleteEndpoint");
351
+ exportedBindings.push(`export type UseDelete${pascalCase}ResourceMutationOptions = UseEndpointMutationOptions<
352
+ \t${pascalCase}DeleteQuery,
353
+ \t${pascalCase}DeleteResponse,
354
+ \tunknown
355
+ >;
356
+
357
+ export function useDelete${pascalCase}ResourceMutation(
358
+ \toptions: UseDelete${pascalCase}ResourceMutationOptions = {}
359
+ ) {
360
+ \treturn useEndpointMutation( restResourceDeleteEndpoint, options );
361
+ }`);
362
+ }
363
+ return `import {
364
+ \tuseEndpointMutation,
365
+ \tuseEndpointQuery,
366
+ \ttype UseEndpointMutationOptions,
367
+ \ttype UseEndpointQueryOptions,
368
+ } from '@wp-typia/rest/react';
369
+
370
+ import type {
371
+ \t${Array.from(typeImports).sort().join(",\n\t")},
372
+ } from './api-types';
373
+ import {
374
+ \t${endpointImports.sort().join(",\n\t")},
375
+ } from './api';
376
+
377
+ ${exportedBindings.join("\n\n")}
378
+ `;
379
+ }