@lunora/studio 0.0.0 → 1.0.0-alpha.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 (81) hide show
  1. package/LICENSE.md +105 -0
  2. package/README.md +123 -9
  3. package/__assets__/package-og.svg +14 -0
  4. package/dist/index.d.ts +1402 -0
  5. package/dist/index.js +41 -0
  6. package/dist/mount.d.ts +21 -0
  7. package/dist/mount.js +26 -0
  8. package/dist/packem_shared/ADMIN_FUNCTION_PREFIX-DmBqMZ-z.js +45 -0
  9. package/dist/packem_shared/ApiDocsPanel-DpRjJhG5.js +842 -0
  10. package/dist/packem_shared/ApiReferencePanel-DMIUp-kK.js +229 -0
  11. package/dist/packem_shared/ApiTab-DURGU15e.js +251 -0
  12. package/dist/packem_shared/AuditPanel-BC59Nhst.js +212 -0
  13. package/dist/packem_shared/CommandPalette-Dx_CoB9i.js +373 -0
  14. package/dist/packem_shared/ConfirmButton-WQVUoGFb.js +59 -0
  15. package/dist/packem_shared/ConnectionBadge-Bxagrip8.js +111 -0
  16. package/dist/packem_shared/DEFAULT_AUTO_REFRESH_MS-Vxwaxx51.js +50 -0
  17. package/dist/packem_shared/DEFAULT_INSIGHT_THRESHOLDS-DjF0h-gA.js +89 -0
  18. package/dist/packem_shared/DataBrowser-Coz6jJE6.js +4542 -0
  19. package/dist/packem_shared/DataFilters-FNquMaiu.js +249 -0
  20. package/dist/packem_shared/ErrorBoundary-BzAApI7J.js +66 -0
  21. package/dist/packem_shared/ExportImportPanel-WO34fJxy.js +193 -0
  22. package/dist/packem_shared/FileBrowser-Zcr-Qgxo.js +2932 -0
  23. package/dist/packem_shared/FunctionRunner-j0Rd5m9t.js +343 -0
  24. package/dist/packem_shared/FunctionStatsPanel-DboBl-XL.js +432 -0
  25. package/dist/packem_shared/GlobalDataBrowser-9MhPEfgN.js +318 -0
  26. package/dist/packem_shared/HealthPanel-DOIgbUtx.js +640 -0
  27. package/dist/packem_shared/HomePanel-bdOCNA-p.js +1273 -0
  28. package/dist/packem_shared/InsightsPanel-DaZPnSgt.js +423 -0
  29. package/dist/packem_shared/LogsPanel-CWdqAGpQ.js +839 -0
  30. package/dist/packem_shared/MailPanel-D_EGtDnS.js +447 -0
  31. package/dist/packem_shared/MetricsPanel-E4Gv6wTO.js +1625 -0
  32. package/dist/packem_shared/MigrationsPanel-DQdPY9io.js +246 -0
  33. package/dist/packem_shared/OpenRpcReferencePanel-j2p3HB0s.js +191 -0
  34. package/dist/packem_shared/PitrPanel-BbBkQR6t.js +252 -0
  35. package/dist/packem_shared/STUDIO_ROOT_CLASS-D12gX2dV.js +3 -0
  36. package/dist/packem_shared/ScheduledJobs-Ok1CYYwI.js +159 -0
  37. package/dist/packem_shared/SchemaViewer-D8XGnp-X.js +2512 -0
  38. package/dist/packem_shared/SecurityAdvisorPanel-Cdm2IxLW.js +79 -0
  39. package/dist/packem_shared/SettingsPanel-D3WF2mBU.js +176 -0
  40. package/dist/packem_shared/ShardInput-DNCsT1KW.js +107 -0
  41. package/dist/packem_shared/SqlEditorPanel-BuQ7f2Hs.js +13 -0
  42. package/dist/packem_shared/Studio-D36od9Oz.js +33 -0
  43. package/dist/packem_shared/StudioApp-dvywkJ8I.js +383 -0
  44. package/dist/packem_shared/StudioI18nProvider-Dcajsznk.js +48 -0
  45. package/dist/packem_shared/TableEditor-DIVDk3vT.js +371 -0
  46. package/dist/packem_shared/advisor-view-DBlzJi6C.js +159 -0
  47. package/dist/packem_shared/aggregateMetrics-D4nUHEKU.js +108 -0
  48. package/dist/packem_shared/app.d-CCmwDEVs.d.ts +300 -0
  49. package/dist/packem_shared/badge-B2PKA1-5.js +49 -0
  50. package/dist/packem_shared/bar-chart-CzJAgqkp.js +3245 -0
  51. package/dist/packem_shared/button-BhsN2uZH.js +49 -0
  52. package/dist/packem_shared/card-DURq3ElK.js +175 -0
  53. package/dist/packem_shared/cf-links-BZfRdxSE.js +8 -0
  54. package/dist/packem_shared/checkbox-UNkzAxl-.js +63 -0
  55. package/dist/packem_shared/createStudioI18n-CgvlmDkN.js +27 -0
  56. package/dist/packem_shared/data-grid-CCh2Couo.js +183 -0
  57. package/dist/packem_shared/dropdown-menu-WY4B_eJO.js +280 -0
  58. package/dist/packem_shared/empty-state-DY_oe0k6.js +98 -0
  59. package/dist/packem_shared/grid-features-DTjG6Sex.js +840 -0
  60. package/dist/packem_shared/input-XH4r1Pt1.js +53 -0
  61. package/dist/packem_shared/internal-BBZYexre.js +68 -0
  62. package/dist/packem_shared/label-D8ykjn5J.js +46 -0
  63. package/dist/packem_shared/live-status-bPff1O7Y.js +44 -0
  64. package/dist/packem_shared/reference-view-BCKIoai7.js +2180 -0
  65. package/dist/packem_shared/shard-history-DyebH1R5.js +38 -0
  66. package/dist/packem_shared/sparkline-10dG-_f0.js +93 -0
  67. package/dist/packem_shared/sql-editor-panel-CW2y2x9h.js +2562 -0
  68. package/dist/packem_shared/storage-tier-CL98eOvn.js +85 -0
  69. package/dist/packem_shared/studio-BDVd7rIV.js +10303 -0
  70. package/dist/packem_shared/table-_RzNvy3R.js +246 -0
  71. package/dist/packem_shared/table-list-sidebar-aZHLq70w.js +832 -0
  72. package/dist/packem_shared/textarea-D3gaCU_-.js +46 -0
  73. package/dist/packem_shared/use-live-admin-D1h1Fzsd.js +73 -0
  74. package/dist/packem_shared/use-live-shard-seed-B74RYcOy.js +76 -0
  75. package/dist/packem_shared/useDebounced-Dxncpg6z.js +32 -0
  76. package/dist/packem_shared/utils-B05Dmz_H.js +8 -0
  77. package/dist/packem_shared/virtual-rect-CVMUskSm.js +10 -0
  78. package/dist/standalone/studio.js +356 -0
  79. package/dist/styles.css +2 -0
  80. package/package.json +77 -17
  81. package/src/theme.css +59 -0
