sonamu 0.7.21 → 0.7.23

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 (200) hide show
  1. package/dist/ai/agents/agent.d.ts +6 -1
  2. package/dist/ai/agents/agent.d.ts.map +1 -1
  3. package/dist/ai/agents/agent.js +20 -5
  4. package/dist/api/base-frame.d.ts +4 -0
  5. package/dist/api/base-frame.d.ts.map +1 -1
  6. package/dist/api/base-frame.js +9 -1
  7. package/dist/api/caster.d.ts.map +1 -1
  8. package/dist/api/caster.js +2 -2
  9. package/dist/api/config.d.ts +35 -3
  10. package/dist/api/config.d.ts.map +1 -1
  11. package/dist/api/config.js +1 -1
  12. package/dist/api/decorators.d.ts +4 -4
  13. package/dist/api/decorators.d.ts.map +1 -1
  14. package/dist/api/decorators.js +80 -18
  15. package/dist/api/index.d.ts +1 -0
  16. package/dist/api/index.d.ts.map +1 -1
  17. package/dist/api/index.js +2 -1
  18. package/dist/api/secret.d.ts +7 -0
  19. package/dist/api/secret.d.ts.map +1 -0
  20. package/dist/api/secret.js +17 -0
  21. package/dist/api/sonamu.d.ts +17 -8
  22. package/dist/api/sonamu.d.ts.map +1 -1
  23. package/dist/api/sonamu.js +265 -47
  24. package/dist/cache/cache-manager.d.ts +11 -0
  25. package/dist/cache/cache-manager.d.ts.map +1 -0
  26. package/dist/cache/cache-manager.js +22 -0
  27. package/dist/cache/decorator.d.ts +31 -0
  28. package/dist/cache/decorator.d.ts.map +1 -0
  29. package/dist/cache/decorator.js +86 -0
  30. package/dist/cache/drivers.d.ts +33 -0
  31. package/dist/cache/drivers.d.ts.map +1 -0
  32. package/dist/cache/drivers.js +36 -0
  33. package/dist/cache/index.d.ts +4 -0
  34. package/dist/cache/index.d.ts.map +1 -0
  35. package/dist/cache/index.js +8 -0
  36. package/dist/cache/types.d.ts +28 -0
  37. package/dist/cache/types.d.ts.map +1 -0
  38. package/dist/cache/types.js +6 -0
  39. package/dist/database/base-model.d.ts +4 -2
  40. package/dist/database/base-model.d.ts.map +1 -1
  41. package/dist/database/base-model.js +9 -4
  42. package/dist/database/code-generator.d.ts +3 -1
  43. package/dist/database/code-generator.d.ts.map +1 -1
  44. package/dist/database/code-generator.js +3 -2
  45. package/dist/database/db.d.ts +1 -1
  46. package/dist/database/db.d.ts.map +1 -1
  47. package/dist/database/db.js +5 -5
  48. package/dist/database/knex.d.ts +3 -0
  49. package/dist/database/knex.d.ts.map +1 -0
  50. package/dist/database/knex.js +29 -0
  51. package/dist/database/puri.types.d.ts.map +1 -1
  52. package/dist/database/puri.types.js +1 -1
  53. package/dist/database/upsert-builder.d.ts.map +1 -1
  54. package/dist/database/upsert-builder.js +49 -5
  55. package/dist/index.d.ts +4 -0
  56. package/dist/index.d.ts.map +1 -1
  57. package/dist/index.js +4 -1
  58. package/dist/logger/category.d.ts +4 -0
  59. package/dist/logger/category.d.ts.map +1 -0
  60. package/dist/logger/category.js +34 -0
  61. package/dist/logger/configure.d.ts +9 -0
  62. package/dist/logger/configure.d.ts.map +1 -0
  63. package/dist/logger/configure.js +115 -0
  64. package/dist/migration/code-generation.d.ts +5 -1
  65. package/dist/migration/code-generation.d.ts.map +1 -1
  66. package/dist/migration/code-generation.js +13 -7
  67. package/dist/migration/migrator.d.ts +1 -1
  68. package/dist/migration/migrator.d.ts.map +1 -1
  69. package/dist/migration/migrator.js +7 -7
  70. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  71. package/dist/migration/postgresql-schema-reader.js +5 -3
  72. package/dist/naite/naite.d.ts +0 -4
  73. package/dist/naite/naite.d.ts.map +1 -1
  74. package/dist/naite/naite.js +11 -19
  75. package/dist/ssr/index.d.ts +4 -0
  76. package/dist/ssr/index.d.ts.map +1 -0
  77. package/dist/ssr/index.js +4 -0
  78. package/dist/ssr/registry.d.ts +10 -0
  79. package/dist/ssr/registry.d.ts.map +1 -0
  80. package/dist/ssr/registry.js +43 -0
  81. package/dist/ssr/renderer.d.ts +6 -0
  82. package/dist/ssr/renderer.d.ts.map +1 -0
  83. package/dist/ssr/renderer.js +70 -0
  84. package/dist/ssr/types.d.ts +19 -0
  85. package/dist/ssr/types.d.ts.map +1 -0
  86. package/dist/ssr/types.js +4 -0
  87. package/dist/syncer/syncer.d.ts +1 -0
  88. package/dist/syncer/syncer.d.ts.map +1 -1
  89. package/dist/syncer/syncer.js +58 -1
  90. package/dist/tasks/decorator.d.ts +1 -0
  91. package/dist/tasks/decorator.d.ts.map +1 -1
  92. package/dist/tasks/decorator.js +9 -7
  93. package/dist/tasks/step-wrapper.d.ts +5 -0
  94. package/dist/tasks/step-wrapper.d.ts.map +1 -1
  95. package/dist/tasks/step-wrapper.js +11 -6
  96. package/dist/tasks/workflow-manager.d.ts +2 -0
  97. package/dist/tasks/workflow-manager.d.ts.map +1 -1
  98. package/dist/tasks/workflow-manager.js +5 -2
  99. package/dist/template/implementations/entry-server.template.d.ts +17 -0
  100. package/dist/template/implementations/entry-server.template.d.ts.map +1 -0
  101. package/dist/template/implementations/entry-server.template.js +78 -0
  102. package/dist/template/implementations/model.template.d.ts.map +1 -1
  103. package/dist/template/implementations/model.template.js +5 -3
  104. package/dist/template/implementations/queries.template.d.ts +17 -0
  105. package/dist/template/implementations/queries.template.d.ts.map +1 -0
  106. package/dist/template/implementations/queries.template.js +83 -0
  107. package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -1
  108. package/dist/template/implementations/view_enums_select.template.js +34 -20
  109. package/dist/template/implementations/view_form.template.d.ts +2 -1
  110. package/dist/template/implementations/view_form.template.d.ts.map +1 -1
  111. package/dist/template/implementations/view_form.template.js +301 -129
  112. package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -1
  113. package/dist/template/implementations/view_id_async_select.template.js +136 -57
  114. package/dist/template/implementations/view_list.template.d.ts +2 -0
  115. package/dist/template/implementations/view_list.template.d.ts.map +1 -1
  116. package/dist/template/implementations/view_list.template.js +392 -227
  117. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
  118. package/dist/template/implementations/view_search_input.template.js +46 -30
  119. package/dist/template/zod-converter.d.ts.map +1 -1
  120. package/dist/template/zod-converter.js +2 -2
  121. package/dist/testing/bootstrap.d.ts +28 -0
  122. package/dist/testing/bootstrap.d.ts.map +1 -0
  123. package/dist/testing/bootstrap.js +120 -0
  124. package/dist/testing/fixture-loader.d.ts +21 -0
  125. package/dist/testing/fixture-loader.d.ts.map +1 -0
  126. package/dist/testing/fixture-loader.js +28 -0
  127. package/dist/testing/fixture-manager.d.ts +1 -1
  128. package/dist/testing/fixture-manager.d.ts.map +1 -1
  129. package/dist/testing/fixture-manager.js +7 -7
  130. package/dist/testing/index.d.ts +4 -0
  131. package/dist/testing/index.d.ts.map +1 -0
  132. package/dist/testing/index.js +5 -0
  133. package/dist/testing/naite-vitest-reporter.d.ts +12 -0
  134. package/dist/testing/naite-vitest-reporter.d.ts.map +1 -0
  135. package/dist/testing/naite-vitest-reporter.js +17 -0
  136. package/dist/types/types.d.ts +5 -6
  137. package/dist/types/types.d.ts.map +1 -1
  138. package/dist/types/types.js +7 -8
  139. package/dist/ui/ai-client.d.ts +3 -1
  140. package/dist/ui/ai-client.d.ts.map +1 -1
  141. package/dist/ui/ai-client.js +27 -8
  142. package/dist/ui-web/assets/index-CTYv3qL6.js +92 -0
  143. package/dist/ui-web/index.html +1 -1
  144. package/package.json +43 -20
  145. package/src/ai/agents/agent.ts +38 -19
  146. package/src/api/base-frame.ts +8 -0
  147. package/src/api/caster.ts +6 -1
  148. package/src/api/config.ts +38 -4
  149. package/src/api/decorators.ts +106 -20
  150. package/src/api/index.ts +1 -0
  151. package/src/api/secret.ts +23 -0
  152. package/src/api/sonamu.ts +334 -61
  153. package/src/cache/cache-manager.ts +23 -0
  154. package/src/cache/decorator.ts +116 -0
  155. package/src/cache/drivers.ts +42 -0
  156. package/src/cache/index.ts +16 -0
  157. package/src/cache/types.ts +32 -0
  158. package/src/database/base-model.ts +7 -3
  159. package/src/database/code-generator.ts +3 -1
  160. package/src/database/db.ts +5 -5
  161. package/src/database/knex.ts +34 -0
  162. package/src/database/puri.types.ts +2 -3
  163. package/src/database/upsert-builder.ts +58 -4
  164. package/src/index.ts +4 -0
  165. package/src/logger/category.ts +42 -0
  166. package/src/logger/configure.ts +132 -0
  167. package/src/migration/code-generation.ts +19 -6
  168. package/src/migration/migrator.ts +7 -6
  169. package/src/migration/postgresql-schema-reader.ts +7 -2
  170. package/src/naite/naite.ts +10 -18
  171. package/src/shared/web.shared.ts.txt +1 -1
  172. package/src/ssr/index.ts +13 -0
  173. package/src/ssr/registry.ts +52 -0
  174. package/src/ssr/renderer.ts +105 -0
  175. package/src/ssr/types.ts +20 -0
  176. package/src/syncer/syncer.ts +59 -0
  177. package/src/tasks/decorator.ts +20 -4
  178. package/src/tasks/step-wrapper.ts +14 -5
  179. package/src/tasks/workflow-manager.ts +9 -1
  180. package/src/template/implementations/entry-server.template.ts +81 -0
  181. package/src/template/implementations/model.template.ts +4 -2
  182. package/src/template/implementations/queries.template.ts +111 -0
  183. package/src/template/implementations/view_enums_select.template.ts +33 -19
  184. package/src/template/implementations/view_form.template.ts +324 -145
  185. package/src/template/implementations/view_id_async_select.template.ts +145 -56
  186. package/src/template/implementations/view_list.template.ts +446 -236
  187. package/src/template/implementations/view_search_input.template.ts +45 -29
  188. package/src/template/zod-converter.ts +4 -1
  189. package/src/testing/bootstrap.ts +176 -0
  190. package/src/testing/fixture-loader.ts +28 -0
  191. package/src/testing/fixture-manager.ts +7 -6
  192. package/src/testing/index.ts +3 -0
  193. package/src/testing/naite-vitest-reporter.ts +18 -0
  194. package/src/types/types.ts +4 -5
  195. package/src/ui/ai-client.ts +82 -50
  196. package/dist/template/implementations/view_enums_dropdown.template.d.ts +0 -17
  197. package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +0 -1
  198. package/dist/template/implementations/view_enums_dropdown.template.js +0 -50
  199. package/dist/ui-web/assets/index-B87IyofX.js +0 -92
  200. package/src/template/implementations/view_enums_dropdown.template.ts +0 -53