@@ -0,0 +1,249 @@
1
+ import { c } from 'react/compiler-runtime';
2
+ import { jsxDEV } from 'react/jsx-dev-runtime';
3
+
4
+ const OPERATORS = [{
5
+ label: "=",
6
+ value: "eq"
7
+ }, {
8
+ label: "≠",
9
+ value: "ne"
10
+ }, {
11
+ label: "<",
12
+ value: "lt"
13
+ }, {
14
+ label: "≤",
15
+ value: "lte"
16
+ }, {
17
+ label: ">",
18
+ value: "gt"
19
+ }, {
20
+ label: "≥",
21
+ value: "gte"
22
+ }, {
23
+ label: "contains",
24
+ value: "contains"
25
+ }];
26
+ const toFilterClauses = (filters) => filters.filter((filter) => filter.column !== "").map((filter) => {
27
+ const numeric = filter.operator !== "contains" && filter.value.trim() !== "" && !Number.isNaN(Number(filter.value));
28
+ return {
29
+ column: filter.column,
30
+ operator: filter.operator,
31
+ value: numeric ? Number(filter.value) : filter.value
32
+ };
33
+ });
34
+ const DataFilters = (t0) => {
35
+ const $ = c(40);
36
+ const {
37
+ columns,
38
+ filters,
39
+ onFiltersChange,
40
+ onSearchChange,
41
+ search
42
+ } = t0;
43
+ let t1;
44
+ if ($[0] !== columns[0] || $[1] !== filters || $[2] !== onFiltersChange) {
45
+ t1 = () => {
46
+ onFiltersChange([...filters, {
47
+ column: columns[0] ?? "",
48
+ operator: "eq",
49
+ value: ""
50
+ }]);
51
+ };
52
+ $[0] = columns[0];
53
+ $[1] = filters;
54
+ $[2] = onFiltersChange;
55
+ $[3] = t1;
56
+ } else {
57
+ t1 = $[3];
58
+ }
59
+ const addFilter = t1;
60
+ let t2;
61
+ if ($[4] !== filters || $[5] !== onFiltersChange) {
62
+ t2 = (event) => {
63
+ const index = Number(event.currentTarget.dataset.index);
64
+ onFiltersChange(filters.filter((_, position) => position !== index));
65
+ };
66
+ $[4] = filters;
67
+ $[5] = onFiltersChange;
68
+ $[6] = t2;
69
+ } else {
70
+ t2 = $[6];
71
+ }
72
+ const removeFilter = t2;
73
+ let t3;
74
+ if ($[7] !== filters || $[8] !== onFiltersChange) {
75
+ t3 = (event_0) => {
76
+ const index_0 = Number(event_0.currentTarget.dataset.index);
77
+ const next = event_0.currentTarget.value;
78
+ onFiltersChange(filters.map((filter, position_0) => position_0 === index_0 ? {
79
+ ...filter,
80
+ column: next
81
+ } : filter));
82
+ };
83
+ $[7] = filters;
84
+ $[8] = onFiltersChange;
85
+ $[9] = t3;
86
+ } else {
87
+ t3 = $[9];
88
+ }
89
+ const changeColumn = t3;
90
+ let t4;
91
+ if ($[10] !== filters || $[11] !== onFiltersChange) {
92
+ t4 = (event_1) => {
93
+ const index_1 = Number(event_1.currentTarget.dataset.index);
94
+ const next_0 = event_1.currentTarget.value;
95
+ onFiltersChange(filters.map((filter_0, position_1) => position_1 === index_1 ? {
96
+ ...filter_0,
97
+ operator: next_0
98
+ } : filter_0));
99
+ };
100
+ $[10] = filters;
101
+ $[11] = onFiltersChange;
102
+ $[12] = t4;
103
+ } else {
104
+ t4 = $[12];
105
+ }
106
+ const changeOperator = t4;
107
+ let t5;
108
+ if ($[13] !== filters || $[14] !== onFiltersChange) {
109
+ t5 = (event_2) => {
110
+ const index_2 = Number(event_2.currentTarget.dataset.index);
111
+ const next_1 = event_2.currentTarget.value;
112
+ onFiltersChange(filters.map((filter_1, position_2) => position_2 === index_2 ? {
113
+ ...filter_1,
114
+ value: next_1
115
+ } : filter_1));
116
+ };
117
+ $[13] = filters;
118
+ $[14] = onFiltersChange;
119
+ $[15] = t5;
120
+ } else {
121
+ t5 = $[15];
122
+ }
123
+ const changeValue = t5;
124
+ let t6;
125
+ if ($[16] !== onSearchChange || $[17] !== search) {
126
+ t6 = /* @__PURE__ */ jsxDEV("input", {
127
+ "aria-label": "Search rows",
128
+ "data-testid": "db-filter",
129
+ onChange: onSearchChange,
130
+ placeholder: "search table…",
131
+ value: search
132
+ }, void 0, false);
133
+ $[16] = onSearchChange;
134
+ $[17] = search;
135
+ $[18] = t6;
136
+ } else {
137
+ t6 = $[18];
138
+ }
139
+ let t7;
140
+ if ($[19] !== addFilter) {
141
+ t7 = /* @__PURE__ */ jsxDEV("button", {
142
+ "data-testid": "db-add-filter",
143
+ onClick: addFilter,
144
+ type: "button",
145
+ children: "Add filter"
146
+ }, void 0, false);
147
+ $[19] = addFilter;
148
+ $[20] = t7;
149
+ } else {
150
+ t7 = $[20];
151
+ }
152
+ let t8;
153
+ if ($[21] !== t6 || $[22] !== t7) {
154
+ t8 = /* @__PURE__ */ jsxDEV("div", {
155
+ className: "flex flex-wrap items-center gap-1.5",
156
+ children: [t6, t7]
157
+ }, void 0, true);
158
+ $[21] = t6;
159
+ $[22] = t7;
160
+ $[23] = t8;
161
+ } else {
162
+ t8 = $[23];
163
+ }
164
+ let t9;
165
+ if ($[24] !== changeColumn || $[25] !== changeOperator || $[26] !== changeValue || $[27] !== columns || $[28] !== filters || $[29] !== removeFilter) {
166
+ let t102;
167
+ if ($[31] !== changeColumn || $[32] !== changeOperator || $[33] !== changeValue || $[34] !== columns || $[35] !== removeFilter) {
168
+ t102 = (filter_2, index_3) => /* @__PURE__ */ jsxDEV("div", {
169
+ className: "flex flex-wrap items-center gap-1.5",
170
+ "data-testid": "db-filter-row",
171
+ children: [/* @__PURE__ */ jsxDEV("select", {
172
+ "aria-label": "Filter column",
173
+ "data-index": index_3,
174
+ "data-testid": "db-filter-column",
175
+ onChange: changeColumn,
176
+ value: filter_2.column,
177
+ children: columns.map(_temp)
178
+ }, void 0, false), /* @__PURE__ */ jsxDEV("select", {
179
+ "aria-label": "Filter operator",
180
+ "data-index": index_3,
181
+ "data-testid": "db-filter-operator",
182
+ onChange: changeOperator,
183
+ value: filter_2.operator,
184
+ children: OPERATORS.map(_temp2)
185
+ }, void 0, false), /* @__PURE__ */ jsxDEV("input", {
186
+ "aria-label": "Filter value",
187
+ "data-index": index_3,
188
+ "data-testid": "db-filter-value",
189
+ onChange: changeValue,
190
+ placeholder: "value",
191
+ value: filter_2.value
192
+ }, void 0, false), /* @__PURE__ */ jsxDEV("button", {
193
+ "aria-label": "Remove filter",
194
+ "data-index": index_3,
195
+ "data-testid": "db-filter-remove",
196
+ onClick: removeFilter,
197
+ type: "button",
198
+ children: "✕"
199
+ }, void 0, false)]
200
+ }, index_3, true);
201
+ $[31] = changeColumn;
202
+ $[32] = changeOperator;
203
+ $[33] = changeValue;
204
+ $[34] = columns;
205
+ $[35] = removeFilter;
206
+ $[36] = t102;
207
+ } else {
208
+ t102 = $[36];
209
+ }
210
+ t9 = filters.map(t102);
211
+ $[24] = changeColumn;
212
+ $[25] = changeOperator;
213
+ $[26] = changeValue;
214
+ $[27] = columns;
215
+ $[28] = filters;
216
+ $[29] = removeFilter;
217
+ $[30] = t9;
218
+ } else {
219
+ t9 = $[30];
220
+ }
221
+ let t10;
222
+ if ($[37] !== t8 || $[38] !== t9) {
223
+ t10 = /* @__PURE__ */ jsxDEV("div", {
224
+ className: "flex flex-col gap-1.5",
225
+ "data-testid": "db-filters",
226
+ children: [t8, t9]
227
+ }, void 0, true);
228
+ $[37] = t8;
229
+ $[38] = t9;
230
+ $[39] = t10;
231
+ } else {
232
+ t10 = $[39];
233
+ }
234
+ return t10;
235
+ };
236
+ function _temp(column) {
237
+ return /* @__PURE__ */ jsxDEV("option", {
238
+ value: column,
239
+ children: column
240
+ }, column, false);
241
+ }
242
+ function _temp2(operator) {
243
+ return /* @__PURE__ */ jsxDEV("option", {
244
+ value: operator.value,
245
+ children: operator.label
246
+ }, operator.value, false);
247
+ }
248
+
249
+ export { DataFilters, toFilterClauses };
@@ -0,0 +1,66 @@
1
+ import { Component } from 'react';
2
+ import { jsxDEV, Fragment } from 'react/jsx-dev-runtime';
3
+
4
+ const CONTAINER_STYLE = {
5
+ border: "1px solid var(--destructive)",
6
+ color: "var(--destructive)",
7
+ padding: 12
8
+ };
9
+ const MESSAGE_STYLE = {
10
+ overflow: "auto",
11
+ whiteSpace: "pre-wrap"
12
+ };
13
+ class ErrorBoundary extends Component {
14
+ static getDerivedStateFromError(error) {
15
+ return {
16
+ error
17
+ };
18
+ }
19
+ state = {
20
+ error: null
21
+ };
22
+ componentDidCatch(error, info) {
23
+ console.error("[lunora-studio] panel error", error, info.componentStack);
24
+ }
25
+ reset = () => {
26
+ this.setState({
27
+ error: null
28
+ });
29
+ };
30
+ render() {
31
+ const {
32
+ children,
33
+ fallbackTitle,
34
+ label,
35
+ retryLabel
36
+ } = this.props;
37
+ const {
38
+ error
39
+ } = this.state;
40
+ if (error === null) {
41
+ return /* @__PURE__ */ jsxDEV(Fragment, {
42
+ children
43
+ }, void 0, false);
44
+ }
45
+ const title = fallbackTitle ?? (label === void 0 ? "Something went wrong" : `${label} failed`);
46
+ return /* @__PURE__ */ jsxDEV("div", {
47
+ "data-testid": "dash-error-boundary",
48
+ role: "alert",
49
+ style: CONTAINER_STYLE,
50
+ children: [/* @__PURE__ */ jsxDEV("strong", {
51
+ children: title
52
+ }, void 0, false), /* @__PURE__ */ jsxDEV("pre", {
53
+ "data-testid": "dash-error-message",
54
+ style: MESSAGE_STYLE,
55
+ children: error.message
56
+ }, void 0, false), /* @__PURE__ */ jsxDEV("button", {
57
+ "data-testid": "dash-error-retry",
58
+ onClick: this.reset,
59
+ type: "button",
60
+ children: retryLabel ?? "Try again"
61
+ }, void 0, false)]
62
+ }, void 0, true);
63
+ }
64
+ }
65
+
66
+ export { ErrorBoundary };
@@ -0,0 +1,193 @@
1
+ import { useLunora } from '@lunora/react';
2
+ import { useState } from 'react';
3
+ import { ConfirmButton } from './ConfirmButton-WQVUoGFb.js';
4
+ import { ShardInput } from './ShardInput-DNCsT1KW.js';
5
+ import { B as Button } from './button-BhsN2uZH.js';
6
+ import { C as Card, a as CardContent } from './card-DURq3ElK.js';
7
+ import { T as Textarea } from './textarea-D3gaCU_-.js';
8
+ import { useT } from './createStudioI18n-CgvlmDkN.js';
9
+ import { ADMIN_FUNCTIONS } from './ADMIN_FUNCTION_PREFIX-DmBqMZ-z.js';
10
+ import { f as fireAndForget, c as callOptions, e as errorMessage, a as adminRef } from './internal-BBZYexre.js';
11
+ import { r as recordShard } from './shard-history-DyebH1R5.js';
12
+ import { jsxDEV } from 'react/jsx-dev-runtime';
13
+
14
+ const EXPORT_SHARD = adminRef(ADMIN_FUNCTIONS.exportShard);
15
+ const IMPORT_SHARD = adminRef(ADMIN_FUNCTIONS.importShard);
16
+ const toNdjson = (rows) => rows.map((row) => JSON.stringify(row)).join("\n");
17
+ const parseNdjson = (text) => {
18
+ const rows = [];
19
+ const lines = text.split("\n");
20
+ for (const [index, rawLine] of lines.entries()) {
21
+ const line = rawLine.trim();
22
+ if (line === "") {
23
+ continue;
24
+ }
25
+ const parsed = JSON.parse(line);
26
+ if (typeof parsed !== "object" || parsed === null) {
27
+ throw new Error(`line ${(index + 1).toString()}: expected a { table, doc } object`);
28
+ }
29
+ const row = parsed;
30
+ if (typeof row.table !== "string" || typeof row.doc !== "object" || row.doc === null || Array.isArray(row.doc)) {
31
+ throw new Error(`line ${(index + 1).toString()}: expected a { table, doc } object`);
32
+ }
33
+ rows.push(row);
34
+ }
35
+ return rows;
36
+ };
37
+ const ExportImportPanel = ({
38
+ initialShardKey
39
+ }) => {
40
+ const client = useLunora();
41
+ const t = useT();
42
+ const [shardKey, setShardKey] = useState(initialShardKey ?? "");
43
+ const [ndjson, setNdjson] = useState("");
44
+ const [busy, setBusy] = useState(false);
45
+ const [error, setError] = useState(null);
46
+ const [importResult, setImportResult] = useState(null);
47
+ const [exportCount, setExportCount] = useState(null);
48
+ const exportShard = async () => {
49
+ setBusy(true);
50
+ setError(null);
51
+ setImportResult(null);
52
+ try {
53
+ const result = await client.query(EXPORT_SHARD, {}, callOptions(shardKey));
54
+ recordShard(shardKey);
55
+ setNdjson(toNdjson(result.rows));
56
+ setExportCount(result.rows.length);
57
+ } catch (error_) {
58
+ setError(errorMessage(error_));
59
+ } finally {
60
+ setBusy(false);
61
+ }
62
+ };
63
+ const importShard = async () => {
64
+ setBusy(true);
65
+ setError(null);
66
+ setImportResult(null);
67
+ setExportCount(null);
68
+ let rows;
69
+ try {
70
+ rows = parseNdjson(ndjson);
71
+ } catch (error__0) {
72
+ setError(t("Invalid NDJSON: {message}", {
73
+ message: errorMessage(error__0)
74
+ }));
75
+ setBusy(false);
76
+ return;
77
+ }
78
+ try {
79
+ const result_0 = await client.query(IMPORT_SHARD, {
80
+ rows
81
+ }, callOptions(shardKey));
82
+ recordShard(shardKey);
83
+ setImportResult(result_0);
84
+ } catch (error__1) {
85
+ setError(errorMessage(error__1));
86
+ } finally {
87
+ setBusy(false);
88
+ }
89
+ };
90
+ const insertedTotal = importResult === null ? 0 : Object.values(importResult.inserted).reduce((sum, count) => sum + count, 0);
91
+ const runExport = () => {
92
+ fireAndForget(exportShard());
93
+ };
94
+ const runImport = () => {
95
+ fireAndForget(importShard());
96
+ };
97
+ const onNdjsonChange = (event) => {
98
+ setNdjson(event.target.value);
99
+ };
100
+ return /* @__PURE__ */ jsxDEV("div", {
101
+ className: "flex flex-col gap-4",
102
+ "data-testid": "lunora-export-import",
103
+ children: [/* @__PURE__ */ jsxDEV(Card, {
104
+ className: "rounded-xl border border-border bg-card shadow-xs",
105
+ children: [/* @__PURE__ */ jsxDEV("header", {
106
+ className: "border-b border-border px-4 py-3",
107
+ children: /* @__PURE__ */ jsxDEV("span", {
108
+ className: "font-mono text-[11px] tracking-wide text-muted-foreground uppercase",
109
+ children: t("Export")
110
+ }, void 0, false)
111
+ }, void 0, false), /* @__PURE__ */ jsxDEV(CardContent, {
112
+ className: "flex flex-col gap-3 p-4",
113
+ children: [/* @__PURE__ */ jsxDEV("div", {
114
+ className: "flex flex-wrap items-center gap-2",
115
+ children: [/* @__PURE__ */ jsxDEV(ShardInput, {
116
+ onChange: setShardKey,
117
+ testId: "ei-shard-input",
118
+ value: shardKey
119
+ }, void 0, false), /* @__PURE__ */ jsxDEV(Button, {
120
+ "data-testid": "ei-export",
121
+ disabled: busy,
122
+ onClick: runExport,
123
+ type: "button",
124
+ children: t("Export")
125
+ }, void 0, false)]
126
+ }, void 0, true), exportCount !== null && /* @__PURE__ */ jsxDEV("p", {
127
+ className: "text-sm text-muted-foreground",
128
+ "data-testid": "ei-export-result",
129
+ children: t("Exported {count} rows.", {
130
+ count: exportCount
131
+ })
132
+ }, void 0, false)]
133
+ }, void 0, true)]
134
+ }, void 0, true), /* @__PURE__ */ jsxDEV(Card, {
135
+ className: "rounded-xl border border-border bg-card shadow-xs",
136
+ children: [/* @__PURE__ */ jsxDEV("header", {
137
+ className: "border-b border-border px-4 py-3",
138
+ children: /* @__PURE__ */ jsxDEV("span", {
139
+ className: "font-mono text-[11px] tracking-wide text-muted-foreground uppercase",
140
+ children: t("Import")
141
+ }, void 0, false)
142
+ }, void 0, false), /* @__PURE__ */ jsxDEV(CardContent, {
143
+ className: "flex flex-col gap-3 p-4",
144
+ children: [/* @__PURE__ */ jsxDEV(Textarea, {
145
+ "aria-label": "NDJSON",
146
+ className: "font-mono text-xs min-h-40 rounded-md",
147
+ "data-testid": "ei-ndjson",
148
+ onChange: onNdjsonChange,
149
+ placeholder: '{"table":"messages","doc":{…}}',
150
+ value: ndjson
151
+ }, void 0, false), /* @__PURE__ */ jsxDEV("div", {
152
+ className: "flex flex-wrap items-center gap-2",
153
+ children: /* @__PURE__ */ jsxDEV(ConfirmButton, {
154
+ confirmLabel: t("Import (writes rows)?"),
155
+ disabled: busy || ndjson.trim() === "",
156
+ onConfirm: runImport,
157
+ testId: "ei-import",
158
+ children: t("Import")
159
+ }, void 0, false)
160
+ }, void 0, false), importResult !== null && /* @__PURE__ */ jsxDEV("div", {
161
+ className: "flex flex-col gap-2",
162
+ "data-testid": "ei-import-result",
163
+ children: [/* @__PURE__ */ jsxDEV("p", {
164
+ className: "text-sm text-muted-foreground",
165
+ children: t("Inserted {inserted}, {conflicts} conflicts, {errors} errors.", {
166
+ conflicts: importResult.conflicts,
167
+ errors: importResult.errors.length,
168
+ inserted: insertedTotal
169
+ })
170
+ }, void 0, false), importResult.errors.length > 0 && /* @__PURE__ */ jsxDEV("ul", {
171
+ className: "flex flex-col gap-1",
172
+ "data-testid": "ei-import-errors",
173
+ children: importResult.errors.map((rowError) => /* @__PURE__ */ jsxDEV("li", {
174
+ className: "text-xs text-destructive",
175
+ children: t("line {line} ({table}): {message}", {
176
+ line: rowError.line,
177
+ message: rowError.message,
178
+ table: rowError.table
179
+ })
180
+ }, `${rowError.table}-${rowError.line.toString()}`, false))
181
+ }, void 0, false)]
182
+ }, void 0, true)]
183
+ }, void 0, true)]
184
+ }, void 0, true), error !== null && /* @__PURE__ */ jsxDEV("p", {
185
+ className: "text-sm text-destructive",
186
+ "data-testid": "ei-error",
187
+ role: "alert",
188
+ children: error
189
+ }, void 0, false)]
190
+ }, void 0, true);
191
+ };
192
+
193
+ export { ExportImportPanel };