@@ -18,16 +18,41 @@ export class Template__view_id_async_select extends Template {
18
18
  const names = EntityManager.getNamesFromId(entityId);
19
19
 
20
20
  const entity = EntityManager.get(entityId);
21
+
22
+ // textField가 지정되지 않은 경우 모든 subset에 공통으로 있는 필드 찾기
21
23
  if (!textField) {
22
- const pickedProp = entity.props.find((prop) => ["name", "title"].includes(prop.name));
23
- if (pickedProp) {
24
- textField = pickedProp.name;
25
- } else {
26
- const candidateProp = entity.props.find((prop) => prop.type === "string");
27
- if (candidateProp) {
28
- textField = candidateProp.name;
24
+ const subsetKeys = Object.keys(entity.subsets);
25
+ if (subsetKeys.length > 0) {
26
+ // 모든 subset에 공통으로 포함된 직접 필드만 추출
27
+ const commonFields = subsetKeys.reduce(
28
+ (common, key) => {
29
+ const fields = entity.subsets[key]
30
+ .filter((path) => !path.includes(".")) // 직접 필드만
31
+ .map((path) => path);
32
+ return common.filter((field) => fields.includes(field));
33
+ },
34
+ entity.subsets[subsetKeys[0]].filter((path) => !path.includes(".")),
35
+ );
36
+
37
+ // 우선순위: name > title > 첫 번째 string 타입
38
+ const pickedProp = entity.props.find(
39
+ (prop) => ["name", "title"].includes(prop.name) && commonFields.includes(prop.name),
40
+ );
41
+ if (pickedProp) {
42
+ textField = pickedProp.name;
29
43
  } else {
30
- console.log("textField 찾을 없음");
44
+ const candidateProp = entity.props.find(
45
+ (prop) => prop.type === "string" && commonFields.includes(prop.name),
46
+ );
47
+ if (candidateProp) {
48
+ textField = candidateProp.name;
49
+ } else {
50
+ // 공통 필드가 없으면 id 사용
51
+ textField = "id";
52
+ console.log(
53
+ `Warning: ${entityId}에 모든 subset에 공통으로 포함된 string 필드가 없어 id를 사용합니다`,
54
+ );
55
+ }
31
56
  }
32
57
  }
33
58
  }
@@ -35,69 +60,133 @@ export class Template__view_id_async_select extends Template {
35
60
  return {
36
61
  ...this.getTargetAndPath(names),
37
62
  body: `
38
- import React, { useState, useEffect, SyntheticEvent } from "react";
39
- import { DropdownProps, DropdownItemProps, DropdownOnSearchChangeData, Dropdown } from "semantic-ui-react";
40
- import { ${names.capital}SubsetKey, ${
41
- names.capital
42
- }SubsetMapping } from "@/services/sonamu.generated";
63
+ import React, { useState, useEffect, useMemo, useCallback } from "react";
64
+ import { AsyncSelect, AsyncSelectOption, MultiSelect, MultiSelectOption } from "@sonamu-kit/react-components/components";
65
+ import { ${names.capital}SubsetKey, ${names.capital}SubsetMapping } from "@/services/sonamu.generated";
43
66
  import { ${names.capital}Service } from "@/services/services.generated";
44
67
  import { ${names.capital}ListParams } from "@/services/${names.fs}/${names.fs}.types";
45
68
 
46
- export function ${names.capital}IdAsyncSelect<T extends ${names.capital}SubsetKey>(
47
- { subset, baseListParams, textField, valueField, ...props }: DropdownProps & {
48
- subset: T;
49
- baseListParams?: ${names.capital}ListParams;
50
- textField${textField ? "?" : ""}: keyof ${names.capital}SubsetMapping[T];
51
- valueField?: keyof ${names.capital}SubsetMapping[T];
52
- },
53
- ) {
54
- const [options, setOptions] = useState<DropdownItemProps[]>([]);
69
+ export type ${names.capital}IdAsyncSelectProps<T extends ${names.capital}SubsetKey> = {
70
+ subset: T;
71
+ baseListParams?: ${names.capital}ListParams;
72
+ textField${textField ? "?" : ""}: keyof ${names.capital}SubsetMapping[T];
73
+ valueField?: keyof ${names.capital}SubsetMapping[T];
74
+ placeholder?: string;
75
+ clearable?: boolean;
76
+ disabled?: boolean;
77
+ className?: string;
78
+ multiple?: boolean;
79
+ } & (
80
+ | {
81
+ multiple?: false;
82
+ value?: number | null;
83
+ onChange?: (e: React.SyntheticEvent | null, data: { value: number | undefined }) => void;
84
+ }
85
+ | {
86
+ multiple: true;
87
+ value?: number[];
88
+ onChange?: (e: React.SyntheticEvent | null, data: { value: number[] }) => void;
89
+ }
90
+ );
91
+
92
+ export function ${names.capital}IdAsyncSelect<T extends ${names.capital}SubsetKey>({
93
+ subset,
94
+ value,
95
+ onChange,
96
+ baseListParams,
97
+ textField,
98
+ valueField,
99
+ placeholder = "${entity.title ?? names.constant}",
100
+ clearable,
101
+ disabled,
102
+ className,
103
+ multiple = false,
104
+ }: ${names.capital}IdAsyncSelectProps<T>) {
55
105
  const [listParams, setListParams] = useState<${names.capital}ListParams>(
56
- baseListParams ?? {},
106
+ baseListParams ?? {}
57
107
  );
58
108
 
59
- const { data, error } = ${names.capital}Service.use${names.capitalPlural}(subset, listParams);
60
- const { rows: ${names.camelPlural}, total } = data ?? {};
109
+ const { data, isLoading } = ${names.capital}Service.use${names.capitalPlural}(subset, listParams);
110
+ const { rows: ${names.camelPlural} } = data ?? {};
61
111
 
62
- useEffect(() => {
63
- setOptions(
64
- (${names.camelPlural} ?? []).map((${names.camel}) => {
65
- return {
66
- key: ${names.camel}.id,
67
- value: ${names.camel}[valueField ?? 'id'] as string | number,
68
- text: String(${names.camel}[textField${textField ? ` ?? '${textField}'` : ""}]),
69
- };
70
- }),
71
- );
72
- }, [${names.camelPlural}]);
112
+ // 옵션 생성
113
+ const options = useMemo(() => {
114
+ return (${names.camelPlural} ?? []).map((${names.camel}) => {
115
+ // textField가 지정되지 않은 경우 기본값 사용
116
+ const label = (() => {
117
+ if (textField) {
118
+ return String(${names.camel}[textField]);
119
+ }
120
+ return String(${names.camel}.${textField || "id"});
121
+ })();
73
122
 
123
+ return {
124
+ value: String(${names.camel}[valueField ?? "id"] as number),
125
+ label,
126
+ };
127
+ });
128
+ }, [${names.camelPlural}, textField, valueField]);
129
+
130
+ // baseListParams 변경 시 반영
74
131
  useEffect(() => {
75
- setListParams({
76
- ...listParams,
132
+ setListParams((prev) => ({
133
+ ...prev,
77
134
  ...baseListParams,
78
- });
135
+ }));
79
136
  }, [baseListParams]);
80
137
 
81
- const handleSearchChange = (
82
- e: SyntheticEvent<HTMLElement, Event>,
83
- data: DropdownOnSearchChangeData,
84
- ) => {
85
- setListParams({
86
- ...listParams,
87
- keyword: data.searchQuery,
88
- });
138
+ // 검색어 변경 핸들러
139
+ const handleSearch = useCallback((keyword: string) => {
140
+ setListParams((prev) => ({
141
+ ...prev,
142
+ keyword: keyword || undefined,
143
+ }));
144
+ }, []);
145
+
146
+ // Multiple select
147
+ if (multiple) {
148
+ const multiValue = Array.isArray(value) ? value.map(String) : [];
149
+
150
+ const handleMultiChange = (selectedValues: string[]) => {
151
+ if (onChange) {
152
+ const numericValues = selectedValues.map(Number);
153
+ (onChange as (e: React.SyntheticEvent | null, data: { value: number[] }) => void)(null, { value: numericValues });
154
+ }
155
+ };
156
+
157
+ return (
158
+ <MultiSelect
159
+ options={options}
160
+ onValueChange={handleMultiChange}
161
+ defaultValue={multiValue}
162
+ placeholder={placeholder}
163
+ disabled={disabled}
164
+ className={className}
165
+ />
166
+ );
167
+ }
168
+
169
+ // Single select
170
+ const singleValue = typeof value === "number" ? value : undefined;
171
+
172
+ const handleSingleChange = (e: React.SyntheticEvent | null, data: { value: string | undefined }) => {
173
+ if (onChange) {
174
+ const numericValue = data.value ? Number(data.value) : undefined;
175
+ (onChange as (e: React.SyntheticEvent | null, data: { value: number | undefined }) => void)(e, { value: numericValue });
176
+ }
89
177
  };
90
178
 
91
179
  return (
92
- <Dropdown
93
- placeholder="${entity.title ?? names.constant}"
94
- selection
95
- options={options}
96
- onSearchChange={handleSearchChange}
97
- disabled={!${names.camelPlural}}
98
- loading={!${names.camelPlural}}
99
- selectOnBlur={false}
100
- {...props}
180
+ <AsyncSelect
181
+ options={options as AsyncSelectOption<string>[]}
182
+ value={singleValue !== undefined ? String(singleValue) : undefined}
183
+ onChange={handleSingleChange}
184
+ isLoading={isLoading}
185
+ placeholder={placeholder}
186
+ clearable={clearable}
187
+ disabled={disabled}
188
+ className={className}
189
+ onSearch={handleSearch}
101
190
  />
102
191
  );
103
192
  }