@kitecd/cli 1.0.0 → 1.2.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 (80) hide show
  1. package/dist/doctor.js +181 -0
  2. package/dist/export.js +144 -0
  3. package/dist/ignore.js +38 -0
  4. package/dist/import.js +333 -0
  5. package/dist/index.js +271 -9
  6. package/dist/ops.js +338 -0
  7. package/dist/pack.js +15 -5
  8. package/dist/serve.js +27 -1
  9. package/dist/server/index.js +14562 -5621
  10. package/dist/upload.js +11 -3
  11. package/dist/verify.js +161 -0
  12. package/dist/web/assets/AuditLog-BFmJfgzL.js +1 -0
  13. package/dist/web/assets/ConfirmDialog-CJ8lJeUc.js +1 -0
  14. package/dist/web/assets/ConfirmDialog-D8avT8FJ.css +1 -0
  15. package/dist/web/assets/Dashboard-BTliTkq1.js +1 -0
  16. package/dist/web/assets/DefaultLayout-BG_y85yG.js +1 -0
  17. package/dist/web/assets/DefaultLayout-CZENO67n.css +1 -0
  18. package/dist/web/assets/FileExplorer-Bf32_MMS.js +1 -0
  19. package/dist/web/assets/LogBoard-0so-XiW3.css +1 -0
  20. package/dist/web/assets/LogBoard-CDz4n0DY.js +6 -0
  21. package/dist/web/assets/Login-C3zfObzP.js +1 -0
  22. package/dist/web/assets/Migration-BaKlcCkB.js +1 -0
  23. package/dist/web/assets/ProjectDetail-BL_D0OJY.js +1 -0
  24. package/dist/web/assets/ProjectDetail-ia6-z1kZ.css +1 -0
  25. package/dist/web/assets/ProjectList-7AnhOS7h.css +1 -0
  26. package/dist/web/assets/ProjectList-C4KuZEq4.js +1 -0
  27. package/dist/web/assets/Settings-CzWG9312.js +1 -0
  28. package/dist/web/assets/Storage-BCao_e7Y.js +1 -0
  29. package/dist/web/assets/Storage-DNadqpUy.css +1 -0
  30. package/dist/web/assets/{activity-DItEGOtI.js → activity-Ba-YPfmn.js} +1 -1
  31. package/dist/web/assets/archive-CJ5gDBKY.js +1 -0
  32. package/dist/web/assets/arrow-left-WEOMLXIi.js +1 -0
  33. package/dist/web/assets/chevron-right-DLiDJVJl.js +1 -0
  34. package/dist/web/assets/{circle-alert-Bfrn_ovD.js → circle-alert-DQzM-U4P.js} +1 -1
  35. package/dist/web/assets/clock-IwxBKgP4.js +1 -0
  36. package/dist/web/assets/constants-Ch47JPs3.js +1 -0
  37. package/dist/web/assets/copy-C1rADgcQ.js +1 -0
  38. package/dist/web/assets/createLucideIcon-CE-ry2oA.js +1 -0
  39. package/dist/web/assets/database-CI6acTSY.js +1 -0
  40. package/dist/web/assets/eye-Cas8HsmK.js +1 -0
  41. package/dist/web/assets/eye-off-4PD31MPV.js +1 -0
  42. package/dist/web/assets/file-text-C6nzo1us.js +1 -0
  43. package/dist/web/assets/folder-D9pvA6oI.js +1 -0
  44. package/dist/web/assets/folder-open-CGeIri0U.js +1 -0
  45. package/dist/web/assets/hard-drive-CGWwV4Ei.js +1 -0
  46. package/dist/web/assets/house-8ZvvlaEh.js +1 -0
  47. package/dist/web/assets/index-BFE6PEIL.js +2 -0
  48. package/dist/web/assets/index-XrzJwjrk.css +1 -0
  49. package/dist/web/assets/loader-circle-uiC7GaCG.js +1 -0
  50. package/dist/web/assets/plus-CfMi1Jv1.js +1 -0
  51. package/dist/web/assets/refresh-cw-0yb8DZDc.js +1 -0
  52. package/dist/web/assets/rotate-ccw-YevaWXw9.js +1 -0
  53. package/dist/web/assets/save-BuzCP3T1.js +1 -0
  54. package/dist/web/assets/scroll-text-BIYMFNN5.js +1 -0
  55. package/dist/web/assets/{server-C33taHNn.js → server-B31hj0g7.js} +1 -1
  56. package/dist/web/assets/{square-terminal-C8toRwjx.js → square-terminal-DT6aoXm0.js} +1 -1
  57. package/dist/web/assets/sun-BaEbKNDV.js +1 -0
  58. package/dist/web/assets/trash-2-BWqtqkeW.js +1 -0
  59. package/dist/web/index.html +3 -3
  60. package/package.json +2 -2
  61. package/dist/web/assets/Dashboard-pjIWWLub.js +0 -1
  62. package/dist/web/assets/DefaultLayout-Bj8fPWym.css +0 -1
  63. package/dist/web/assets/DefaultLayout-DelfwTTT.js +0 -1
  64. package/dist/web/assets/FileExplorer-xY5ejhhN.js +0 -1
  65. package/dist/web/assets/LogBoard-DzW-cEqH.css +0 -1
  66. package/dist/web/assets/LogBoard-tT61QjOx.js +0 -6
  67. package/dist/web/assets/Login-B4C149oC.js +0 -1
  68. package/dist/web/assets/ProjectDetail-Z8cZoqr5.js +0 -1
  69. package/dist/web/assets/ProjectList-9rbMuJeY.js +0 -1
  70. package/dist/web/assets/Settings-CtCNDUXY.js +0 -1
  71. package/dist/web/assets/clock-BPXGSCIV.js +0 -1
  72. package/dist/web/assets/constants-C4Zrkm2g.js +0 -1
  73. package/dist/web/assets/createLucideIcon-Cgv1AIRL.js +0 -1
  74. package/dist/web/assets/folder-open-jX-_Q7bA.js +0 -1
  75. package/dist/web/assets/index-C615tnMi.js +0 -2
  76. package/dist/web/assets/index-C9LiRc31.css +0 -1
  77. package/dist/web/assets/project-BFuaDcvV.js +0 -1
  78. package/dist/web/assets/refresh-cw-DWmqwQRn.js +0 -1
  79. package/dist/web/assets/save-BkiMrL9q.js +0 -1
  80. package/dist/web/assets/settings-CrCWmNyB.js +0 -1
package/dist/upload.js CHANGED
@@ -1,6 +1,8 @@
1
1
  import fs from 'fs';
2
+ import { randomUUID } from 'crypto';
2
3
  export async function uploadZip(options) {
3
- const { serverUrl, token, zipFilePath, projectId, preDeploy, postDeploy } = options;
4
+ const { serverUrl, token, zipFilePath, projectId, preDeploy, postDeploy, env, startedAt } = options;
5
+ const traceId = options.traceId || randomUUID();
4
6
  const fileData = await fs.promises.readFile(zipFilePath);
5
7
  const blob = new Blob([fileData], { type: 'application/zip' });
6
8
  const form = new FormData();
@@ -10,12 +12,17 @@ export async function uploadZip(options) {
10
12
  form.append('preDeploy', preDeploy);
11
13
  if (postDeploy)
12
14
  form.append('postDeploy', postDeploy);
15
+ if (env && Object.keys(env).length > 0)
16
+ form.append('env', JSON.stringify(env));
17
+ if (startedAt)
18
+ form.append('startedAt', startedAt);
13
19
  const endpoint = `${serverUrl.replace(/\/$/, '')}/api/deploy/upload`;
14
20
  try {
15
21
  const response = await fetch(endpoint, {
16
22
  method: 'POST',
17
23
  headers: {
18
24
  'Authorization': `Bearer ${token}`,
25
+ 'X-Kite-Trace-Id': traceId,
19
26
  },
20
27
  body: form,
21
28
  });
@@ -35,7 +42,7 @@ export async function uploadZip(options) {
35
42
  const reader = response.body.getReader();
36
43
  const decoder = new TextDecoder();
37
44
  let buffer = '';
38
- let result = { success: false };
45
+ let result = { success: false, traceId };
39
46
  while (true) {
40
47
  const { done, value } = await reader.read();
41
48
  if (done)
@@ -56,6 +63,7 @@ export async function uploadZip(options) {
56
63
  success: event.status === 'success',
57
64
  deployId: event.deployId,
58
65
  duration: event.duration,
66
+ traceId,
59
67
  };
60
68
  }
61
69
  }
@@ -71,7 +79,7 @@ export async function uploadZip(options) {
71
79
  if (event.event === 'log')
72
80
  process.stdout.write(event.data + '\n');
73
81
  if (event.event === 'status') {
74
- result = { success: event.status === 'success', deployId: event.deployId, duration: event.duration };
82
+ result = { success: event.status === 'success', deployId: event.deployId, duration: event.duration, traceId };
75
83
  }
76
84
  }
77
85
  catch { }
package/dist/verify.js ADDED
@@ -0,0 +1,161 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import chalk from 'chalk';
4
+ import ora from 'ora';
5
+ import { createClient } from '@libsql/client';
6
+ import { ensureKiteHome, getConfigPath, readGlobalConfig } from './home.js';
7
+ const tableExists = async (client, name) => {
8
+ const result = await client.execute({
9
+ sql: `SELECT name FROM sqlite_master WHERE type='table' AND name = ?`,
10
+ args: [name],
11
+ });
12
+ return result.rows.length > 0;
13
+ };
14
+ export async function runVerify(options = {}) {
15
+ const home = ensureKiteHome();
16
+ const dbPath = path.join(home, 'kite.db');
17
+ const configPath = getConfigPath();
18
+ console.log(chalk.bold('Kite migration verify'));
19
+ console.log(chalk.gray(` home: ${home}`));
20
+ console.log(chalk.gray(` db: ${dbPath}`));
21
+ console.log(chalk.gray(` config: ${configPath}`));
22
+ console.log();
23
+ const result = { ok: true, warnings: 0, failures: 0 };
24
+ if (!fs.existsSync(dbPath)) {
25
+ console.log(chalk.red('✗ kite.db not found. Run `kite serve` once or `kite import <file>` first.'));
26
+ process.exit(1);
27
+ }
28
+ if (!fs.existsSync(configPath)) {
29
+ console.log(chalk.yellow(`! ${configPath} not found. CLI push commands will need explicit --token / --server.`));
30
+ result.warnings++;
31
+ }
32
+ const globalConfig = fs.existsSync(configPath) ? readGlobalConfig() : { projectToken: {} };
33
+ const projectToken = globalConfig.projectToken || {};
34
+ const hasServerUrl = !!globalConfig.serverUrl;
35
+ const client = createClient({ url: `file:${dbPath}` });
36
+ try {
37
+ if (!(await tableExists(client, 'projects'))) {
38
+ console.log(chalk.red('✗ projects table missing in kite.db.'));
39
+ process.exit(1);
40
+ }
41
+ const spinner = ora('Loading projects...').start();
42
+ const projectsRows = (await client.execute(`SELECT id, name, deploy_path, token FROM projects`)).rows;
43
+ const deploymentRows = (await tableExists(client, 'deployments'))
44
+ ? (await client.execute(`SELECT id, project_id FROM deployments`)).rows
45
+ : [];
46
+ spinner.succeed(chalk.green(`Loaded ${projectsRows.length} projects, ${deploymentRows.length} deployment records`));
47
+ console.log(chalk.bold('\nProject checks:'));
48
+ for (const row of projectsRows) {
49
+ const id = String(row.id);
50
+ const name = String(row.name);
51
+ const deployPath = String(row.deploy_path || '');
52
+ const dbToken = row.token != null ? String(row.token) : '';
53
+ const labelHead = ` • ${name} (${id})`;
54
+ const subIssues = [];
55
+ if (!deployPath) {
56
+ subIssues.push(chalk.red('deploy_path is empty'));
57
+ result.failures++;
58
+ }
59
+ else {
60
+ const abs = path.isAbsolute(deployPath) ? deployPath : path.resolve(home, deployPath);
61
+ if (!fs.existsSync(abs)) {
62
+ subIssues.push(chalk.yellow(`deploy_path missing on disk: ${abs}`));
63
+ result.warnings++;
64
+ }
65
+ else {
66
+ const stat = fs.statSync(abs);
67
+ if (!stat.isDirectory()) {
68
+ subIssues.push(chalk.red(`deploy_path is not a directory: ${abs}`));
69
+ result.failures++;
70
+ }
71
+ }
72
+ }
73
+ const cliToken = projectToken[id];
74
+ if (!cliToken) {
75
+ subIssues.push(chalk.yellow('no projectToken in ~/.kite/config.json (run `kite config:set token`)'));
76
+ result.warnings++;
77
+ }
78
+ else if (dbToken && cliToken !== dbToken) {
79
+ subIssues.push(chalk.yellow('projectToken differs from DB token (push will hit auth error)'));
80
+ result.warnings++;
81
+ }
82
+ if (subIssues.length === 0) {
83
+ console.log(`${labelHead} ${chalk.green('ok')}`);
84
+ }
85
+ else {
86
+ console.log(`${labelHead} ${chalk.red(`${subIssues.length} issue(s)`)}`);
87
+ for (const issue of subIssues)
88
+ console.log(` - ${issue}`);
89
+ }
90
+ }
91
+ console.log(chalk.bold('\nGlobal config checks:'));
92
+ if (!hasServerUrl) {
93
+ console.log(` ${chalk.yellow('! serverUrl is empty')} - run \`kite config:set serverUrl <url>\``);
94
+ result.warnings++;
95
+ }
96
+ else {
97
+ console.log(` ${chalk.green('✓ serverUrl:')} ${globalConfig.serverUrl}`);
98
+ }
99
+ const tokenKeys = Object.keys(projectToken);
100
+ console.log(` ${chalk.green('✓ projectToken:')} ${tokenKeys.length} keys`);
101
+ console.log(chalk.bold('\nDeployment integrity:'));
102
+ const projectIdSet = new Set(projectsRows.map(r => String(r.id)));
103
+ let orphanCount = 0;
104
+ for (const d of deploymentRows) {
105
+ if (!projectIdSet.has(String(d.project_id)))
106
+ orphanCount++;
107
+ }
108
+ if (orphanCount === 0) {
109
+ console.log(` ${chalk.green('✓ no orphan deployments')}`);
110
+ }
111
+ else {
112
+ console.log(` ${chalk.yellow(`! ${orphanCount} deployment rows reference missing project_id`)}`);
113
+ result.warnings++;
114
+ }
115
+ if (options.checkServer) {
116
+ console.log(chalk.bold('\nServer health check:'));
117
+ if (!hasServerUrl) {
118
+ console.log(` ${chalk.yellow('! skip: serverUrl not configured')}`);
119
+ result.warnings++;
120
+ }
121
+ else {
122
+ const url = String(globalConfig.serverUrl).replace(/\/$/, '');
123
+ const timeoutMs = Math.max(1000, options.timeout || 5000);
124
+ const ctrl = new AbortController();
125
+ const timer = setTimeout(() => ctrl.abort(), timeoutMs);
126
+ try {
127
+ const resp = await fetch(url, { signal: ctrl.signal });
128
+ if (resp.status < 500) {
129
+ console.log(` ${chalk.green('✓')} ${url} -> ${resp.status}`);
130
+ }
131
+ else {
132
+ console.log(` ${chalk.yellow('!')} ${url} -> ${resp.status}`);
133
+ result.warnings++;
134
+ }
135
+ }
136
+ catch (err) {
137
+ console.log(` ${chalk.red('✗')} ${url} -> ${err.message}`);
138
+ console.log(chalk.gray(' Tip: run `kite serve` on the target machine first.'));
139
+ result.failures++;
140
+ }
141
+ finally {
142
+ clearTimeout(timer);
143
+ }
144
+ }
145
+ }
146
+ console.log();
147
+ if (result.failures > 0) {
148
+ console.log(chalk.red(`Verify failed: ${result.failures} error(s), ${result.warnings} warning(s).`));
149
+ process.exit(1);
150
+ }
151
+ else if (result.warnings > 0) {
152
+ console.log(chalk.yellow(`Verify finished with ${result.warnings} warning(s).`));
153
+ }
154
+ else {
155
+ console.log(chalk.green('Verify passed. Migration looks complete.'));
156
+ }
157
+ }
158
+ finally {
159
+ client.close();
160
+ }
161
+ }
@@ -0,0 +1 @@
1
+ import{B as e,E as t,H as n,K as r,R as i,_ as a,d as o,dt as s,j as c,k as l,m as u,n as ee,p as d,pt as f,s as p,t as m,u as h,v as g,y as _}from"./createLucideIcon-CE-ry2oA.js";import{t as te}from"./chevron-right-DLiDJVJl.js";import{t as ne}from"./circle-alert-DQzM-U4P.js";import{t as re}from"./file-text-C6nzo1us.js";import{t as ie}from"./refresh-cw-0yb8DZDc.js";import{t as ae}from"./scroll-text-BIYMFNN5.js";import{g as v,h as oe,l as se,o as y,r as b,s as x,u as ce,v as le}from"./index-BFE6PEIL.js";var ue=m(`chevron-left`,[[`path`,{d:`m15 18-6-6 6-6`,key:`1wnfg3`}]]),de=m(`funnel`,[[`path`,{d:`M10 20a1 1 0 0 0 .553.895l2 1A1 1 0 0 0 14 21v-7a2 2 0 0 1 .517-1.341L21.74 4.67A1 1 0 0 0 21 3H3a1 1 0 0 0-.742 1.67l7.225 7.989A2 2 0 0 1 10 14z`,key:`sc7q7i`}]]),fe={class:`space-y-6`},pe={class:`flex items-center justify-between`},me={class:`flex items-center gap-3`},he=[`disabled`],ge={class:`bg-panel border border-border rounded-lg p-4`},_e={class:`flex items-center gap-2 mb-3 text-sm text-textMuted`},ve={class:`grid grid-cols-1 md:grid-cols-4 gap-3`},ye=[`value`],be={class:`flex items-end`},xe=[`disabled`],Se={class:`bg-panel border border-border rounded-lg overflow-hidden`},Ce={key:0,class:`p-12 text-center text-textMuted`},S={key:1,class:`p-12 text-center`},C={key:2,class:`overflow-x-auto`},w={class:`w-full text-sm`},T=[`onClick`],E={class:`px-4 py-3 text-textMuted font-mono text-xs whitespace-nowrap`},D={class:`px-4 py-3`},O={class:`px-4 py-3`},k={key:0,class:`text-textMain`},A={key:1,class:`text-xs text-textMuted font-mono`},we={key:2,class:`text-textMuted`},Te={class:`px-4 py-3 text-textMain max-w-xs truncate`},Ee={class:`px-4 py-3 text-textMuted font-mono text-xs`},De={class:`px-4 py-3`},Oe={key:0,class:`inline-flex items-center gap-1 text-success text-xs`},ke={key:1,class:`inline-flex items-center gap-1 text-danger text-xs`},Ae={class:`px-4 py-3 text-right`},je={key:3,class:`flex items-center justify-between px-4 py-3 border-t border-border bg-base/30`},Me={class:`text-xs text-textMuted`},Ne={class:`flex items-center gap-2`},Pe=[`disabled`],Fe=[`disabled`],Ie={class:`relative w-full max-w-2xl h-full bg-panel border-l border-border overflow-y-auto`},Le={class:`sticky top-0 bg-panel border-b border-border px-6 py-4 flex items-center justify-between`},Re={class:`text-xs text-textMuted font-mono mt-1`},j={class:`p-6 space-y-5`},ze={class:`grid grid-cols-2 gap-4 text-sm`},Be={class:`text-xs text-textMuted font-mono mt-1`},Ve={key:0,class:`inline-flex items-center gap-1 text-success text-xs`},He={key:1,class:`inline-flex items-center gap-1 text-danger text-xs`},Ue={class:`text-textMain font-mono text-xs`},We={class:`text-textMain font-mono text-xs`},Ge={class:`col-span-2`},Ke={class:`text-textMain`},qe={key:0,class:`text-xs text-textMuted font-mono`},Je={key:0,class:`col-span-2`},Ye={class:`text-textMain`},Xe={key:1,class:`col-span-2`},Ze={class:`text-danger text-sm`},Qe={key:0,class:`space-y-3`},$e={class:`grid grid-cols-1 md:grid-cols-2 gap-3`},et={class:`bg-base border border-border rounded p-3 text-xs text-textMain font-mono overflow-auto max-h-96`},tt={class:`bg-base border border-border rounded p-3 text-xs text-textMain font-mono overflow-auto max-h-96`},M=50,N=_({__name:`AuditLog`,setup(m){let _=ee(),N=se(),nt=ce(),P=n([]),F=n(0),I=n(!1),L=n(null),R=n(0),z=n(``),B=n(``),V=n(``),H=[{value:``,label:`全部操作`},{value:`project.create`,label:`创建项目`},{value:`project.update`,label:`更新项目`},{value:`project.delete`,label:`删除项目`},{value:`project.token.rotate`,label:`重置项目 Token`},{value:`settings.update`,label:`更新系统设置`},{value:`admin_token.change`,label:`修改 Admin Token`},{value:`migration.export`,label:`导出迁移包`},{value:`migration.import`,label:`导入迁移包`},{value:`auth.login_failed`,label:`登录失败`}],U=e=>H.find(t=>t.value===e)?.label||e,W=e=>e.endsWith(`.delete`)?`text-danger bg-danger/10 border-danger/30`:e.endsWith(`.create`)?`text-success bg-success/10 border-success/30`:e.includes(`token`)||e.includes(`admin_token`)?`text-yellow-400 bg-yellow-400/10 border-yellow-400/30`:e.startsWith(`migration.`)?`text-primary bg-primary/10 border-primary/30`:e.startsWith(`auth.`)?`text-textMuted bg-white/5 border-border`:`text-primary bg-primary/10 border-primary/30`,G=h(()=>Math.max(1,Math.ceil(F.value/M))),K=h(()=>Math.floor(R.value/M)+1),q=e=>{if(!e)return`-`;let t=new Date(e),n=e=>String(e).padStart(2,`0`);return`${t.getFullYear()}-${n(t.getMonth()+1)}-${n(t.getDate())} ${n(t.getHours())}:${n(t.getMinutes())}:${n(t.getSeconds())}`},J=e=>{if(!e)return null;try{return JSON.parse(e)}catch{return e}},Y=h(()=>J(L.value?.before??null)),X=h(()=>J(L.value?.after??null)),rt=h(()=>Y.value!==null||X.value!==null);async function Z(){I.value=!0;try{let e=await _.fetchAuditLogs({action:z.value||void 0,targetId:B.value||void 0,targetType:V.value||void 0,limit:M,offset:R.value});P.value=e.rows||[],F.value=e.total||0}finally{I.value=!1}}function it(){z.value=``,B.value=``,V.value=``}function at(){K.value<G.value&&(R.value+=M,Z())}function ot(){R.value>0&&(R.value=Math.max(0,R.value-M),Z())}function st(e){L.value=e}function Q(){L.value=null}t(()=>{let e=N.query;typeof e.action==`string`&&(z.value=e.action),typeof e.targetId==`string`&&(B.value=e.targetId),typeof e.targetType==`string`&&(V.value=e.targetType),Z()});let $=null;return i([z,B,V],([e],[t])=>{$&&clearTimeout($),$=setTimeout(()=>{R.value=0;let e={};z.value&&(e.action=z.value),B.value&&(e.targetId=B.value),V.value&&(e.targetType=V.value),nt.replace({query:e}),Z()},e===t?300:0)}),i(()=>N.query.targetId,e=>{typeof e==`string`&&e!==B.value&&(B.value=e)}),(t,n)=>(l(),u(`div`,fe,[o(`div`,pe,[o(`div`,me,[g(r(ae),{class:`w-6 h-6 text-primary`}),n[3]||=o(`div`,null,[o(`h1`,{class:`text-2xl font-bold text-textMain`},`操作日志`),o(`p`,{class:`text-sm text-textMuted mt-1`},`所有服务端运维操作的审计记录,含变更前后状态`)],-1)]),o(`button`,{onClick:Z,disabled:I.value,class:`flex items-center gap-2 px-4 py-2 bg-panel border border-border rounded-md text-textMain hover:border-primary/50 transition-colors disabled:opacity-50`},[g(r(ie),{class:s([`w-4 h-4`,{"animate-spin":I.value}])},null,8,[`class`]),n[4]||=o(`span`,{class:`text-sm`},`刷新`,-1)],8,he)]),o(`div`,ge,[o(`div`,_e,[g(r(de),{class:`w-4 h-4`}),n[5]||=o(`span`,null,`筛选`,-1)]),o(`div`,ve,[o(`div`,null,[n[6]||=o(`label`,{class:`block text-xs text-textMuted mb-1`},`操作类型`,-1),e(o(`select`,{"onUpdate:modelValue":n[0]||=e=>z.value=e,class:`w-full bg-base border border-border rounded px-3 py-2 text-sm text-textMain focus:border-primary focus:outline-none`},[(l(),u(p,null,c(H,e=>o(`option`,{key:e.value,value:e.value},f(e.label),9,ye)),64))],512),[[oe,z.value]])]),o(`div`,null,[n[7]||=o(`label`,{class:`block text-xs text-textMuted mb-1`},`目标 ID`,-1),e(o(`input`,{"onUpdate:modelValue":n[1]||=e=>B.value=e,type:`text`,placeholder:`如 proj_xxxxx`,class:`w-full bg-base border border-border rounded px-3 py-2 text-sm text-textMain focus:border-primary focus:outline-none font-mono`},null,512),[[v,B.value]])]),o(`div`,null,[n[8]||=o(`label`,{class:`block text-xs text-textMuted mb-1`},`目标类型`,-1),e(o(`input`,{"onUpdate:modelValue":n[2]||=e=>V.value=e,type:`text`,placeholder:`如 project / settings`,class:`w-full bg-base border border-border rounded px-3 py-2 text-sm text-textMain focus:border-primary focus:outline-none`},null,512),[[v,V.value]])]),o(`div`,be,[o(`button`,{onClick:it,disabled:!z.value&&!B.value&&!V.value,class:`w-full px-3 py-2 bg-base border border-border text-textMuted rounded text-sm hover:border-textMuted hover:text-textMain transition-colors disabled:opacity-40 disabled:cursor-not-allowed flex items-center justify-center gap-1.5`},[g(r(b),{class:`w-3.5 h-3.5`}),n[9]||=a(` 清空筛选 `,-1)],8,xe)])])]),o(`div`,Se,[I.value&&P.value.length===0?(l(),u(`div`,Ce,` 加载中... `)):P.value.length===0?(l(),u(`div`,S,[g(r(ne),{class:`w-10 h-10 mx-auto text-textMuted mb-3`}),n[10]||=o(`p`,{class:`text-textMuted`},`暂无操作日志`,-1)])):(l(),u(`div`,C,[o(`table`,w,[n[13]||=o(`thead`,{class:`bg-base/50 border-b border-border`},[o(`tr`,{class:`text-left text-xs text-textMuted`},[o(`th`,{class:`px-4 py-3 font-medium`},`时间`),o(`th`,{class:`px-4 py-3 font-medium`},`操作`),o(`th`,{class:`px-4 py-3 font-medium`},`目标`),o(`th`,{class:`px-4 py-3 font-medium`},`摘要`),o(`th`,{class:`px-4 py-3 font-medium`},`来源 IP`),o(`th`,{class:`px-4 py-3 font-medium`},`状态`),o(`th`,{class:`px-4 py-3 font-medium`})])],-1),o(`tbody`,null,[(l(!0),u(p,null,c(P.value,e=>(l(),u(`tr`,{key:e.id,class:`border-b border-border last:border-0 hover:bg-base/50 transition-colors cursor-pointer`,onClick:t=>st(e)},[o(`td`,E,f(q(e.createdAt)),1),o(`td`,D,[o(`span`,{class:s([`inline-flex items-center px-2 py-0.5 rounded text-xs border`,W(e.action)])},f(U(e.action)),3)]),o(`td`,O,[e.targetName?(l(),u(`div`,k,f(e.targetName),1)):d(``,!0),e.targetId?(l(),u(`div`,A,f(e.targetId),1)):d(``,!0),!e.targetName&&!e.targetId?(l(),u(`span`,we,`-`)):d(``,!0)]),o(`td`,Te,f(e.summary||`-`),1),o(`td`,Ee,f(e.actorIp||`-`),1),o(`td`,De,[e.status===`success`?(l(),u(`span`,Oe,[g(r(x),{class:`w-3.5 h-3.5`}),n[11]||=a(` 成功 `,-1)])):(l(),u(`span`,ke,[g(r(y),{class:`w-3.5 h-3.5`}),n[12]||=a(` 失败 `,-1)]))]),o(`td`,Ae,[g(r(re),{class:`w-4 h-4 text-textMuted inline`})])],8,T))),128))])])])),P.value.length>0?(l(),u(`div`,je,[o(`span`,Me,`共 `+f(F.value)+` 条,第 `+f(K.value)+` / `+f(G.value)+` 页`,1),o(`div`,Ne,[o(`button`,{onClick:ot,disabled:R.value===0||I.value,class:`p-1.5 border border-border rounded text-textMuted hover:text-textMain hover:border-textMuted disabled:opacity-40 disabled:cursor-not-allowed`},[g(r(ue),{class:`w-4 h-4`})],8,Pe),o(`button`,{onClick:at,disabled:K.value>=G.value||I.value,class:`p-1.5 border border-border rounded text-textMuted hover:text-textMain hover:border-textMuted disabled:opacity-40 disabled:cursor-not-allowed`},[g(r(te),{class:`w-4 h-4`})],8,Fe)])])):d(``,!0)]),L.value?(l(),u(`div`,{key:0,class:`fixed inset-0 z-50 flex justify-end`,onClick:le(Q,[`self`])},[o(`div`,{class:`fixed inset-0 bg-black/40`,onClick:Q}),o(`div`,Ie,[o(`div`,Le,[o(`div`,null,[n[14]||=o(`h2`,{class:`text-lg font-bold text-textMain`},`操作详情`,-1),o(`p`,Re,f(L.value.id),1)]),o(`button`,{onClick:Q,class:`p-1.5 hover:bg-base rounded text-textMuted hover:text-textMain`},[g(r(b),{class:`w-5 h-5`})])]),o(`div`,j,[o(`div`,ze,[o(`div`,null,[n[15]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`操作类型`,-1),o(`span`,{class:s([`inline-flex items-center px-2 py-0.5 rounded text-xs border`,W(L.value.action)])},f(U(L.value.action)),3),o(`div`,Be,f(L.value.action),1)]),o(`div`,null,[n[18]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`状态`,-1),L.value.status===`success`?(l(),u(`span`,Ve,[g(r(x),{class:`w-3.5 h-3.5`}),n[16]||=a(` 成功 `,-1)])):(l(),u(`span`,He,[g(r(y),{class:`w-3.5 h-3.5`}),n[17]||=a(` 失败 `,-1)]))]),o(`div`,null,[n[19]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`时间`,-1),o(`div`,Ue,f(q(L.value.createdAt)),1)]),o(`div`,null,[n[20]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`来源 IP`,-1),o(`div`,We,f(L.value.actorIp||`-`),1)]),o(`div`,Ge,[n[21]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`目标`,-1),o(`div`,Ke,f(L.value.targetName||`-`),1),L.value.targetId?(l(),u(`div`,qe,f(L.value.targetType)+`: `+f(L.value.targetId),1)):d(``,!0)]),L.value.summary?(l(),u(`div`,Je,[n[22]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`摘要`,-1),o(`div`,Ye,f(L.value.summary),1)])):d(``,!0),L.value.errorMessage?(l(),u(`div`,Xe,[n[23]||=o(`div`,{class:`text-xs text-textMuted mb-1`},`错误信息`,-1),o(`div`,Ze,f(L.value.errorMessage),1)])):d(``,!0)]),rt.value?(l(),u(`div`,Qe,[n[26]||=o(`div`,{class:`text-xs text-textMuted`},`变更前后`,-1),o(`div`,$e,[o(`div`,null,[n[24]||=o(`div`,{class:`text-xs text-danger mb-1`},`Before`,-1),o(`pre`,et,[o(`code`,null,f(Y.value===null?`(无)`:JSON.stringify(Y.value,null,2)),1)])]),o(`div`,null,[n[25]||=o(`div`,{class:`text-xs text-success mb-1`},`After`,-1),o(`pre`,tt,[o(`code`,null,f(X.value===null?`(无)`:JSON.stringify(X.value,null,2)),1)])])])])):d(``,!0)])])])):d(``,!0)]))}});export{N as default};
@@ -0,0 +1 @@
1
+ import{B as e,H as t,K as n,M as r,P as i,R as a,_ as o,d as s,dt as c,f as l,k as u,m as d,p as f,pt as p,u as m,v as h,w as g,y as _,z as v}from"./createLucideIcon-CE-ry2oA.js";import{t as y}from"./refresh-cw-0yb8DZDc.js";import{_ as b,a as x,d as S,g as C,i as w,r as T,t as E,v as D}from"./index-BFE6PEIL.js";var O={class:`flex items-start space-x-4`},k={class:`flex-1 min-w-0`},A={class:`text-base font-semibold text-textMain`},j={key:0,class:`text-sm text-textMuted mt-1 whitespace-pre-line`},M=[`disabled`],N={key:0,class:`mt-5`},P={key:0,class:`text-xs text-textMuted mb-2`},F=[`disabled`,`placeholder`,`onKeydown`],I={class:`mt-6 flex items-center justify-end space-x-2`},L=[`disabled`],R=[`disabled`],z=E(_({__name:`ConfirmDialog`,props:{open:{type:Boolean},title:{default:`确认操作`},message:{default:``},tone:{default:`info`},confirmText:{default:`确认`},cancelText:{default:`取消`},loading:{type:Boolean,default:!1},requireText:{default:``},requireTextPlaceholder:{default:``},requireTextHint:{default:``}},emits:[`confirm`,`cancel`,`update:open`],setup(_,{emit:E}){let z=_,B=E,V=t(``),H=t(null),U=m(()=>z.tone===`info`?x:w),W=m(()=>{switch(z.tone){case`danger`:return{border:`border-danger/30`,iconBg:`bg-danger/10`,iconText:`text-danger`,confirmBtn:`bg-danger text-white hover:bg-danger/90`};case`warning`:return{border:`border-yellow-400/30`,iconBg:`bg-yellow-400/10`,iconText:`text-yellow-400`,confirmBtn:`bg-yellow-400 text-black hover:bg-yellow-300`};default:return{border:`border-border`,iconBg:`bg-primary/10`,iconText:`text-primary`,confirmBtn:`bg-primary text-white hover:bg-primary/90`}}}),G=m(()=>z.requireText.length>0),K=m(()=>z.loading?!1:G.value?V.value.trim()===z.requireText:!0);function q(){z.loading||(B(`cancel`),B(`update:open`,!1))}function J(){K.value&&B(`confirm`)}return a(()=>z.open,async e=>{e&&(V.value=``,await g(),H.value?.focus())}),(t,a)=>(u(),l(S,{name:`fade`},{default:v(()=>[_.open?(u(),d(`div`,{key:0,class:`fixed inset-0 z-[60] flex items-center justify-center bg-black/60 backdrop-blur-sm`,onClick:D(q,[`self`])},[s(`div`,{class:c([`bg-panel border rounded-xl w-full max-w-lg p-6 shadow-2xl`,W.value.border]),onKeydown:b(q,[`esc`])},[s(`div`,O,[s(`div`,{class:c([`flex-shrink-0 w-10 h-10 rounded-lg flex items-center justify-center`,W.value.iconBg])},[(u(),l(i(U.value),{class:c([`w-5 h-5`,W.value.iconText])},null,8,[`class`]))],2),s(`div`,k,[s(`h3`,A,p(_.title),1),_.message?(u(),d(`p`,j,p(_.message),1)):f(``,!0),r(t.$slots,`default`,{},void 0,!0)]),s(`button`,{onClick:q,disabled:_.loading,class:`text-textMuted hover:text-textMain rounded p-1 disabled:opacity-50`},[h(n(T),{class:`w-4 h-4`})],8,M)]),G.value?(u(),d(`div`,N,[_.requireTextHint?(u(),d(`p`,P,p(_.requireTextHint),1)):f(``,!0),e(s(`input`,{ref_key:`inputRef`,ref:H,"onUpdate:modelValue":a[0]||=e=>V.value=e,type:`text`,disabled:_.loading,placeholder:_.requireTextPlaceholder||_.requireText,class:`w-full bg-base border border-border rounded-md px-3 py-2 text-textMain font-mono focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 text-sm disabled:opacity-60`,onKeydown:b(D(J,[`prevent`]),[`enter`])},null,40,F),[[C,V.value]])])):f(``,!0),s(`div`,I,[s(`button`,{onClick:q,disabled:_.loading,class:`px-4 py-2 text-sm font-medium text-textMuted hover:text-textMain dark:hover:bg-white/5 hover:bg-black/5 rounded-md transition-colors disabled:opacity-50 disabled:cursor-not-allowed`},p(_.cancelText),9,L),s(`button`,{onClick:J,disabled:!K.value,class:c([`px-4 py-2 text-sm font-medium rounded-md transition-colors flex items-center disabled:opacity-50 disabled:cursor-not-allowed`,W.value.confirmBtn])},[_.loading?(u(),l(n(y),{key:0,class:`w-4 h-4 mr-2 animate-spin`})):f(``,!0),o(` `+p(_.loading?`处理中...`:_.confirmText),1)],10,R)])],34)])):f(``,!0)]),_:3}))}}),[[`__scopeId`,`data-v-543f8e0c`]]);export{z as t};
@@ -0,0 +1 @@
1
+ .fade-enter-active[data-v-543f8e0c],.fade-leave-active[data-v-543f8e0c]{transition:opacity .15s}.fade-enter-from[data-v-543f8e0c],.fade-leave-to[data-v-543f8e0c]{opacity:0}
@@ -0,0 +1 @@
1
+ import{E as e,H as t,K as n,P as r,_ as i,d as a,dt as o,f as s,g as c,j as l,k as u,m as d,n as f,p,pt as m,s as h,u as g,v as _,y as v}from"./createLucideIcon-CE-ry2oA.js";import{t as ee}from"./activity-Ba-YPfmn.js";import{t as te}from"./circle-alert-DQzM-U4P.js";import{t as ne}from"./clock-IwxBKgP4.js";import{t as re}from"./server-B31hj0g7.js";import{i as ie,u as ae}from"./index-BFE6PEIL.js";import{t as oe}from"./constants-Ch47JPs3.js";var y={class:`bg-panel border border-border rounded-xl p-6 shadow-sm`},b={class:`flex items-center justify-between mb-4`},x={class:`text-xs text-textMuted mt-1`},S={key:0,class:`text-sm text-textMuted py-8 text-center`},C={key:1,class:`text-sm text-textMuted py-8 text-center`},w={key:2,class:`flex gap-3 overflow-x-auto`},T={class:`flex flex-col justify-around text-[10px] text-textMuted pt-1 pb-1 select-none`},E={class:`flex gap-[3px]`},D=[`title`,`aria-label`],se=v({__name:`DeployHeatmap`,props:{cells:{},loading:{type:Boolean}},setup(e){let t=e,n=[`日`,`一`,`二`,`三`,`四`,`五`,`六`],r=g(()=>{if(!t.cells.length)return{columns:[],max:0};let e=[...t.cells].sort((e,t)=>e.date.localeCompare(t.date)),n=new Date(e[0].date+`T00:00:00Z`).getUTCDay(),r=Array(n).fill(null).concat(e),i=[];for(let e=0;e<r.length;e+=7)i.push(r.slice(e,e+7));if(i.length&&i[i.length-1].length<7)for(;i[i.length-1].length<7;)i[i.length-1].push(null);return{columns:i,max:e.reduce((e,t)=>Math.max(e,t.count),0)}});function i(e,t){if(!e)return`bg-border/40`;if(t<=1)return`bg-emerald-500`;let n=e/t;return n>=.75?`bg-emerald-400`:n>=.5?`bg-emerald-500/80`:n>=.25?`bg-emerald-600/70`:`bg-emerald-700/60`}function s(e){return e?`${e.date} · ${e.count} 次部署`:``}let f=g(()=>t.cells.reduce((e,t)=>e+t.count,0));return(t,p)=>(u(),d(`div`,y,[a(`div`,b,[a(`div`,null,[p[0]||=a(`h3`,{class:`text-base font-semibold text-textMain`},`部署频次热力图`,-1),a(`p`,x,`最近 `+m(e.cells.length)+` 天 · 共 `+m(f.value)+` 次`,1)]),p[1]||=c(`<div class="flex items-center gap-1 text-xs text-textMuted"><span class="mr-1">少</span><span class="w-3 h-3 rounded-sm bg-border/40"></span><span class="w-3 h-3 rounded-sm bg-emerald-700/60"></span><span class="w-3 h-3 rounded-sm bg-emerald-600/70"></span><span class="w-3 h-3 rounded-sm bg-emerald-500/80"></span><span class="w-3 h-3 rounded-sm bg-emerald-400"></span><span class="ml-1">多</span></div>`,1)]),e.loading?(u(),d(`div`,S,`加载中…`)):e.cells.length?(u(),d(`div`,w,[a(`div`,T,[(u(),d(h,null,l(n,e=>a(`span`,{key:e,class:`h-3 leading-3`},m(e),1)),64))]),a(`div`,E,[(u(!0),d(h,null,l(r.value.columns,(e,t)=>(u(),d(`div`,{key:t,class:`flex flex-col gap-[3px]`},[(u(!0),d(h,null,l(e,(e,t)=>(u(),d(`div`,{key:t,class:o([`w-3 h-3 rounded-sm transition-transform hover:scale-125`,e?i(e.count,r.value.max):`bg-transparent`]),title:s(e),"aria-label":s(e)||`无数据`},null,10,D))),128))]))),128))])])):(u(),d(`div`,C,`暂无数据`))]))}}),O={class:`bg-panel border border-border rounded-xl p-6 shadow-sm`},k={class:`flex items-center justify-between mb-4`},A={class:`text-base font-semibold text-textMain`},j={class:`text-xs text-textMuted mt-1`},M={class:`text-success`},N={class:`text-danger`},P={class:`font-mono`},F={key:0,class:`text-sm text-textMuted py-8 text-center`},I={key:1,class:`text-sm text-textMuted py-8 text-center`},L=[`viewBox`],R={class:`text-textMuted`,stroke:`currentColor`,"stroke-opacity":`0.15`},z=[`x2`],B=[`y1`,`x2`,`y2`],V=[`y1`,`x2`,`y2`],H={class:`text-textMuted`,fill:`currentColor`,"font-size":`10`,"font-family":`monospace`},U=[`x`,`y`],W=[`x`,`y`],G=[`x`,`y`],K=[`points`],ce=[`cx`,`cy`,`r`],le={class:`text-textMuted`,fill:`currentColor`,"font-size":`10`,"font-family":`monospace`},ue=[`x`,`y`],q=600,J=180,Y=32,X=12,Z=16,Q=28,de=v({__name:`SuccessRateChart`,props:{points:{},loading:{type:Boolean}},setup(e){let t=e,n=g(()=>{let e=t.points;if(!e.length)return{coords:[],polyline:``,avg:null};let n=q-Y-X,r=J-Z-Q,i=e.length>1?n/(e.length-1):0,a=e.map((e,t)=>{let n=e.rate??1;return{x:Y+i*t,y:Z+r*(1-n),p:e}}),o=e.filter(e=>e.rate!==null),s=o.length?o.reduce((e,t)=>e+(t.rate??0),0)/o.length:null;return{coords:a,polyline:a.map(e=>`${e.x.toFixed(1)},${e.y.toFixed(1)}`).join(` `),avg:s}}),r=g(()=>{let e=t.points;return e.length?(e.length<=7?e.map((e,t)=>t):[0,Math.floor(e.length/2),e.length-1]).map(t=>({x:n.value.coords[t]?.x??0,label:e[t].date.slice(5)})):[]}),s=g(()=>{let e=0,n=0,r=0;for(let i of t.points)e+=i.success,n+=i.failed,r+=i.total;return{success:e,failed:n,total:r}});function c(e){return e==null?`—`:`${(e*100).toFixed(1)}%`}return(t,f)=>(u(),d(`div`,O,[a(`div`,k,[a(`div`,null,[a(`h3`,A,`近 `+m(e.points.length)+` 天部署成功率`,1),a(`p`,j,[i(` 总计 `+m(s.value.total)+` 次 · 成功 `,1),a(`span`,M,m(s.value.success),1),f[0]||=i(` · 失败 `,-1),a(`span`,N,m(s.value.failed),1),f[1]||=i(` · 均值 `,-1),a(`span`,P,m(c(n.value.avg)),1)])])]),e.loading?(u(),d(`div`,F,`加载中…`)):s.value.total?(u(),d(`svg`,{key:2,viewBox:`0 0 ${q} ${J}`,class:`w-full h-44 overflow-visible`,role:`img`,"aria-label":`部署成功率折线图`},[a(`g`,R,[a(`line`,{x1:Y,y1:Z,x2:q-X,y2:Z},null,8,z),a(`line`,{x1:Y,y1:Z+(J-Z-Q)*.5,x2:q-X,y2:Z+(J-Z-Q)*.5},null,8,B),a(`line`,{x1:Y,y1:J-Q,x2:q-X,y2:J-Q},null,8,V)]),a(`g`,H,[a(`text`,{x:Y-6,y:Z+3,"text-anchor":`end`},`100%`,8,U),a(`text`,{x:Y-6,y:Z+(J-Z-Q)*.5+3,"text-anchor":`end`},`50%`,8,W),a(`text`,{x:Y-6,y:J-Q+3,"text-anchor":`end`},`0%`,8,G)]),a(`polyline`,{points:n.value.polyline,fill:`none`,class:`text-primary`,stroke:`currentColor`,"stroke-width":`1.5`,"stroke-linejoin":`round`,"stroke-linecap":`round`},null,8,K),a(`g`,null,[(u(!0),d(h,null,l(n.value.coords,(e,t)=>(u(),d(`circle`,{key:t,cx:e.x,cy:e.y,r:e.p.total===0?1.5:3,class:o(e.p.rate===null?`text-textMuted`:e.p.rate<.8?`text-danger`:`text-primary`),fill:`currentColor`},[a(`title`,null,m(e.p.date)+` · 成功 `+m(e.p.success)+` / 失败 `+m(e.p.failed)+` · `+m(c(e.p.rate)),1)],10,ce))),128))]),a(`g`,le,[(u(!0),d(h,null,l(r.value,(e,t)=>(u(),d(`text`,{key:t,x:e.x,y:J-Q+14,"text-anchor":`middle`},m(e.label),9,ue))),128))])],8,L)):(u(),d(`div`,I,`暂无部署`))]))}}),fe={class:`space-y-6 max-w-7xl mx-auto`},pe={class:`flex justify-between items-center mb-8`},me={class:`text-sm text-textMuted bg-panel px-3 py-1 rounded-full border border-border`},he={class:`grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6`},ge={class:`ml-4`},_e={class:`text-sm font-medium text-textMuted`},ve={class:`text-3xl font-bold text-textMain mt-1 font-mono`},ye={class:`grid grid-cols-1 lg:grid-cols-3 gap-6`},be={class:`lg:col-span-2`},xe={class:`bg-panel border border-border rounded-xl p-6 shadow-sm`},Se={class:`flex items-center justify-between mb-4`},Ce={class:`text-base font-semibold text-textMain flex items-center gap-2`},we={key:0,class:`text-sm text-textMuted py-6 text-center`},Te={key:1,class:`space-y-2`},Ee=[`onClick`],De={class:`min-w-0`},Oe={class:`text-sm font-medium text-textMain truncate`},ke={class:`text-xs text-textMuted font-mono mt-0.5`},Ae={key:2,class:`text-sm text-textMuted py-6 text-center`},je={class:`mt-8`},Me={class:`bg-panel border border-border rounded-xl overflow-hidden shadow-sm`},Ne={class:`divide-y divide-border`},Pe={key:0,class:`px-6 py-8 text-center text-textMuted`},$=[`onClick`],Fe={class:`flex items-center space-x-4`},Ie={class:`text-sm font-medium text-textMain`},Le={class:`font-mono text-primary ml-1`},Re={class:`text-xs text-textMuted mt-1`},ze={class:`text-right`},Be={class:`text-sm text-textMuted font-mono`},Ve=v({__name:`Dashboard`,setup(c){let v=f(),y=ae(),b=t([]),x=t([]),S=t([]),C=t(!0);e(async()=>{v.fetchProjects(),v.fetchLogs();try{let[e,t,n]=await Promise.all([v.fetchHeatmap(30),v.fetchSuccessRate(14),v.fetchFailureTop(5,30)]);b.value=e?.cells??[],x.value=t?.points??[],S.value=n?.items??[]}finally{C.value=!1}});let w=g(()=>v.logs.slice(0,5));function T(e){return!Number.isFinite(e)||e<0?`-`:e<1e3?`${e}ms`:e<6e4?`${(e/1e3).toFixed(1)}s`:`${Math.floor(e/6e4)}m ${Math.round(e%6e4/1e3)}s`}function E(e){if(!e.endTime||!e.startTime)return null;let t=new Date(e.endTime).getTime()-new Date(e.startTime).getTime();return Number.isFinite(t)&&t>=0?t:null}let D=g(()=>{let e=v.logs.filter(e=>e.status!==`running`).map(E).filter(e=>e!==null);if(e.length===0)return`-`;let t=e.reduce((e,t)=>e+t,0);return T(Math.round(t/e.length))}),O=g(()=>[{label:`总计项目数`,value:v.projects.length,icon:re,color:`text-primary`},{label:`成功部署`,value:v.logs.filter(e=>e.status===`success`).length,icon:ee,color:`text-success`},{label:`失败任务`,value:v.logs.filter(e=>e.status===`failed`).length,icon:te,color:`text-danger`},{label:`平均耗时`,value:D.value,icon:ne,color:`text-blue-400`}]);function k(e){y.push({name:`LogBoard`,query:{id:e}})}function A(e){y.push({name:`ProjectDetail`,params:{id:e}})}function j(e){return`${(e*100).toFixed(1)}%`}return(e,t)=>(u(),d(`div`,fe,[a(`div`,pe,[t[0]||=a(`h1`,{class:`text-2xl font-bold text-textMain tracking-tight`},`Kite 概览`,-1),a(`div`,me,` 当前版本:v`+m(n(oe)),1)]),a(`div`,he,[(u(!0),d(h,null,l(O.value,e=>(u(),d(`div`,{key:e.label,class:`bg-panel border border-border rounded-xl p-6 flex items-center shadow-sm hover:border-primary/50 transition-colors`},[a(`div`,{class:o([`w-12 h-12 rounded-lg bg-base flex items-center justify-center border border-border`,e.color])},[(u(),s(r(e.icon),{class:`w-6 h-6`}))],2),a(`div`,ge,[a(`p`,_e,m(e.label),1),a(`p`,ve,m(e.value),1)])]))),128))]),_(se,{cells:b.value,loading:C.value},null,8,[`cells`,`loading`]),a(`div`,ye,[a(`div`,be,[_(de,{points:x.value,loading:C.value},null,8,[`points`,`loading`])]),a(`div`,xe,[a(`div`,Se,[a(`div`,null,[a(`h3`,Ce,[_(n(ie),{class:`w-4 h-4 text-danger`}),t[1]||=i(` 失败率 TopN `,-1)]),t[2]||=a(`p`,{class:`text-xs text-textMuted mt-1`},`近 30 天 · 至少 3 次部署`,-1)])]),C.value?(u(),d(`div`,we,`加载中…`)):S.value.length?(u(),d(`ul`,Te,[(u(!0),d(h,null,l(S.value,e=>(u(),d(`li`,{key:e.projectId,onClick:t=>A(e.projectId),class:`flex items-center justify-between px-3 py-2 rounded-lg border border-border bg-base hover:border-primary/40 cursor-pointer transition-colors`},[a(`div`,De,[a(`p`,Oe,m(e.projectName),1),a(`p`,ke,m(e.failed)+` / `+m(e.total)+` 失败`,1)]),a(`span`,{class:o([`font-mono text-sm shrink-0 ml-3`,e.rate>=.5?`text-danger`:e.rate>=.2?`text-yellow-400`:`text-textMuted`])},m(j(e.rate)),3)],8,Ee))),128))])):(u(),d(`div`,Ae,`暂无符合条件的项目`))])]),a(`div`,je,[t[3]||=a(`h2`,{class:`text-lg font-semibold text-textMain mb-4`},`最近部署活动`,-1),a(`div`,Me,[a(`ul`,Ne,[w.value.length===0?(u(),d(`li`,Pe,`暂无部署活动`)):p(``,!0),(u(!0),d(h,null,l(w.value,e=>(u(),d(`li`,{key:e.id,onClick:t=>k(e.id),class:`px-6 py-4 flex items-center justify-between dark:hover:bg-white/5 hover:bg-black/5 transition-colors cursor-pointer`},[a(`div`,Fe,[a(`div`,{class:o([`w-2 h-2 rounded-full shadow-[0_0_8px_currentColor]`,e.status===`success`?`bg-success text-success`:e.status===`failed`?`bg-danger text-danger`:`bg-primary text-primary`])},null,2),a(`div`,null,[a(`p`,Ie,[i(`部署`+m(e.status===`success`?`完成`:e.status===`failed`?`失败`:`中`)+`: `,1),a(`span`,Le,m(e.projectName),1)]),a(`p`,Re,`触发源: `+m(e.triggerSource),1)])]),a(`div`,ze,[a(`p`,Be,m(new Date(e.startTime).toLocaleString()),1),a(`p`,{class:o([`text-xs mt-1`,e.status===`success`?`text-success`:e.status===`failed`?`text-danger`:`text-primary`])},m(T(E(e)??NaN)),3)])],8,$))),128))])])])]))}});export{Ve as default};
@@ -0,0 +1 @@
1
+ import{K as e,N as t,P as n,_ as r,d as i,dt as a,f as o,j as s,k as c,m as l,n as u,pt as d,s as f,t as p,u as m,v as h,y as g,z as _}from"./createLucideIcon-CE-ry2oA.js";import{t as v}from"./database-CI6acTSY.js";import{t as y}from"./hard-drive-CGWwV4Ei.js";import{i as b,n as x,r as ee,t as te}from"./sun-BaEbKNDV.js";import{t as ne}from"./scroll-text-BIYMFNN5.js";import{t as S}from"./square-terminal-DT6aoXm0.js";import{c as C,d as w,l as T,t as E,u as D}from"./index-BFE6PEIL.js";import{t as O}from"./constants-Ch47JPs3.js";var k=p(`folder-archive`,[[`circle`,{cx:`15`,cy:`19`,r:`2`,key:`u2pros`}],[`path`,{d:`M20.9 19.8A2 2 0 0 0 22 18V8a2 2 0 0 0-2-2h-7.9a2 2 0 0 1-1.69-.9L9.6 3.9A2 2 0 0 0 7.93 3H4a2 2 0 0 0-2 2v13a2 2 0 0 0 2 2h5.1`,key:`1jj40k`}],[`path`,{d:`M15 11v-1`,key:`cntcp`}],[`path`,{d:`M15 17v-2`,key:`1279jj`}]]),A=p(`layout-dashboard`,[[`rect`,{width:`7`,height:`9`,x:`3`,y:`3`,rx:`1`,key:`10lvy0`}],[`rect`,{width:`7`,height:`5`,x:`14`,y:`3`,rx:`1`,key:`16une8`}],[`rect`,{width:`7`,height:`9`,x:`14`,y:`12`,rx:`1`,key:`1hutg5`}],[`rect`,{width:`7`,height:`5`,x:`3`,y:`16`,rx:`1`,key:`ldoo1y`}]]),j=p(`log-out`,[[`path`,{d:`m16 17 5-5-5-5`,key:`1bji2h`}],[`path`,{d:`M21 12H9`,key:`dn1m92`}],[`path`,{d:`M9 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h4`,key:`1uf3rs`}]]),M=`data:image/svg+xml,%3csvg%20width='32'%20height='32'%20viewBox='0%200%2032%2032'%20fill='none'%20xmlns='http://www.w3.org/2000/svg'%3e%3crect%20width='32'%20height='32'%20rx='8'%20fill='%2309090B'/%3e%3cpath%20d='M7%2022L16%205L25%2022L16%2018L7%2022Z'%20fill='%233B82F6'/%3e%3cpath%20d='M16%2018L12%2027L16%2024L20%2027L16%2018Z'%20fill='%2310B981'/%3e%3c/svg%3e`,N={class:`min-h-screen bg-base flex`},P={class:`w-64 border-r border-border bg-panel flex flex-col hidden md:flex`},F={class:`h-16 flex items-center px-6 border-b border-border`},I=[`src`],L={class:`ml-2 text-xs bg-primary/20 text-primary px-2 py-0.5 rounded-full`},re=[`title`,`aria-label`],R={class:`flex-1 py-6 px-3 space-y-1`},z={class:`font-medium text-sm`},B={class:`p-4 border-t border-border space-y-1`},V={class:`flex-1 flex flex-col h-screen overflow-hidden`},H={class:`h-16 border-b border-border bg-panel flex items-center px-4 md:hidden`},U=[`src`],W=[`title`,`aria-label`],G={class:`flex-1 overflow-auto p-6 md:p-8`},K=E(g({__name:`DefaultLayout`,setup(p){let g=T(),E=D(),K=u(),q=C(),J=[{name:`概览`,path:`/`,icon:A},{name:`项目管理`,path:`/projects`,icon:k},{name:`部署日志`,path:`/logs`,icon:S},{name:`操作日志`,path:`/audit`,icon:ne},{name:`存储`,path:`/storage`,icon:y},{name:`数据迁移`,path:`/migration`,icon:v}],Y=[`light`,`dark`,`system`],X={light:`浅色`,dark:`深色`,system:`跟随系统`},Z=m(()=>q.mode===`system`?b:q.mode===`dark`?ee:te),Q=m(()=>`外观:${X[q.mode]}(点击切换)`),$=()=>{let e=Y[(Y.indexOf(q.mode)+1)%Y.length];q.setMode(e)},ie=()=>{K.logout(),E.push(`/login`)};return(u,p)=>{let m=t(`router-link`),v=t(`router-view`);return c(),l(`div`,N,[i(`aside`,P,[i(`div`,F,[i(`img`,{src:e(M),alt:`Kite Logo`,class:`w-6 h-6 mr-2`},null,8,I),p[0]||=i(`span`,{class:`text-lg font-bold text-textMain tracking-wide`},`KITE`,-1),i(`span`,L,`v`+d(e(O)),1),i(`button`,{onClick:$,title:Q.value,"aria-label":Q.value,class:`ml-auto p-1.5 rounded-md text-textMuted hover:text-textMain dark:hover:bg-white/5 hover:bg-black/5 transition-colors`},[(c(),o(n(Z.value),{class:`w-4 h-4`}))],8,re)]),i(`nav`,R,[(c(),l(f,null,s(J,t=>h(m,{key:t.path,to:t.path,class:a([`flex items-center px-3 py-2.5 rounded-md transition-all duration-200 group`,[e(g).path===t.path||t.path!==`/`&&e(g).path.startsWith(t.path)?`bg-primary/10 text-primary shadow-[inset_2px_0_0_0_#3b82f6]`:`text-textMuted dark:hover:bg-white/5 hover:bg-black/5 hover:text-textMain`]])},{default:_(()=>[(c(),o(n(t.icon),{class:a([`w-5 h-5 mr-3`,e(g).path===t.path||t.path!==`/`&&e(g).path.startsWith(t.path)?`text-primary`:`text-textMuted group-hover:text-textMain`])},null,8,[`class`])),i(`span`,z,d(t.name),1)]),_:2},1032,[`to`,`class`])),64))]),i(`div`,B,[h(m,{to:`/settings`,class:a([`flex items-center w-full px-3 py-2 text-sm rounded-md transition-colors`,e(g).path===`/settings`?`bg-primary/10 text-primary shadow-[inset_2px_0_0_0_#3b82f6]`:`text-textMuted hover:text-textMain dark:hover:bg-white/5 hover:bg-black/5`])},{default:_(()=>[h(e(x),{class:a([`w-5 h-5 mr-3`,e(g).path===`/settings`?`text-primary`:`text-textMuted`])},null,8,[`class`]),p[1]||=r(` 系统设置 `,-1)]),_:1},8,[`class`]),i(`button`,{onClick:ie,class:`flex items-center w-full px-3 py-2 text-sm text-danger hover:bg-danger/10 rounded-md transition-colors`},[h(e(j),{class:`w-5 h-5 mr-3`}),p[2]||=r(` 退出登录 `,-1)])])]),i(`main`,V,[i(`header`,H,[i(`img`,{src:e(M),alt:`Kite Logo`,class:`w-6 h-6 mr-2`},null,8,U),p[3]||=i(`span`,{class:`text-lg font-bold text-textMain`},`KITE`,-1),i(`button`,{onClick:$,title:Q.value,"aria-label":Q.value,class:`ml-auto p-1.5 rounded-md text-textMuted hover:text-textMain dark:hover:bg-white/5 hover:bg-black/5 transition-colors`},[(c(),o(n(Z.value),{class:`w-4 h-4`}))],8,W)]),i(`div`,G,[h(v,null,{default:_(({Component:e})=>[h(w,{name:`fade`,mode:`out-in`},{default:_(()=>[(c(),o(n(e)))]),_:2},1024)]),_:1})])])])}}}),[[`__scopeId`,`data-v-3db5d7a9`]]);export{K as default};
@@ -0,0 +1 @@
1
+ .fade-enter-active[data-v-3db5d7a9],.fade-leave-active[data-v-3db5d7a9]{transition:opacity .2s,transform .2s}.fade-enter-from[data-v-3db5d7a9],.fade-leave-to[data-v-3db5d7a9]{opacity:0;transform:translateY(10px)}
@@ -0,0 +1 @@
1
+ import{E as e,H as t,K as n,N as r,P as i,_ as a,d as o,dt as s,f as c,ft as l,j as u,k as d,m as f,n as p,p as m,pt as h,s as g,t as _,u as v,v as y,y as b}from"./createLucideIcon-CE-ry2oA.js";import{t as ee}from"./arrow-left-WEOMLXIi.js";import{t as x}from"./chevron-right-DLiDJVJl.js";import{t as S}from"./file-text-C6nzo1us.js";import{t as C}from"./folder-open-CGeIri0U.js";import{t as w}from"./folder-D9pvA6oI.js";import{t as T}from"./house-8ZvvlaEh.js";import{t as E}from"./loader-circle-uiC7GaCG.js";import{l as te,u as D}from"./index-BFE6PEIL.js";var O=_(`file-braces`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M10 12a1 1 0 0 0-1 1v1a1 1 0 0 1-1 1 1 1 0 0 1 1 1v1a1 1 0 0 0 1 1`,key:`1oajmo`}],[`path`,{d:`M14 18a1 1 0 0 0 1-1v-1a1 1 0 0 1 1-1 1 1 0 0 1-1-1v-1a1 1 0 0 0-1-1`,key:`mpwhp6`}]]),k=_(`file-code`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M10 12.5 8 15l2 2.5`,key:`1tg20x`}],[`path`,{d:`m14 12.5 2 2.5-2 2.5`,key:`yinavb`}]]),A=_(`file-cog`,[[`path`,{d:`M15 8a1 1 0 0 1-1-1V2a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8z`,key:`1ckgky`}],[`path`,{d:`M20 8v12a2 2 0 0 1-2 2h-4.182`,key:`1726p0`}],[`path`,{d:`m3.305 19.53.923-.382`,key:`ao1pio`}],[`path`,{d:`M4 10.592V4a2 2 0 0 1 2-2h8`,key:`1foop0`}],[`path`,{d:`m4.228 16.852-.924-.383`,key:`1fv9zy`}],[`path`,{d:`m5.852 15.228-.383-.923`,key:`1a9hc2`}],[`path`,{d:`m5.852 20.772-.383.924`,key:`1sh9ke`}],[`path`,{d:`m8.148 15.228.383-.923`,key:`4yu6lf`}],[`path`,{d:`m8.53 21.696-.382-.924`,key:`18b0s9`}],[`path`,{d:`m9.773 16.852.922-.383`,key:`ti6xop`}],[`path`,{d:`m9.773 19.148.922.383`,key:`rws47d`}],[`circle`,{cx:`7`,cy:`18`,r:`3`,key:`lvkj7j`}]]),j=_(`file-image`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`circle`,{cx:`10`,cy:`12`,r:`2`,key:`737tya`}],[`path`,{d:`m20 17-1.296-1.296a2.41 2.41 0 0 0-3.408 0L9 22`,key:`wt3hpn`}]]),M=_(`file`,[[`path`,{d:`M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z`,key:`1oefj6`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}]]),N={key:1,class:`w-3.5 mr-1`},P={class:`truncate`},F={key:0},I=b({__name:`FileTreeNode`,props:{node:{},depth:{},selectedFile:{}},emits:[`toggle`,`select`],setup(e,{emit:t}){let p=t;function _(e,t,n){if(t)return n?C:w;let r=e.split(`.`).pop()?.toLowerCase()||``;return[`json`,`jsonc`].includes(r)?O:[`js`,`ts`,`jsx`,`tsx`,`vue`,`svelte`,`py`,`rb`,`go`,`rs`,`java`,`c`,`cpp`,`h`].includes(r)?k:[`png`,`jpg`,`jpeg`,`gif`,`svg`,`webp`,`ico`].includes(r)?j:[`yml`,`yaml`,`toml`,`ini`,`cfg`,`conf`,`env`].includes(r)?A:S}return(t,v)=>{let b=r(`FileTreeNode`,!0);return d(),f(`div`,null,[o(`button`,{class:s([`w-full flex items-center py-1.5 text-sm hover:bg-primary/5 transition-colors`,e.selectedFile===e.node.path&&!e.node.isDir?`bg-primary/10 text-primary`:`text-textMain`]),style:l({paddingLeft:e.depth*16+12+`px`}),onClick:v[0]||=t=>e.node.isDir?p(`toggle`,e.node):p(`select`,e.node)},[e.node.isDir?(d(),c(n(x),{key:0,class:s([`w-3.5 h-3.5 mr-1 shrink-0 transition-transform`,e.node.expanded?`rotate-90`:``])},null,8,[`class`])):(d(),f(`span`,N)),(d(),c(i(_(e.node.name,e.node.isDir,e.node.expanded)),{class:s([`w-4 h-4 mr-1.5 shrink-0`,e.node.isDir?`text-primary/70`:`text-textMuted`])},null,8,[`class`])),o(`span`,P,h(e.node.name),1)],6),e.node.isDir&&e.node.expanded?(d(),f(`div`,F,[e.node.loading?(d(),f(`div`,{key:0,class:`py-1 text-textMuted text-xs flex items-center`,style:l({paddingLeft:e.depth*16+28+`px`})},[y(n(E),{class:`w-3 h-3 animate-spin mr-1`}),v[3]||=a(` 加载中... `,-1)],4)):e.node.children?(d(!0),f(g,{key:1},u(e.node.children,t=>(d(),c(b,{key:t.path,node:t,depth:e.depth+1,"selected-file":e.selectedFile,onToggle:v[1]||=e=>p(`toggle`,e),onSelect:v[2]||=e=>p(`select`,e)},null,8,[`node`,`depth`,`selected-file`]))),128)):m(``,!0)])):m(``,!0)])}}}),L={class:`h-[calc(100vh-4rem)] flex flex-col`},R={class:`flex items-center space-x-4 mb-4`},z={class:`text-xl font-bold text-textMain tracking-tight`},B={class:`flex-1 flex border border-border rounded-xl overflow-hidden bg-panel min-h-0`},V={class:`w-64 border-r border-border overflow-auto bg-base/50 shrink-0`},H={key:0,class:`flex items-center justify-center py-8 text-textMuted`},U={key:1,class:`py-8 text-center text-textMuted text-sm`},W={key:2,class:`py-2`},G={class:`flex-1 flex flex-col min-w-0`},K={key:0,class:`flex items-center px-4 py-2 border-b border-border text-sm bg-base/30`},q={class:`flex-1 overflow-auto`},J={key:0,class:`h-full flex items-center justify-center text-textMuted`},Y={class:`text-center`},X={key:1,class:`h-full flex items-center justify-center text-textMuted`},Z={key:2,class:`h-full flex items-center justify-center text-textMuted`},ne={class:`text-center`},re={key:0,class:`text-xs mt-1`},ie={key:3,class:`p-4 text-sm font-mono text-textMain leading-relaxed whitespace-pre-wrap break-all overflow-x-auto`},Q=b({__name:`FileExplorer`,setup(r){let i=te(),l=D(),_=p(),b=i.params.id,S=v(()=>_.getProjectById(b)),C=t([]),w=t(``),O=t(null),k=t(!1),A=t(!0);e(async()=>{if(await _.fetchProjects(),!S.value){l.replace(`/projects`);return}await j(``)});async function j(e){A.value=!0,C.value=(await _.fetchFiles(b,e)).map(e=>({...e,expanded:!1,children:void 0,loading:!1})),A.value=!1}async function N(e){if(e.isDir){if(e.expanded){e.expanded=!1;return}e.expanded=!0,e.children||(e.loading=!0,e.children=(await _.fetchFiles(b,e.path)).map(e=>({...e,expanded:!1,children:void 0,loading:!1})),e.loading=!1)}}async function P(e){e.isDir||(w.value=e.path,k.value=!0,O.value=null,O.value=await _.fetchFileContent(b,e.path),k.value=!1)}function F(e){return e<1024?e+` B`:e<1024*1024?(e/1024).toFixed(1)+` KB`:(e/(1024*1024)).toFixed(1)+` MB`}function Q(e){return e?e.split(`/`).map((e,t,n)=>({name:e,path:n.slice(0,t+1).join(`/`)})):[]}let $=v(()=>Q(w.value));return(e,t)=>(d(),f(`div`,L,[o(`div`,R,[o(`button`,{onClick:t[0]||=e=>n(l).push(`/projects/${n(b)}`),class:`p-2 dark:hover:bg-white/10 hover:bg-black/10 rounded-full transition-colors text-textMuted hover:text-textMain`},[y(n(ee),{class:`w-5 h-5`})]),o(`div`,null,[o(`h1`,z,[a(h(S.value?.name||`项目`)+` `,1),t[1]||=o(`span`,{class:`text-textMuted font-normal`},`/ 文件浏览`,-1)])])]),o(`div`,B,[o(`aside`,V,[A.value?(d(),f(`div`,H,[y(n(E),{class:`w-5 h-5 animate-spin mr-2`}),t[2]||=o(`span`,{class:`text-sm`},`加载中...`,-1)])):C.value.length===0?(d(),f(`div`,U,` 暂无文件 `)):(d(),f(`div`,W,[(d(!0),f(g,null,u(C.value,e=>(d(),c(I,{key:e.path,node:e,depth:0,"selected-file":w.value,onToggle:N,onSelect:P},null,8,[`node`,`selected-file`]))),128))]))]),o(`div`,G,[w.value?(d(),f(`div`,K,[y(n(T),{class:`w-3.5 h-3.5 text-textMuted mr-1`}),(d(!0),f(g,null,u($.value,(e,t)=>(d(),f(`span`,{key:e.path,class:`flex items-center`},[y(n(x),{class:`w-3 h-3 text-textMuted mx-1`}),o(`span`,{class:s(t===$.value.length-1?`text-textMain font-medium`:`text-textMuted`)},h(e.name),3)]))),128))])):m(``,!0),o(`div`,q,[w.value?k.value?(d(),f(`div`,X,[y(n(E),{class:`w-6 h-6 animate-spin`})])):O.value?.type===`binary`?(d(),f(`div`,Z,[o(`div`,ne,[y(n(M),{class:`w-12 h-12 mx-auto mb-3 opacity-30`}),t[4]||=o(`p`,{class:`text-sm`},`二进制文件,无法预览`,-1),O.value.size?(d(),f(`p`,re,`大小: `+h(F(O.value.size)),1)):m(``,!0)])])):O.value?.type===`text`?(d(),f(`pre`,ie,[o(`code`,null,h(O.value.content),1)])):m(``,!0):(d(),f(`div`,J,[o(`div`,Y,[y(n(M),{class:`w-12 h-12 mx-auto mb-3 opacity-30`}),t[3]||=o(`p`,{class:`text-sm`},`选择左侧文件查看内容`,-1)])]))])])])]))}});export{Q as default};
@@ -0,0 +1 @@
1
+ .overflow-y-auto[data-v-c3868b1e]::-webkit-scrollbar{width:10px}.overflow-y-auto[data-v-c3868b1e]::-webkit-scrollbar-track{background:#09090b}.overflow-y-auto[data-v-c3868b1e]::-webkit-scrollbar-thumb{background:#27272a;border:2px solid #09090b;border-radius:5px}.overflow-y-auto[data-v-c3868b1e]::-webkit-scrollbar-thumb:hover{background:#3f3f46}
@@ -0,0 +1,6 @@
1
+ import{B as e,D as t,E as n,H as r,K as i,R as a,_ as o,d as s,dt as c,f as l,j as u,k as d,m as f,n as ee,p,pt as m,s as h,t as g,u as _,v,w as y,y as b}from"./createLucideIcon-CE-ry2oA.js";import{n as x,r as te,t as ne}from"./rotate-ccw-YevaWXw9.js";import{t as re}from"./archive-CJ5gDBKY.js";import{t as S}from"./circle-alert-DQzM-U4P.js";import{t as ie}from"./clock-IwxBKgP4.js";import{t as C}from"./copy-C1rADgcQ.js";import{t as w}from"./refresh-cw-0yb8DZDc.js";import{g as ae,h as oe,l as se,n as ce,o as le,s as ue,t as T,u as de,v as fe}from"./index-BFE6PEIL.js";import{t as pe}from"./ConfirmDialog-CJ8lJeUc.js";var me=g(`terminal`,[[`path`,{d:`M12 19h8`,key:`baeox8`}],[`path`,{d:`m4 17 6-6-6-6`,key:`1yngyt`}]]),E={30:`#1e1e1e`,31:`#e06c75`,32:`#98c379`,33:`#e5c07b`,34:`#61afef`,35:`#c678dd`,36:`#56b6c2`,37:`#abb2bf`,90:`#5c6370`,91:`#e06c75`,92:`#98c379`,93:`#e5c07b`,94:`#61afef`,95:`#c678dd`,96:`#56b6c2`,97:`#ffffff`},D={40:`#1e1e1e`,41:`#e06c75`,42:`#98c379`,43:`#e5c07b`,44:`#61afef`,45:`#c678dd`,46:`#56b6c2`,47:`#abb2bf`},O=/\x1b\[([0-9;]*)m/g;function he(e){let t=``,n=0,r=0;e=e.replace(/&/g,`&amp;`).replace(/</g,`&lt;`).replace(/>/g,`&gt;`);let i;for(O.lastIndex=0;(i=O.exec(e))!==null;){t+=e.slice(r,i.index),r=i.index+i[0].length;let a=i[1].split(`;`).map(Number);if(a.length===0||a.length===1&&a[0]===0){for(;n>0;)t+=`</span>`,n--;continue}for(let e of a)if(e===0)for(;n>0;)t+=`</span>`,n--;else e===1?(t+=`<span style="font-weight:bold">`,n++):e===2?(t+=`<span style="opacity:0.7">`,n++):e===3?(t+=`<span style="font-style:italic">`,n++):e===4?(t+=`<span style="text-decoration:underline">`,n++):E[e]?(t+=`<span style="color:${E[e]}">`,n++):D[e]&&(t+=`<span style="background-color:${D[e]}">`,n++)}for(t+=e.slice(r);n>0;)t+=`</span>`,n--;return t}function ge(e,n){let i=r([]),o=r(``),s=null,c=!1;async function l(e){u(),i.value=[],o.value=``,c=!1,s=new AbortController;try{let t=await fetch(`/api/logs/${e}/stream`,{headers:{Authorization:`Bearer ${n.value}`},signal:s.signal});if(!t.ok||!t.body)return;let r=t.body.getReader(),a=new TextDecoder,l=``;for(;;){let{done:e,value:t}=await r.read();if(e)break;l+=a.decode(t,{stream:!0});let n=l.split(`
2
+
3
+ `);l=n.pop();for(let e of n){if(!e.trim())continue;let t=`message`,n=``;for(let r of e.split(`
4
+ `))r.startsWith(`event: `)?t=r.slice(7):r.startsWith(`data: `)&&(n=r.slice(6));if(t===`log`){let e=JSON.parse(n);c?i.value.push(e):(i.value=e.split(`
5
+ `),c=!0)}else if(t===`status`){o.value=JSON.parse(n).status,u();return}}}}catch(e){e.name!==`AbortError`&&console.error(`SSE connection error:`,e)}}function u(){s&&=(s.abort(),null)}return a(e,e=>{e?l(e):u()}),t(u),{lines:i,status:o,disconnect:u}}var _e={class:`h-full flex flex-col space-y-6 max-w-7xl mx-auto`},k={class:`flex justify-between items-center shrink-0`},A={class:`flex-1 flex flex-col lg:flex-row gap-6 min-h-0`},j={class:`w-full lg:w-1/3 bg-panel border border-border rounded-xl shadow-sm overflow-hidden flex flex-col h-[400px] lg:h-auto`},M={class:`p-4 border-b border-border bg-base/50 shrink-0 space-y-2`},N=[`value`],P={key:0,class:`flex items-center justify-between text-xs text-textMuted`},F={class:`flex-1 overflow-y-auto p-2 space-y-1`},I=[`onClick`],ve={class:`mt-0.5`},ye={class:`flex-1 min-w-0`},be={class:`flex justify-between items-center mb-1`},xe={class:`font-medium text-textMain text-sm truncate`},Se={class:`text-xs text-textMuted font-mono shrink-0`},Ce={class:`flex items-center text-xs text-textMuted gap-1.5 flex-wrap`},we=[`title`,`onClick`],Te={key:0,class:`inline-flex items-center px-1.5 py-0.5 rounded text-[10px] font-medium bg-success/10 border border-success/30 text-success`,title:`该版本为当前线上版本`},Ee={class:`flex items-center`},De={class:`flex items-center`},Oe={key:1,class:`ml-auto text-[10px] font-mono px-1 py-0 rounded bg-yellow-400/10 border border-yellow-400/30 text-yellow-400`},ke={key:0,class:`flex flex-col items-center justify-center py-10 text-textMuted text-sm`},Ae={class:`flex-1 bg-[#09090b] border border-border rounded-xl shadow-sm flex flex-col overflow-hidden h-[500px] lg:h-auto font-mono text-sm`},je={class:`h-10 bg-panel border-b border-border flex items-center px-4 shrink-0 gap-2`},Me={class:`flex-1 text-center text-textMuted text-xs font-sans truncate flex items-center justify-center gap-1.5`},Ne=[`title`],L={key:0,class:`flex items-center gap-1.5`},Pe={key:0,class:`text-[10px] font-medium px-1.5 py-0.5 rounded bg-success/10 border border-success/30 text-success`,title:`该版本为当前线上版本`},Fe=[`title`],Ie=[`title`],Le={class:`flex-1 p-4 overflow-y-auto bg-[#09090b] text-[#f4f4f5] leading-relaxed selection:bg-primary/30`},Re={key:0,class:`space-y-1 whitespace-pre-wrap break-all`},ze={class:`w-8 text-right mr-4 text-textMuted/30 select-none group-hover:text-textMuted/50 transition-colors`},Be=[`innerHTML`],Ve={key:0,class:`flex px-2 -mx-2 mt-2`},He={class:`w-8 text-right mr-4 text-textMuted/30 select-none`},Ue={key:1,class:`h-full flex flex-col items-center justify-center text-textMuted font-sans`},R=T(b({__name:`LogBoard`,setup(t){let g=ee(),b=se(),T=de(),E=ce(),D=r(``),O=r(``),R=_(()=>{let e=g.projects.map(e=>({id:e.id,name:e.name})),t=new Set(e.map(e=>e.id));for(let n of g.logs)t.has(n.projectId)||(t.add(n.projectId),e.push({id:n.projectId,name:n.projectName}));return e}),z=_(()=>{let e=D.value.trim().toLowerCase();return g.logs.filter(t=>O.value&&t.projectId!==O.value?!1:e?t.projectName.toLowerCase().includes(e)||t.id.toLowerCase().includes(e):!0)}),B=r(null),V=_(()=>B.value?.status===`running`),H=r({}),We=e=>t=>{H.value[e]=t};async function U(e){if(!e)return;let t=g.logs.find(t=>t.id===e);if(!t)return;B.value=t,await y();let n=H.value[e];n&&typeof n.scrollIntoView==`function`&&n.scrollIntoView({block:`nearest`})}n(async()=>{g.projects.length===0&&g.fetchProjects(),await g.fetchLogs();let e=typeof b.query.projectId==`string`?b.query.projectId:``;e&&(O.value=e),await U(typeof b.query.id==`string`?b.query.id:null)}),a(()=>b.query.id,async e=>{typeof e==`string`&&await U(e)}),a(()=>b.query.projectId,e=>{O.value=typeof e==`string`?e:``}),a(O,e=>{if(e===(typeof b.query.projectId==`string`?b.query.projectId:``))return;let t={...b.query};e?t.projectId=e:delete t.projectId,T.replace({query:t}),B.value&&e&&B.value.projectId!==e&&(B.value=null)});let{lines:W,status:Ge}=ge(_(()=>V.value?B.value?.id:null),_(()=>g.adminToken));a(Ge,e=>{e&&g.fetchLogs().then(()=>{if(B.value){let e=g.logs.find(e=>e.id===B.value.id);e&&(B.value=e)}})});let G=_(()=>V.value&&W.value.length>0?W.value:B.value?.output?.split(`
6
+ `)||[]),Ke=e=>{B.value=e},qe=async()=>{await g.fetchLogs(),B.value&&=z.value.find(e=>e.id===B.value.id)||null};function Je(e){return he(e)}let K=_(()=>{let e=B.value;return!e||e.status===`running`||e.triggerSource===`rollback`?!1:!!e.artifactPath}),q=_(()=>{let e=B.value;return e?e.status===`running`?`部署进行中,无法回滚`:e.triggerSource===`rollback`?`回滚记录不可再被回滚`:e.artifactPath?``:`该版本归档已被清理或过早,无法回滚`:``}),J=r(!1),Y=r(!1);function X(e){return e?e.slice(0,8):``}let Z=r(``);async function Q(e,t){if(t&&(t.stopPropagation(),t.preventDefault()),e)try{await navigator.clipboard.writeText(e),Z.value=e,E.success(`已复制部署 ID`,X(e)),setTimeout(()=>{Z.value===e&&(Z.value=``)},2e3)}catch(e){E.error(`复制失败`,e?.message||`请手动选择文本复制`)}}let Ye=_(()=>{let e={},t=[...g.logs].sort((e,t)=>{let n=new Date(e.startTime).getTime()||0;return(new Date(t.startTime).getTime()||0)-n});for(let n of t)n.status===`success`&&n.triggerSource!==`rollback`&&(e[n.projectId]||(e[n.projectId]=n.id));return e});function $(e){return Ye.value[e.projectId]===e.id}function Xe(){K.value&&(J.value=!0)}async function Ze(){let e=B.value?.id;if(e){Y.value=!0;try{let t=await g.rollbackDeployment(e);E.success(`回滚已完成`,`新部署 ${X(t.deployId)}`),J.value=!1,await g.fetchLogs();let n=g.logs.find(e=>e.id===t.deployId);if(n){B.value=n,await y();let e=H.value[n.id];e&&typeof e.scrollIntoView==`function`&&e.scrollIntoView({block:`nearest`})}}catch(e){E.error(`回滚失败`,e?.message||`未知错误`)}finally{Y.value=!1}}}return(t,n)=>(d(),f(`div`,_e,[s(`div`,k,[n[6]||=s(`div`,null,[s(`h1`,{class:`text-2xl font-bold text-textMain tracking-tight`},`部署日志`),s(`p`,{class:`text-textMuted text-sm mt-1`},`实时查看所有项目的自动化部署过程及终端输出`)],-1),s(`button`,{onClick:qe,class:`flex items-center px-4 py-2 bg-panel dark:hover:bg-white/5 hover:bg-black/5 border border-border text-textMain rounded-md transition-colors text-sm font-medium shadow-sm`},[v(i(w),{class:`w-4 h-4 mr-2`}),n[5]||=o(` 刷新 `,-1)])]),s(`div`,A,[s(`div`,j,[s(`div`,M,[e(s(`select`,{"onUpdate:modelValue":n[0]||=e=>O.value=e,class:`w-full bg-base border border-border rounded-md px-3 py-2 text-sm text-textMain focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all`},[n[7]||=s(`option`,{value:``},`全部项目`,-1),(d(!0),f(h,null,u(R.value,e=>(d(),f(`option`,{key:e.id,value:e.id},m(e.name),9,N))),128))],512),[[oe,O.value]]),e(s(`input`,{"onUpdate:modelValue":n[1]||=e=>D.value=e,type:`text`,placeholder:`搜索项目或记录 ID...`,class:`w-full bg-base border border-border rounded-md px-3 py-2 text-sm text-textMain focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all`},null,512),[[ae,D.value]]),O.value||D.value?(d(),f(`div`,P,[s(`span`,null,`已过滤 `+m(z.value.length)+` 条记录`,1),s(`button`,{onClick:n[2]||=e=>{O.value=``,D.value=``},class:`text-primary hover:underline`},`清除过滤`)])):p(``,!0)]),s(`div`,F,[(d(!0),f(h,null,u(z.value,e=>(d(),f(`div`,{key:e.id,ref_for:!0,ref:We(e.id),onClick:t=>Ke(e),class:c([`p-3 rounded-lg cursor-pointer transition-all border border-transparent flex items-start space-x-3`,B.value?.id===e.id?`bg-primary/10 border-primary/20 shadow-[inset_2px_0_0_0_#3b82f6]`:`dark:hover:bg-white/5 hover:bg-black/5`])},[s(`div`,ve,[e.status===`success`?(d(),l(i(ue),{key:0,class:`w-5 h-5 text-success`})):e.status===`failed`?(d(),l(i(le),{key:1,class:`w-5 h-5 text-danger`})):(d(),l(i(w),{key:2,class:`w-5 h-5 text-primary animate-spin`}))]),s(`div`,ye,[s(`div`,be,[s(`span`,xe,m(e.projectName),1),s(`span`,Se,m(new Date(e.startTime).toLocaleString()),1)]),s(`div`,Ce,[s(`span`,{class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded bg-base border border-border font-mono text-[10px] text-textMuted hover:text-primary hover:border-primary/40 transition-colors cursor-pointer`,title:`点击复制完整 ID: ${e.id}`,onClick:fe(t=>Q(e.id,t),[`stop`])},[Z.value===e.id?(d(),l(i(x),{key:0,class:`w-3 h-3 text-success`})):(d(),l(i(C),{key:1,class:`w-3 h-3`})),o(` `+m(X(e.id)),1)],8,we),$(e)?(d(),f(`span`,Te,`当前`)):p(``,!0),s(`span`,Ee,[v(i(me),{class:`w-3 h-3 mr-1`}),o(` `+m(e.triggerSource),1)]),s(`span`,De,[v(i(ie),{class:`w-3 h-3 mr-1`}),o(` `+m(e.duration),1)]),e.triggerSource===`rollback`?(d(),f(`span`,Oe,`RB`)):p(``,!0)])])],10,I))),128)),z.value.length===0?(d(),f(`div`,ke,[v(i(S),{class:`w-8 h-8 mb-2 opacity-50`}),n[8]||=s(`p`,null,`暂无匹配的部署记录`,-1)])):p(``,!0)])]),s(`div`,Ae,[s(`div`,je,[n[10]||=s(`div`,{class:`flex space-x-2 mr-2`},[s(`div`,{class:`w-3 h-3 rounded-full bg-danger/80`}),s(`div`,{class:`w-3 h-3 rounded-full bg-yellow-500/80`}),s(`div`,{class:`w-3 h-3 rounded-full bg-success/80`})],-1),s(`div`,Me,[B.value?(d(),f(h,{key:0},[s(`span`,null,`bash - `+m(B.value.projectName),1),s(`button`,{type:`button`,class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded bg-base border border-border font-mono text-[10px] text-textMuted hover:text-primary hover:border-primary/40 transition-colors`,title:`点击复制完整 ID: ${B.value.id}`,onClick:n[3]||=e=>Q(B.value.id,e)},[Z.value===B.value.id?(d(),l(i(x),{key:0,class:`w-3 h-3 text-success`})):(d(),l(i(C),{key:1,class:`w-3 h-3`})),o(` `+m(X(B.value.id)),1)],8,Ne)],64)):(d(),f(h,{key:1},[o(`等待选择...`)],64))]),B.value?(d(),f(`div`,L,[$(B.value)?(d(),f(`span`,Pe,`当前版本`)):p(``,!0),B.value.triggerSource===`rollback`?(d(),f(`span`,{key:1,class:`text-[10px] font-mono px-1.5 py-0.5 rounded bg-yellow-400/10 border border-yellow-400/30 text-yellow-400`,title:`rollbackOf=${B.value.rollbackOf||``}`},`rollback`,8,Fe)):p(``,!0),B.value.artifactPath?(d(),l(i(re),{key:2,class:`w-3.5 h-3.5 text-success/70`,title:`已归档,可回滚`})):(d(),l(i(te),{key:3,class:`w-3.5 h-3.5 text-textMuted/50`,title:`无归档`})),K.value?(d(),f(`button`,{key:4,onClick:Xe,class:`flex items-center px-2.5 py-1 text-[11px] font-medium bg-yellow-400/10 border border-yellow-400/30 text-yellow-400 hover:bg-yellow-400 hover:text-black rounded transition-colors`,type:`button`},[v(i(ne),{class:`w-3 h-3 mr-1`}),n[9]||=o(` 回滚到此版本 `,-1)])):q.value?(d(),f(`span`,{key:5,class:`text-[10px] text-textMuted/70 italic`,title:q.value},`不可回滚`,8,Ie)):p(``,!0)])):p(``,!0)]),s(`div`,Le,[B.value?(d(),f(`div`,Re,[(d(!0),f(h,null,u(G.value,(e,t)=>(d(),f(`div`,{key:t,class:`flex dark:hover:bg-white/5 hover:bg-black/5 px-2 -mx-2 rounded transition-colors group`},[s(`span`,ze,m(Number(t)+1),1),s(`span`,{innerHTML:Je(e)},null,8,Be)]))),128)),B.value.status===`running`?(d(),f(`div`,Ve,[s(`span`,He,m(G.value.length+1),1),n[11]||=s(`span`,{class:`w-2 h-4 bg-textMain animate-pulse inline-block align-middle`},null,-1)])):p(``,!0)])):(d(),f(`div`,Ue,[v(i(S),{class:`w-12 h-12 mb-4 opacity-50`}),n[12]||=s(`p`,null,`请选择左侧部署记录以查看详细终端输出`,-1)]))])])]),v(pe,{open:J.value,"onUpdate:open":n[4]||=e=>J.value=e,tone:`warning`,title:`确认回滚到此版本?`,message:`将以归档 ${X(B.value?.id)} 重新部署到项目 ${B.value?.projectName}。会按当前项目的 cleanMode / protectPaths 执行清理后再解压,运行时数据按保护规则保留。`,"confirm-text":`确认回滚`,"cancel-text":`取消`,loading:Y.value,onConfirm:Ze},null,8,[`open`,`message`,`loading`])]))}}),[[`__scopeId`,`data-v-c3868b1e`]]);export{R as default};
@@ -0,0 +1 @@
1
+ import{B as e,E as t,H as n,K as r,_ as i,d as a,dt as o,f as s,k as c,m as l,n as u,p as d,pt as f,t as p,v as m,y as h}from"./createLucideIcon-CE-ry2oA.js";import{g,u as _,v}from"./index-BFE6PEIL.js";var y=p(`log-in`,[[`path`,{d:`m10 17 5-5-5-5`,key:`1bsop3`}],[`path`,{d:`M15 12H3`,key:`6jk70r`}],[`path`,{d:`M15 3h4a2 2 0 0 1 2 2v14a2 2 0 0 1-2 2h-4`,key:`u53s6r`}]]),b=p(`rocket`,[[`path`,{d:`M12 15v5s3.03-.55 4-2c1.08-1.62 0-5 0-5`,key:`qeys4`}],[`path`,{d:`M4.5 16.5c-1.5 1.26-2 5-2 5s3.74-.5 5-2c.71-.84.7-2.13-.09-2.91a2.18 2.18 0 0 0-2.91-.09`,key:`u4xsad`}],[`path`,{d:`M9 12a22 22 0 0 1 2-3.95A12.88 12.88 0 0 1 22 2c0 2.72-.78 7.5-6 11a22.4 22.4 0 0 1-4 2z`,key:`676m9`}],[`path`,{d:`M9 12H4s.55-3.03 2-4c1.62-1.08 5 .05 5 .05`,key:`92ym6u`}]]),x={class:`min-h-screen bg-base flex flex-col items-center justify-center p-4`},S={class:`w-full max-w-md`},C={class:`flex flex-col items-center justify-center mb-8`},w={class:`w-16 h-16 bg-primary/10 border border-primary/20 rounded-2xl flex items-center justify-center mb-4 shadow-[0_0_30px_rgba(59,130,246,0.15)]`},T={class:`bg-panel border border-border rounded-2xl p-8 shadow-xl`},E={key:0,class:`text-danger text-xs mt-2 flex items-center`},D=[`disabled`],O={key:1,class:`animate-spin -ml-1 mr-3 h-5 w-5 text-white`,xmlns:`http://www.w3.org/2000/svg`,fill:`none`,viewBox:`0 0 24 24`},k=h({__name:`Login`,setup(p){let h=_(),k=u(),A=n(``),j=n(!1),M=n(``);t(()=>{k.adminToken&&h.replace(`/`)});let N=async()=>{if(A.value){j.value=!0,M.value=``;try{await k.login(A.value)?h.replace(`/`):M.value=`Token 校验失败,请检查配置。`}catch(e){M.value=e.message||`登录异常,请稍后重试。`}finally{j.value=!1}}};return(t,n)=>(c(),l(`div`,x,[a(`div`,S,[a(`div`,C,[a(`div`,w,[m(r(b),{class:`w-8 h-8 text-primary`})]),n[1]||=a(`h1`,{class:`text-3xl font-bold text-textMain tracking-tight`},`Kite Deploy`,-1),n[2]||=a(`p`,{class:`text-textMuted text-sm mt-2`},`云原生极简部署管理面板`,-1)]),a(`div`,T,[n[5]||=a(`h2`,{class:`text-lg font-medium text-textMain mb-6`},`管理员登录`,-1),a(`form`,{onSubmit:v(N,[`prevent`]),class:`space-y-5`},[a(`div`,null,[n[3]||=a(`label`,{class:`block text-sm font-medium text-textMuted mb-2`},`Admin Token`,-1),e(a(`input`,{"onUpdate:modelValue":n[0]||=e=>A.value=e,type:`password`,placeholder:`请输入服务端生成的 ADMIN_TOKEN`,class:o([`w-full bg-base border border-border rounded-lg px-4 py-3 text-textMain font-mono text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all placeholder:text-textMuted/50 placeholder:font-sans`,{"border-danger focus:border-danger focus:ring-danger/50":M.value}])},null,2),[[g,A.value]]),M.value?(c(),l(`p`,E,f(M.value),1)):d(``,!0)]),a(`button`,{type:`submit`,disabled:!A.value||j.value,class:`w-full flex items-center justify-center px-4 py-3 bg-primary text-white rounded-lg font-medium transition-all hover:bg-primary/90 disabled:opacity-50 disabled:cursor-not-allowed shadow-[0_0_15px_rgba(59,130,246,0.3)]`},[j.value?(c(),l(`svg`,O,[...n[4]||=[a(`circle`,{class:`opacity-25`,cx:`12`,cy:`12`,r:`10`,stroke:`currentColor`,"stroke-width":`4`},null,-1),a(`path`,{class:`opacity-75`,fill:`currentColor`,d:`M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z`},null,-1)]])):(c(),s(r(y),{key:0,class:`w-5 h-5 mr-2`})),i(` `+f(j.value?`验证中...`:`进入控制台`),1)],8,D)],32),n[6]||=a(`div`,{class:`mt-6 text-center`},[a(`p`,{class:`text-xs text-textMuted`},[i(` Token 可以在服务端启动目录的 `),a(`code`,{class:`font-mono bg-base px-1 py-0.5 rounded border border-border`},`.env.local`),i(` 文件中找到。 `)])],-1)])])]))}});export{k as default};
@@ -0,0 +1 @@
1
+ import{B as e,E as t,H as n,K as r,_ as i,d as a,dt as o,f as s,j as c,k as l,m as u,n as ee,p as d,pt as f,s as p,t as m,u as h,v as g,y as _}from"./createLucideIcon-CE-ry2oA.js";import{t as te}from"./database-CI6acTSY.js";import{t as v}from"./loader-circle-uiC7GaCG.js";import{t as ne}from"./refresh-cw-0yb8DZDc.js";import{f as y,g as re,h as ie,i as b,m as x,s as S,v as ae}from"./index-BFE6PEIL.js";import{t as oe}from"./ConfirmDialog-CJ8lJeUc.js";var C=m(`download`,[[`path`,{d:`M12 15V3`,key:`m9g1x1`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}],[`path`,{d:`m7 10 5 5 5-5`,key:`brsn70`}]]),se=m(`file-archive`,[[`path`,{d:`M13.659 22H18a2 2 0 0 0 2-2V8a2.4 2.4 0 0 0-.706-1.706l-3.588-3.588A2.4 2.4 0 0 0 14 2H6a2 2 0 0 0-2 2v11.5`,key:`4pqfef`}],[`path`,{d:`M14 2v5a1 1 0 0 0 1 1h5`,key:`wfsgrz`}],[`path`,{d:`M8 12v-1`,key:`1ej8lb`}],[`path`,{d:`M8 18v-2`,key:`qcmpov`}],[`path`,{d:`M8 7V6`,key:`1nbb54`}],[`circle`,{cx:`8`,cy:`20`,r:`2`,key:`ckkr5m`}]]),w=m(`upload`,[[`path`,{d:`M12 3v12`,key:`1x0j5s`}],[`path`,{d:`m17 8-5-5-5 5`,key:`7q97r8`}],[`path`,{d:`M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4`,key:`ih7n3h`}]]),ce={class:`max-w-5xl mx-auto space-y-6 pb-12`},le={class:`flex items-center space-x-3 mb-8`},ue={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},de={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02] flex items-center justify-between`},fe={class:`text-lg font-semibold text-textMain flex items-center`},pe=[`disabled`],me={class:`p-6 space-y-6`},he={key:0,class:`p-3 rounded-lg bg-danger/10 border border-danger/30 text-danger text-sm flex items-center`},ge={class:`flex items-center justify-between mb-2`},_e={class:`text-xs text-textMuted`},ve={class:`border border-border rounded-lg overflow-hidden`},ye={class:`w-full text-sm`},be={class:`dark:bg-white/[0.02] bg-black/[0.02] text-textMuted`},xe={class:`w-10 px-3 py-2 text-left`},T=[`checked`,`.indeterminate`],E={key:0},D={key:1},O=[`onClick`],k={class:`px-3 py-2`},A=[`checked`,`onChange`],j={class:`px-3 py-2`},M={class:`font-medium text-textMain`},Se={class:`font-mono text-xs text-textMuted`},Ce={class:`px-3 py-2`},we={key:0,class:`text-xs text-yellow-500 mt-0.5 flex items-center`},Te={class:`px-3 py-2 text-right text-textMuted`},Ee={class:`space-y-3`},De={class:`flex items-start space-x-3 cursor-pointer`},Oe={class:`flex items-start space-x-3 cursor-pointer`},ke={key:0,class:`ml-7 space-y-2`},Ae={class:`flex items-center space-x-4 text-sm`},je={class:`flex items-center space-x-2 cursor-pointer`},Me={class:`flex items-center space-x-2 cursor-pointer`},Ne=[`disabled`],Pe={class:`flex justify-end`},Fe=[`disabled`],Ie={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},Le={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02]`},Re={class:`text-lg font-semibold text-textMain flex items-center`},ze={class:`p-6 space-y-5`},Be={class:`flex items-center space-x-3`},Ve={class:`inline-flex items-center px-3 py-2 rounded-lg border border-border bg-base text-sm text-textMain cursor-pointer hover:border-primary`},He={key:0,class:`text-sm text-textMuted`},Ue={class:`grid grid-cols-1 md:grid-cols-2 gap-4`},We={class:`flex items-center space-x-2 cursor-pointer mt-2`},Ge={key:1,class:`border border-border rounded-lg p-4 bg-base space-y-3`},Ke={class:`grid grid-cols-1 md:grid-cols-3 gap-3 text-sm`},qe={class:`p-3 rounded bg-panel border border-border`},Je={class:`mt-1 text-textMain`},Ye={class:`p-3 rounded bg-panel border border-border`},Xe={class:`mt-1 text-textMain`},Ze={class:`p-3 rounded bg-panel border border-border`},Qe={class:`mt-1 text-textMain`},$e={class:`text-sm`},et={class:`text-success`},tt={key:0,class:`space-y-1 text-xs`},nt={key:0},rt={class:`text-xs text-textMuted`},it={class:`flex justify-end`},at=[`disabled`],N=_({__name:`Migration`,setup(m){let _=ee(),N=n([]),P=n(!1),F=n(``),I=n([]),L=n({includeArtifacts:!0,includeDeployments:!0,deploymentMode:`all`,deploymentLimitPerProject:50}),R=n(!1),z=n(``),B=n(`success`),V=n(null),H=n(`skip-existing`),U=n(!0),W=n(!1),G=n(``),K=n(`success`),q=n(null),J=h(()=>N.value.length>0&&I.value.length===N.value.length),ot=h(()=>I.value.length>0&&I.value.length<N.value.length);function st(){J.value?I.value=[]:I.value=N.value.map(e=>e.id)}function Y(e){let t=I.value.indexOf(e);t>=0?I.value.splice(t,1):I.value.push(e)}async function X(){P.value=!0,F.value=``;try{let e=await fetch(`/api/migration/projects`,{headers:{Authorization:`Bearer ${_.adminToken}`}}),t=await e.json();if(!e.ok)throw Error(t.error||`获取项目失败`);N.value=t.projects||[],I.value=N.value.map(e=>e.id)}catch(e){F.value=e.message}finally{P.value=!1}}async function Z(){if(z.value=``,I.value.length===0){z.value=`请至少选择一个项目`,B.value=`error`;return}R.value=!0;try{let e={projectIds:I.value,includeArtifacts:L.value.includeArtifacts,includeDeployments:L.value.includeDeployments,deploymentLimitPerProject:L.value.includeDeployments&&L.value.deploymentMode===`limit`?Math.max(0,Math.floor(Number(L.value.deploymentLimitPerProject)||0)):0},t=await fetch(`/api/migration/export`,{method:`POST`,headers:{Authorization:`Bearer ${_.adminToken}`,"Content-Type":`application/json`},body:JSON.stringify(e)});if(!t.ok){let e=`HTTP ${t.status}`;try{e=(await t.json()).error||e}catch{}throw Error(e)}let n=t.headers.get(`Content-Disposition`)||``,r=/filename="?([^"]+)"?/.exec(n),i=r?r[1]:`kite-export-${Date.now()}.zip`,a=await t.blob(),o=URL.createObjectURL(a),s=document.createElement(`a`);s.href=o,s.download=i,document.body.appendChild(s),s.click(),document.body.removeChild(s),URL.revokeObjectURL(o),z.value=`导出成功:${i}(${(a.size/1024).toFixed(1)} KB)`,B.value=`success`}catch(e){z.value=`导出失败:${e.message}`,B.value=`error`}finally{R.value=!1}}function ct(e){V.value=e.target.files?.[0]||null,G.value=``,q.value=null}async function lt(){if(G.value=``,q.value=null,!V.value){G.value=`请选择要导入的 zip 文件`,K.value=`error`;return}if(H.value===`overwrite`){Q.value=!0;return}await $()}let Q=n(!1);async function ut(){Q.value=!1,await $()}async function $(){if(V.value){W.value=!0;try{let e=new FormData;e.append(`file`,V.value),e.append(`strategy`,H.value),e.append(`restoreArtifacts`,U.value?`true`:`false`);let t={Authorization:`Bearer ${_.adminToken}`};H.value===`overwrite`&&(t[`X-Confirm-Overwrite`]=`yes`);let n=await fetch(`/api/migration/import`,{method:`POST`,headers:t,body:e}),r=await n.json();if(!n.ok)throw Error(r.error||`HTTP ${n.status}`);q.value=r.summary,G.value=`导入完成`,K.value=`success`,await X()}catch(e){G.value=`导入失败:${e.message}`,K.value=`error`}finally{W.value=!1}}}return t(X),(t,n)=>(l(),u(`div`,ce,[a(`div`,le,[g(r(te),{class:`w-7 h-7 text-primary`}),n[9]||=a(`h1`,{class:`text-2xl font-bold text-textMain tracking-tight`},`数据迁移`,-1)]),a(`div`,ue,[a(`div`,de,[a(`div`,null,[a(`h2`,fe,[g(r(C),{class:`w-5 h-5 mr-2 text-primary`}),n[10]||=i(` 导出 `,-1)]),n[11]||=a(`p`,{class:`text-sm text-textMuted mt-1`},`选择项目并下载 zip 归档,可用于备份或迁移到另一台机器。`,-1)]),a(`button`,{onClick:X,class:`flex items-center text-sm text-textMuted hover:text-primary transition-colors`,disabled:P.value},[g(r(ne),{class:o([`w-4 h-4 mr-1`,P.value?`animate-spin`:``])},null,8,[`class`]),n[12]||=i(` 刷新 `,-1)],8,pe)]),a(`div`,me,[F.value?(l(),u(`div`,he,[g(r(b),{class:`w-4 h-4 mr-2`}),i(f(F.value),1)])):d(``,!0),a(`div`,null,[a(`div`,ge,[n[13]||=a(`h3`,{class:`text-sm font-medium text-textMain`},`项目列表`,-1),a(`span`,_e,`已选 `+f(I.value.length)+` / `+f(N.value.length),1)]),a(`div`,ve,[a(`table`,ye,[a(`thead`,be,[a(`tr`,null,[a(`th`,xe,[a(`input`,{type:`checkbox`,checked:J.value,".indeterminate":ot.value,onChange:st,class:`rounded border-border`},null,40,T)]),n[14]||=a(`th`,{class:`px-3 py-2 text-left font-medium`},`项目名 / ID`,-1),n[15]||=a(`th`,{class:`px-3 py-2 text-left font-medium`},`部署路径`,-1),n[16]||=a(`th`,{class:`px-3 py-2 text-right font-medium`},`部署日志`,-1)])]),a(`tbody`,null,[P.value?(l(),u(`tr`,E,[...n[17]||=[a(`td`,{colspan:`4`,class:`px-3 py-6 text-center text-textMuted`},`加载中...`,-1)]])):N.value.length===0?(l(),u(`tr`,D,[...n[18]||=[a(`td`,{colspan:`4`,class:`px-3 py-6 text-center text-textMuted`},`暂无项目`,-1)]])):d(``,!0),(l(!0),u(p,null,c(N.value,e=>(l(),u(`tr`,{key:e.id,class:`border-t border-border dark:hover:bg-white/[0.02] hover:bg-black/[0.02] cursor-pointer`,onClick:t=>Y(e.id)},[a(`td`,k,[a(`input`,{type:`checkbox`,checked:I.value.includes(e.id),onClick:n[0]||=ae(()=>{},[`stop`]),onChange:t=>Y(e.id),class:`rounded border-border`},null,40,A)]),a(`td`,j,[a(`div`,M,f(e.name),1),a(`div`,Se,f(e.id),1)]),a(`td`,Ce,[a(`div`,{class:o([`font-mono text-xs break-all`,e.deployPathExists?`text-textMain`:`text-textMuted`])},f(e.deployPath),3),e.deployPathExists?d(``,!0):(l(),u(`div`,we,[g(r(b),{class:`w-3 h-3 mr-1`}),n[19]||=i(`路径不存在,artifacts 会被跳过 `,-1)]))]),a(`td`,Te,f(e.deploymentCount),1)],8,O))),128))])])])]),a(`div`,Ee,[n[25]||=a(`h3`,{class:`text-sm font-medium text-textMain`},`导出选项`,-1),a(`label`,De,[e(a(`input`,{type:`checkbox`,"onUpdate:modelValue":n[1]||=e=>L.value.includeArtifacts=e,class:`mt-0.5 rounded border-border`},null,512),[[y,L.value.includeArtifacts]]),n[20]||=a(`div`,null,[a(`div`,{class:`text-sm text-textMain`},`包含部署 artifacts(项目 deployPath 内的文件)`),a(`div`,{class:`text-xs text-textMuted`},`关闭后仅导出数据库元数据与部署日志,体积更小。`)],-1)]),a(`label`,Oe,[e(a(`input`,{type:`checkbox`,"onUpdate:modelValue":n[2]||=e=>L.value.includeDeployments=e,class:`mt-0.5 rounded border-border`},null,512),[[y,L.value.includeDeployments]]),n[21]||=a(`div`,null,[a(`div`,{class:`text-sm text-textMain`},`包含部署日志`),a(`div`,{class:`text-xs text-textMuted`},`默认导出所有日志以保证信息完全一致。`)],-1)]),L.value.includeDeployments?(l(),u(`div`,ke,[a(`div`,Ae,[a(`label`,je,[e(a(`input`,{type:`radio`,value:`all`,"onUpdate:modelValue":n[3]||=e=>L.value.deploymentMode=e},null,512),[[x,L.value.deploymentMode]]),n[22]||=a(`span`,{class:`text-textMain`},`保留全部`,-1)]),a(`label`,Me,[e(a(`input`,{type:`radio`,value:`limit`,"onUpdate:modelValue":n[4]||=e=>L.value.deploymentMode=e},null,512),[[x,L.value.deploymentMode]]),n[23]||=a(`span`,{class:`text-textMain`},`每项目最近`,-1),e(a(`input`,{type:`number`,"onUpdate:modelValue":n[5]||=e=>L.value.deploymentLimitPerProject=e,min:`1`,class:`w-20 px-2 py-1 rounded bg-base border border-border text-sm text-textMain`,disabled:L.value.deploymentMode!==`limit`},null,8,Ne),[[re,L.value.deploymentLimitPerProject,void 0,{number:!0}]]),n[24]||=a(`span`,{class:`text-textMuted`},`条`,-1)])])])):d(``,!0)]),z.value?(l(),u(`div`,{key:1,class:o([`p-3 rounded-lg border text-sm flex items-center`,B.value===`success`?`bg-success/10 border-success/30 text-success`:`bg-danger/10 border-danger/30 text-danger`])},[B.value===`success`?(l(),s(r(S),{key:0,class:`w-4 h-4 mr-2`})):(l(),s(r(b),{key:1,class:`w-4 h-4 mr-2`})),i(` `+f(z.value),1)],2)):d(``,!0),a(`div`,Pe,[a(`button`,{onClick:Z,disabled:R.value||I.value.length===0,class:`inline-flex items-center px-4 py-2 rounded-lg bg-primary text-white text-sm font-medium disabled:opacity-50 hover:opacity-90 transition-opacity`},[R.value?(l(),s(r(v),{key:0,class:`w-4 h-4 mr-2 animate-spin`})):(l(),s(r(C),{key:1,class:`w-4 h-4 mr-2`})),i(` `+f(R.value?`导出中...`:`导出选中项目`),1)],8,Fe)])])]),a(`div`,Ie,[a(`div`,Le,[a(`h2`,Re,[g(r(w),{class:`w-5 h-5 mr-2 text-primary`}),n[26]||=i(` 导入 `,-1)]),n[27]||=a(`p`,{class:`text-sm text-textMuted mt-1`},`从 kite-export-*.zip 还原项目、设置、部署日志与 artifacts。`,-1)]),a(`div`,ze,[a(`div`,null,[n[29]||=a(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`选择文件`,-1),a(`div`,Be,[a(`label`,Ve,[g(r(se),{class:`w-4 h-4 mr-2`}),n[28]||=i(` 选择 zip 文件 `,-1),a(`input`,{type:`file`,accept:`.zip,application/zip`,onChange:ct,class:`hidden`},null,32)]),V.value?(l(),u(`span`,He,f(V.value.name)+` (`+f((V.value.size/1024).toFixed(1))+` KB) `,1)):d(``,!0)])]),a(`div`,Ue,[a(`div`,null,[n[31]||=a(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`冲突策略`,-1),e(a(`select`,{"onUpdate:modelValue":n[6]||=e=>H.value=e,class:`w-full px-3 py-2 rounded-lg bg-base border border-border text-sm text-textMain`},[...n[30]||=[a(`option`,{value:`skip-existing`},`skip-existing(默认,跳过已存在)`,-1),a(`option`,{value:`merge`},`merge(仅插入新数据)`,-1),a(`option`,{value:`overwrite`},`overwrite(覆盖同 ID 数据)`,-1)]],512),[[ie,H.value]]),n[32]||=a(`p`,{class:`text-xs text-textMuted mt-1`},`overwrite 会覆盖已有项目/设置/部署日志,请谨慎使用。`,-1)]),a(`div`,null,[n[34]||=a(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`Artifacts 还原`,-1),a(`label`,We,[e(a(`input`,{type:`checkbox`,"onUpdate:modelValue":n[7]||=e=>U.value=e,class:`rounded border-border`},null,512),[[y,U.value]]),n[33]||=a(`span`,{class:`text-sm text-textMain`},`还原 artifacts 到项目 deployPath`,-1)]),n[35]||=a(`p`,{class:`text-xs text-textMuted mt-1`},`关闭则仅恢复数据库内容,不解压 artifacts。`,-1)])]),G.value?(l(),u(`div`,{key:0,class:o([`p-3 rounded-lg border text-sm flex items-center`,K.value===`success`?`bg-success/10 border-success/30 text-success`:`bg-danger/10 border-danger/30 text-danger`])},[K.value===`success`?(l(),s(r(S),{key:0,class:`w-4 h-4 mr-2`})):(l(),s(r(b),{key:1,class:`w-4 h-4 mr-2`})),i(` `+f(G.value),1)],2)):d(``,!0),q.value?(l(),u(`div`,Ge,[n[41]||=a(`div`,{class:`text-sm font-medium text-textMain`},`导入摘要`,-1),a(`div`,Ke,[a(`div`,qe,[n[36]||=a(`div`,{class:`text-textMuted text-xs`},`项目`,-1),a(`div`,Je,` 新增 `+f(q.value.projects.inserted)+` / 更新 `+f(q.value.projects.updated)+` / 跳过 `+f(q.value.projects.skipped),1)]),a(`div`,Ye,[n[37]||=a(`div`,{class:`text-textMuted text-xs`},`设置`,-1),a(`div`,Xe,` 新增 `+f(q.value.settings.inserted)+` / 更新 `+f(q.value.settings.updated)+` / 跳过 `+f(q.value.settings.skipped),1)]),a(`div`,Ze,[n[38]||=a(`div`,{class:`text-textMuted text-xs`},`部署日志`,-1),a(`div`,Qe,` 新增 `+f(q.value.deployments.inserted)+` / 更新 `+f(q.value.deployments.updated)+` / 跳过 `+f(q.value.deployments.skipped),1)])]),a(`div`,$e,[n[39]||=a(`span`,{class:`text-textMuted`},`Artifacts:`,-1),a(`span`,et,`成功 `+f(q.value.artifacts.ok),1),n[40]||=a(`span`,{class:`text-textMuted mx-1`},`/`,-1),a(`span`,{class:o(q.value.artifacts.warnings>0?`text-yellow-500`:`text-textMuted`)},` 警告 `+f(q.value.artifacts.warnings),3)]),q.value.artifacts.items.length>0?(l(),u(`div`,tt,[(l(!0),u(p,null,c(q.value.artifacts.items,e=>(l(),u(`div`,{key:e.projectId,class:o([`font-mono`,e.status===`ok`?`text-textMuted`:`text-yellow-500`])},[i(` [`+f(e.status)+`] `+f(e.projectId),1),e.message?(l(),u(`span`,nt,` — `+f(e.message),1)):d(``,!0)],2))),128))])):d(``,!0),a(`div`,rt,` 来自 schemaVersion=`+f(q.value.manifest.schemaVersion)+`,导出时间 `+f(q.value.manifest.exportedAt)+`,源 kiteVersion=`+f(q.value.manifest.kiteVersion),1)])):d(``,!0),a(`div`,it,[a(`button`,{onClick:lt,disabled:W.value||!V.value,class:`inline-flex items-center px-4 py-2 rounded-lg bg-primary text-white text-sm font-medium disabled:opacity-50 hover:opacity-90 transition-opacity`},[W.value?(l(),s(r(v),{key:0,class:`w-4 h-4 mr-2 animate-spin`})):(l(),s(r(w),{key:1,class:`w-4 h-4 mr-2`})),i(` `+f(W.value?`导入中...`:`开始导入`),1)],8,at)])])]),g(oe,{open:Q.value,"onUpdate:open":n[8]||=e=>Q.value=e,tone:`danger`,title:`确认使用 Overwrite 策略导入?`,message:`Overwrite 策略会覆盖同 ID 的现有项目、设置与部署日志。此操作不可撤销,请确认你已备份当前数据。`,"confirm-text":`确认覆盖导入`,"cancel-text":`取消`,loading:W.value,onConfirm:ut},null,8,[`open`,`loading`])]))}});export{N as default};
@@ -0,0 +1 @@
1
+ import{B as e,E as t,H as n,K as r,N as i,R as a,_ as o,d as s,dt as c,f as l,ft as ee,g as u,j as d,k as f,m as p,n as m,p as h,pt as g,s as _,t as v,u as y,v as b,y as x,z as te}from"./createLucideIcon-CE-ry2oA.js";import{n as ne,r as S,t as re}from"./rotate-ccw-YevaWXw9.js";import{t as ie}from"./archive-CJ5gDBKY.js";import{t as ae}from"./arrow-left-WEOMLXIi.js";import{t as C}from"./chevron-right-DLiDJVJl.js";import{t as oe}from"./copy-C1rADgcQ.js";import{t as se}from"./eye-Cas8HsmK.js";import{t as w}from"./file-text-C6nzo1us.js";import{t as ce}from"./folder-open-CGeIri0U.js";import{t as T}from"./folder-D9pvA6oI.js";import{n as le,t as ue}from"./save-BuzCP3T1.js";import{t as E}from"./loader-circle-uiC7GaCG.js";import{t as de}from"./plus-CfMi1Jv1.js";import{t as fe}from"./refresh-cw-0yb8DZDc.js";import{t as pe}from"./scroll-text-BIYMFNN5.js";import{t as me}from"./square-terminal-DT6aoXm0.js";import{t as he}from"./trash-2-BWqtqkeW.js";import{_ as ge,d as D,g as O,h as _e,i as ve,l as ye,m as be,n as xe,o as Se,r as k,s as Ce,t as A,u as we,v as j}from"./index-BFE6PEIL.js";import{t as Te}from"./ConfirmDialog-CJ8lJeUc.js";var M=v(`chevron-down`,[[`path`,{d:`m6 9 6 6 6-6`,key:`qrunsl`}]]),Ee=v(`history`,[[`path`,{d:`M3 12a9 9 0 1 0 9-9 9.75 9.75 0 0 0-6.74 2.74L3 8`,key:`1357e3`}],[`path`,{d:`M3 3v5h5`,key:`1xhq8a`}],[`path`,{d:`M12 7v5l4 2`,key:`1fdv2h`}]]),De=v(`shield-alert`,[[`path`,{d:`M20 13c0 5-3.5 7.5-7.66 8.95a1 1 0 0 1-.67-.01C7.5 20.5 4 18 4 13V6a1 1 0 0 1 1-1c2 0 4.5-1.2 6.24-2.72a1.17 1.17 0 0 1 1.52 0C14.51 3.81 17 5 19 5a1 1 0 0 1 1 1z`,key:`oel41y`}],[`path`,{d:`M12 8v4`,key:`1got3b`}],[`path`,{d:`M12 16h.01`,key:`1drbdi`}]]),Oe=v(`shield-check`,[[`path`,{d:`M20 13c0 5-3.5 7.5-7.66 8.95a1 1 0 0 1-.67-.01C7.5 20.5 4 18 4 13V6a1 1 0 0 1 1-1c2 0 4.5-1.2 6.24-2.72a1.17 1.17 0 0 1 1.52 0C14.51 3.81 17 5 19 5a1 1 0 0 1 1 1z`,key:`oel41y`}],[`path`,{d:`m9 12 2 2 4-4`,key:`dzmm74`}]]),ke=v(`shield`,[[`path`,{d:`M20 13c0 5-3.5 7.5-7.66 8.95a1 1 0 0 1-.67-.01C7.5 20.5 4 18 4 13V6a1 1 0 0 1 1-1c2 0 4.5-1.2 6.24-2.72a1.17 1.17 0 0 1 1.52 0C14.51 3.81 17 5 19 5a1 1 0 0 1 1 1z`,key:`oel41y`}]]),N={class:`select-none`},P={key:1,class:`w-4 h-4 inline-block`},F={class:`text-textMuted shrink-0 font-mono`},I=x({__name:`DeleteTreeView`,props:{node:{},depth:{},defaultExpand:{type:Boolean}},setup(e){let t=e,a=y(()=>t.depth??0),o=n(t.defaultExpand??a.value<1),u=y(()=>!!t.node.children&&t.node.children.length>0),m=e=>e<1024?`${e} B`:e<1024*1024?`${(e/1024).toFixed(1)} KB`:e<1024*1024*1024?`${(e/1024/1024).toFixed(2)} MB`:`${(e/1024/1024/1024).toFixed(2)} GB`,v=y(()=>t.node.type===`dir`?u.value?t.node.willDelete?`text-danger`:`text-textMain`:`text-textMuted`:t.node.willDelete?`text-danger`:`text-success`);return(t,n)=>{let y=i(`DeleteTreeView`,!0);return f(),p(`div`,N,[s(`div`,{class:`flex items-center gap-1.5 py-0.5 px-1 rounded text-xs hover:bg-white/[0.03] dark:hover:bg-white/[0.04]`,style:ee({paddingLeft:`${a.value*14+4}px`})},[u.value?(f(),p(`button`,{key:0,onClick:n[0]||=e=>o.value=!o.value,class:`w-4 h-4 flex items-center justify-center text-textMuted hover:text-textMain`,type:`button`},[o.value?(f(),l(r(M),{key:0,class:`w-3 h-3`})):(f(),l(r(C),{key:1,class:`w-3 h-3`}))])):(f(),p(`span`,P)),e.node.type===`dir`?(f(),l(r(T),{key:2,class:c([`w-3.5 h-3.5 shrink-0`,v.value])},null,8,[`class`])):(f(),l(r(w),{key:3,class:c([`w-3.5 h-3.5 shrink-0`,v.value])},null,8,[`class`])),s(`span`,{class:c([`font-mono truncate flex-1`,v.value])},g(e.node.name||`/`),3),s(`span`,F,g(m(e.node.size)),1),e.node.willDelete&&e.node.type===`file`?(f(),l(r(he),{key:4,class:`w-3 h-3 text-danger/80 shrink-0`})):!e.node.willDelete&&e.node.type===`file`?(f(),l(r(Oe),{key:5,class:`w-3 h-3 text-success/80 shrink-0`})):h(``,!0)],4),o.value&&u.value?(f(!0),p(_,{key:0},d(e.node.children,e=>(f(),l(y,{key:e.path,node:e,depth:a.value+1},null,8,[`node`,`depth`]))),128)):h(``,!0)])}}}),L={class:`flex-1 min-w-0`},R={class:`text-base font-semibold text-textMain`},z={class:`font-mono`},B={key:0,class:`mx-6 mt-4 p-3 rounded-md bg-danger/10 border border-danger/30 flex items-start gap-2`},V={class:`px-6 py-4 grid grid-cols-3 gap-3 border-b border-border`},H={class:`bg-base border border-border rounded-md p-3`},U={class:`text-lg font-bold text-danger font-mono`},W={class:`text-[10px] text-textMuted`},G={class:`bg-base border border-border rounded-md p-3`},K={class:`text-lg font-bold text-success font-mono`},q={class:`bg-base border border-border rounded-md p-3`},J={class:`text-lg font-bold text-textMain font-mono`},Ae={key:0,class:`text-[10px] text-yellow-400`},je={key:1,class:`text-[10px] text-textMuted`},Me={key:1,class:`px-6 py-3 border-b border-border flex items-center gap-2 flex-wrap`},Ne={class:`flex-1 overflow-y-auto px-4 py-3 min-h-[200px]`},Pe={key:0,class:`flex items-center justify-center py-10 text-textMuted`},Fe={key:1,class:`text-sm text-danger py-6 text-center`},Ie={key:2,class:`text-textMuted text-sm py-6 text-center`},Le={class:`px-6 py-3 border-t border-border flex items-center justify-between`},Re={key:0,class:`text-[11px] text-textMuted`},ze={key:1},Be=A(x({__name:`CleanPreviewDialog`,props:{open:{type:Boolean},loading:{type:Boolean},error:{},preview:{},mode:{},protectPaths:{}},emits:[`update:open`],setup(e,{emit:t}){let i=e,ee=t,u=y(()=>i.mode===`clean-all`),m=y(()=>i.preview?.summary),v=y(()=>i.preview?.tree),x=e=>e==null?`-`:e<1024?`${e} B`:e<1024*1024?`${(e/1024).toFixed(1)} KB`:`${(e/1024/1024).toFixed(2)} MB`;function ne(){ee(`update:open`,!1)}let S=n([]);return a(()=>i.protectPaths,e=>{S.value=[...e||[]]},{immediate:!0}),(t,n)=>(f(),l(D,{name:`fade`},{default:te(()=>[e.open?(f(),p(`div`,{key:0,class:`fixed inset-0 z-[60] flex items-center justify-center bg-black/60 backdrop-blur-sm`,onClick:j(ne,[`self`])},[s(`div`,{class:c([`bg-panel border rounded-xl w-full max-w-2xl shadow-2xl flex flex-col max-h-[85vh]`,u.value?`border-danger/40`:`border-yellow-400/30`])},[s(`div`,{class:c([`px-6 py-4 border-b flex items-start gap-3`,u.value?`border-danger/30`:`border-border`])},[s(`div`,{class:c([`w-10 h-10 rounded-lg flex items-center justify-center shrink-0`,u.value?`bg-danger/10`:`bg-yellow-400/10`])},[u.value?(f(),l(r(De),{key:0,class:`w-5 h-5 text-danger`})):(f(),l(r(se),{key:1,class:`w-5 h-5 text-yellow-400`}))],2),s(`div`,L,[s(`h3`,R,[n[0]||=o(` 清理预览 — `,-1),s(`span`,z,g(e.mode),1),n[1]||=s(`span`,{class:`ml-2 px-1.5 py-0.5 text-[10px] rounded bg-base border border-border text-textMuted align-middle font-mono`},`DRY-RUN`,-1)]),n[2]||=s(`p`,{class:`text-xs text-textMuted mt-1`},` 仅模拟,不会真正删除文件。下一次部署时会按"已保存"的策略自动执行清理。 `,-1)]),s(`button`,{onClick:ne,class:`text-textMuted hover:text-textMain p-1 rounded`,type:`button`},[b(r(k),{class:`w-4 h-4`})])],2),u.value?(f(),p(`div`,B,[b(r(ve),{class:`w-4 h-4 text-danger shrink-0 mt-0.5`}),n[3]||=s(`p`,{class:`text-xs text-danger leading-relaxed`},[s(`strong`,null,`clean-all`),o(` 会清空部署目录下除 `),s(`code`,{class:`font-mono bg-base px-1 rounded`},`.kite-*`),o(` 之外的`),s(`strong`,null,`全部内容`),o(`,请仔细确认 protectPaths 与预览结果。 `)],-1)])):h(``,!0),s(`div`,V,[s(`div`,H,[n[4]||=s(`p`,{class:`text-[10px] text-textMuted uppercase tracking-wider`},`将删除`,-1),s(`p`,U,g(m.value?.deleteFiles??`-`),1),s(`p`,W,g(x(m.value?.deleteBytes)),1)]),s(`div`,G,[n[5]||=s(`p`,{class:`text-[10px] text-textMuted uppercase tracking-wider`},`将保留`,-1),s(`p`,K,g(m.value?.protectFiles??`-`),1),n[6]||=s(`p`,{class:`text-[10px] text-textMuted`},`由 protectPaths 命中`,-1)]),s(`div`,q,[n[7]||=s(`p`,{class:`text-[10px] text-textMuted uppercase tracking-wider`},`总计文件`,-1),s(`p`,J,g(m.value?.totalFiles??`-`),1),m.value?.truncated?(f(),p(`p`,Ae,`已截断 10000 行`)):(f(),p(`p`,je,`完整扫描`))])]),S.value.length?(f(),p(`div`,Me,[n[8]||=s(`span`,{class:`text-xs text-textMuted shrink-0`},`protectPaths:`,-1),(f(!0),p(_,null,d(S.value,e=>(f(),p(`span`,{key:e,class:`text-[11px] px-2 py-0.5 rounded bg-success/10 border border-success/30 text-success font-mono`},g(e),1))),128))])):h(``,!0),s(`div`,Ne,[e.loading?(f(),p(`div`,Pe,[b(r(E),{class:`w-4 h-4 mr-2 animate-spin`}),n[9]||=o(` 正在扫描部署目录... `,-1)])):e.error?(f(),p(`div`,Fe,g(e.error),1)):!v.value||!v.value.children?.length?(f(),p(`div`,Ie,` 部署目录为空,无需清理 `)):(f(),l(I,{key:3,node:v.value,depth:0,"default-expand":!0},null,8,[`node`]))]),s(`div`,Le,[e.preview?.cached?(f(),p(`span`,Re,`使用 30s 内缓存结果`)):(f(),p(`span`,ze)),s(`button`,{onClick:ne,class:`px-4 py-2 text-sm font-medium text-textMuted hover:text-textMain dark:hover:bg-white/5 hover:bg-black/5 rounded-md transition-colors`,type:`button`},`关闭`)])],2)])):h(``,!0)]),_:1}))}}),[[`__scopeId`,`data-v-e5333666`]]),Ve={key:0,class:`max-w-4xl mx-auto space-y-6 pb-12`},He={class:`flex items-center space-x-4 mb-8`},Ue={class:`flex items-center space-x-3`},We={class:`text-2xl font-bold text-textMain tracking-tight`},Ge={class:`text-sm text-textMuted mt-1 font-mono`},Ke={class:`grid grid-cols-1 gap-8`},qe={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},Je={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02]`},Ye={class:`text-lg font-semibold text-textMain flex items-center`},Xe={class:`p-6`},Ze={class:`flex flex-col sm:flex-row items-stretch sm:items-center space-y-4 sm:space-y-0 sm:space-x-4`},Qe={class:`relative flex-1`},$e=[`type`,`value`],et={class:`flex items-center space-x-2`},tt=[`disabled`],nt={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},rt={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02] flex items-center justify-between`},it={class:`text-lg font-semibold text-textMain flex items-center`},at={class:`flex items-center gap-2`},ot=[`disabled`],st={class:`p-6`},ct={key:0,class:`py-10 text-center text-sm text-textMuted`},lt={key:1,class:`py-10 text-center text-sm text-textMuted`},ut={key:2,class:`divide-y divide-border`},dt=[`onClick`],ft={class:`flex items-center gap-2`},pt={key:0,class:`inline-flex items-center px-1.5 py-0.5 rounded text-[10px] font-medium bg-warning/10 border border-warning/20 text-warning`,title:`该部署是一次回滚`},mt=[`title`,`onClick`,`onKeydown`],ht={key:3,class:`inline-flex items-center px-1.5 py-0.5 rounded text-[10px] font-medium bg-success/10 border border-success/30 text-success`,title:`该版本为当前线上版本`},gt={class:`text-xs text-textMuted truncate`},_t={key:4,class:`text-xs text-textMuted`},vt=[`onClick`],yt=[`title`],bt={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},xt={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02]`},St={class:`text-lg font-semibold text-textMain flex items-center`},Ct={class:`p-6 space-y-5`},wt={class:`rounded-lg border border-primary/20 bg-primary/5 p-4`},Tt={class:`flex items-center gap-3`},Et={class:`text-xs text-textMuted mt-2`},Dt={class:`font-mono text-textMain`},Ot={class:`rounded-lg border border-border bg-base p-4`},kt={class:`flex items-center gap-2`},At={class:`rounded-lg border border-border bg-base p-4`},jt={class:`flex items-center gap-2 mb-4`},Mt={class:`flex-1 text-xs text-success font-mono break-all`},Nt={class:`text-xs text-textMuted mb-3`},Pt={class:`font-mono text-textMain bg-panel px-1 py-0.5 rounded border border-border`},Ft={class:`text-xs text-success font-mono whitespace-pre-wrap overflow-x-auto bg-panel rounded-md p-3 border border-border mb-3`},It={class:`space-y-2 text-xs text-textMuted`},Lt={class:`pl-3 space-y-1`},Rt={class:`text-textMuted w-24 shrink-0`},zt={class:`text-success font-mono`},Bt={class:`grid grid-cols-1 md:grid-cols-2 gap-4`},Vt={class:`rounded-lg border border-border bg-base p-4`},Ht={class:`flex items-center gap-2`},Ut={class:`flex-1 text-xs text-success font-mono break-all`},Wt={class:`rounded-lg border border-border bg-base p-4 space-y-3`},Gt={class:`flex items-center gap-2`},Kt={class:`flex-1 text-xs text-success font-mono break-all`},qt={class:`flex items-center gap-2`},Jt={class:`flex-1 text-xs text-success font-mono break-all`},Yt={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},Xt={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02]`},Zt={class:`text-lg font-semibold text-textMain flex items-center`},Qt={class:`p-6 space-y-6`},$t=[`value`],en={class:`relative`},tn={class:`relative`},nn={class:`pt-4 border-t border-border flex justify-end`},rn={class:`bg-panel border border-border rounded-xl shadow-sm overflow-hidden`},an={class:`px-6 py-5 border-b border-border dark:bg-white/[0.02] bg-black/[0.02]`},on={class:`text-lg font-semibold text-textMain flex items-center`},sn={class:`p-6 space-y-5`},cn={class:`grid grid-cols-1 md:grid-cols-3 gap-3`},ln=[`value`],un={class:`text-xs text-textMuted mt-1 leading-relaxed`},dn={key:0},fn={class:`flex gap-2 mb-3`},pn=[`onKeydown`],mn={key:0,class:`flex flex-wrap gap-2`},hn=[`onClick`],gn={key:1,class:`text-xs text-textMuted italic`},_n={key:1,class:`p-3 rounded-md bg-danger/10 border border-danger/30 flex items-start gap-2`},vn={class:`pt-3 border-t border-border flex items-center justify-end gap-3`},yn=[`disabled`],bn={class:`bg-panel border border-danger/20 rounded-xl shadow-sm overflow-hidden`},xn={class:`px-6 py-4`},Sn={class:`text-danger font-medium flex items-center`},Cn={class:`bg-panel border border-danger/30 rounded-xl w-full max-w-lg p-6 shadow-2xl`},wn={class:`flex items-start space-x-3 mb-5`},Tn={class:`p-2 rounded-lg bg-danger/10 border border-danger/20 shrink-0`},En={class:`flex-1 min-w-0`},Dn={class:`text-sm text-textMuted mt-1`},On={class:`font-mono text-textMain`},kn={class:`font-mono text-textMuted`},An={class:`space-y-3 mb-5`},jn={class:`bg-danger/5 border border-danger/20 rounded-lg p-3`},Mn={class:`text-xs font-medium text-danger mb-2 flex items-center`},Nn={class:`bg-success/5 border border-success/20 rounded-lg p-3`},Pn={class:`text-xs font-medium text-success mb-2 flex items-center`},Fn={class:`text-xs text-textMain/90 space-y-1 list-disc list-inside marker:text-success/60`},In={class:`font-mono text-textMain bg-base px-1 py-0.5 rounded text-[11px]`},Ln={class:`mb-2`},Rn={class:`block text-sm font-medium text-textMuted mb-1.5`},zn={class:`font-mono text-textMain`},Bn=[`disabled`,`placeholder`,`onKeydown`],Vn={key:0,class:`text-xs text-danger mt-2`},Hn={class:`mt-6 flex justify-end space-x-3`},Un=[`disabled`],Wn=[`disabled`],Gn=`npm install -g @kitecd/cli`,Kn=x({__name:`ProjectDetail`,setup(a){let ee=ye(),v=we(),x=m(),C=xe(),w=ee.params.id,T=y(()=>x.getProjectById(w)),E=n({destPath:``,preDeploy:``,postDeploy:``,categoryId:``}),D=n({cleanMode:`merge`,protectPaths:[]}),k=n(``),A=n(!1),M=n(!1),Oe=n(!1),N=n(!1),P=n(``),F=n(null),I=n(!1),L=n(!1),R=n(``),z=n(`http://127.0.0.1:3000`),B=n(``);t(async()=>{if(z.value=window.location.origin,await x.fetchProjects(),await x.fetchCategories(),T.value){E.value.destPath=T.value.destPath||``,E.value.preDeploy=T.value.preDeploy||``,E.value.postDeploy=T.value.postDeploy||``,E.value.categoryId=T.value.categoryId||``,B.value=T.value.env||``;let e=T.value.cleanMode;D.value.cleanMode=e===`clean`||e===`clean-all`?e:`merge`;let t=T.value.protectPaths;if(typeof t==`string`&&t.length>0)try{let e=JSON.parse(t);D.value.protectPaths=Array.isArray(e)?e.filter(e=>typeof e==`string`):[]}catch{D.value.protectPaths=[]}else D.value.protectPaths=[]}else v.replace(`/projects`);await K()});let V=n([]),H=n(!1),U=n(!1),W=n(!1),G=n(null);async function K(){H.value=!0;try{await x.fetchLogs(),V.value=x.logs.filter(e=>e.projectId===w).slice(0,10)}catch{V.value=[]}finally{H.value=!1}}function q(e){return e?e.slice(0,8):``}let J=n(``);async function Ae(e,t){if(t&&(t.stopPropagation(),t.preventDefault()),e)try{await navigator.clipboard.writeText(e),J.value=e,C.success(`已复制部署 ID`,q(e)),setTimeout(()=>{J.value===e&&(J.value=``)},2e3)}catch(e){C.error(`复制失败`,e?.message||`请手动选择文本复制`)}}let je=y(()=>[...V.value].sort((e,t)=>{let n=new Date(e.startTime).getTime()||0;return(new Date(t.startTime).getTime()||0)-n}).find(e=>e.status===`success`&&e.triggerSource!==`rollback`)?.id||``);function Me(e){return!e||e.status===`running`||e.triggerSource===`rollback`?!1:!!e.artifactPath}function Ne(e){return e?e.status===`running`?`部署进行中,无法回滚`:e.triggerSource===`rollback`?`回滚记录不可再被回滚`:e.artifactPath?``:`该版本归档已被清理或过早,无法回滚`:``}function Pe(e){Me(e)&&(G.value=e,U.value=!0)}async function Fe(){let e=G.value?.id;if(e){W.value=!0;try{let t=await x.rollbackDeployment(e);C.success(`回滚已完成`,`新部署 ${q(t.deployId)}`),U.value=!1,G.value=null,await K()}catch(e){C.error(`回滚失败`,e?.message||`未知错误`)}finally{W.value=!1}}}function Ie(e){e?v.push({path:`/logs`,query:{id:e.id,projectId:w}}):v.push({path:`/logs`,query:{projectId:w}})}function Le(e){if(!e)return`—`;try{let t=new Date(e);return isNaN(t.getTime())?e:t.toLocaleString()}catch{return e}}let Re=async()=>{try{await x.updateProject(w,{destPath:E.value.destPath,preDeploy:E.value.preDeploy,postDeploy:E.value.postDeploy,categoryId:E.value.categoryId||null}),C.success(`配置已保存`)}catch(e){let t=e?.data?.conflictProject;e?.status===409&&t?C.error(`保存失败`,`部署目录已被项目「${t}」占用,请更换目录或修改对方项目`):C.error(`保存失败`,e?.message||`请稍后重试`)}};function ze(){let e=k.value.trim();if(e){if(D.value.protectPaths.includes(e)){k.value=``;return}D.value.protectPaths.push(e),k.value=``}}function Kn(e){D.value.protectPaths=D.value.protectPaths.filter(t=>t!==e)}async function qn(){A.value=!0;try{await x.updateProject(w,{cleanMode:D.value.cleanMode,protectPaths:D.value.protectPaths.length?D.value.protectPaths:null}),C.success(`清理策略已保存`,D.value.cleanMode===`merge`?`将沿用合并模式(零破坏)`:`下次部署会按 ${D.value.cleanMode} 执行`)}catch(e){C.error(`保存失败`,e?.message)}finally{A.value=!1,M.value=!1}}async function Jn(){if(D.value.cleanMode===`clean-all`){M.value=!0;return}await qn()}async function Yn(){if(D.value.cleanMode===`merge`){C.info(`merge 模式不会删除任何文件,无需预览`);return}Oe.value=!0,N.value=!0,P.value=``,F.value=null;try{F.value=await x.cleanPreview(w,{cleanMode:D.value.cleanMode,protectPaths:D.value.protectPaths})}catch(e){P.value=e?.message||`预览失败`}finally{N.value=!1}}let Xn=()=>{T.value?.token&&(navigator.clipboard.writeText(T.value.token),L.value=!0,setTimeout(()=>L.value=!1,2e3))},Y=(e,t)=>{navigator.clipboard.writeText(t),R.value=e,setTimeout(()=>R.value=``,2e3)},Zn=y(()=>B.value.trim()?` --env ${B.value.trim()}`:``),Qn=y(()=>B.value.trim()?`kite.config.${B.value.trim()}.json`:`kite.config.json`),$n=y(()=>`kite init --project ${w}${Zn.value} --out ./dist --server ${z.value} --token ${T.value?.token||`<DEPLOY_TOKEN>`}`),er=y(()=>`kite push${Zn.value}`),tr=y(()=>`kite push --server ${z.value} --project ${w}${Zn.value} --out ./dist`),nr=y(()=>`kite push --server ${z.value} --project ${w} --token ${T.value?.token||`<DEPLOY_TOKEN>`}${Zn.value} --out ./dist`),rr=y(()=>JSON.stringify({projectId:w,outputDir:`./dist`,files:[`**/*`],postDeploy:T.value?.postDeploy||`pm2 restart your-service`},null,2)),ir=[{label:`打包所有文件`,files:[`**/*`]},{label:`只上传 dist 目录`,files:[`dist/**/*`]},{label:`指定多个目录`,files:[`dist/**/*`,`public/**/*`]},{label:`单个文件`,files:[`index.html`]},{label:`混合配置`,files:[`dist/**/*`,`server.js`,`config/*.json`]}],ar=n(!1),or=n(!1),sr=()=>{ar.value=!0},cr=async()=>{or.value=!0;try{await x.generateToken(w),I.value=!0,ar.value=!1,C.success(`Token 已重新生成`,`旧 Token 已立即失效`)}catch(e){C.error(`Token 重置失败`,e?.message)}finally{or.value=!1}},lr=n(!1),X=n(``),Z=n(!1),Q=n(``),$=y(()=>T.value?.name?.trim()||``),ur=y(()=>!Z.value&&$.value.length>0&&X.value.trim()===$.value);function dr(){X.value=``,Q.value=``,Z.value=!1,lr.value=!0}function fr(){Z.value||(lr.value=!1,X.value=``,Q.value=``)}async function pr(){if(ur.value){Z.value=!0,Q.value=``;try{await x.removeProject(w)?(lr.value=!1,v.replace(`/projects`)):Q.value=`删除失败,请稍后重试`}catch(e){Q.value=e?.message||`删除失败,请稍后重试`}finally{Z.value=!1}}}return(t,n)=>{let a=i(`router-link`);return T.value?(f(),p(`div`,Ve,[s(`div`,He,[s(`button`,{onClick:n[0]||=e=>r(v).back(),class:`p-2 dark:hover:bg-white/10 hover:bg-black/10 rounded-full transition-colors text-textMuted hover:text-textMain`},[b(r(ae),{class:`w-5 h-5`})]),s(`div`,null,[s(`div`,Ue,[s(`h1`,We,g(T.value.name),1),s(`span`,{class:c([`px-2.5 py-0.5 text-xs rounded-md border`,T.value.status===`success`?`bg-success/10 border-success/20 text-success`:`bg-primary/10 border-primary/20 text-primary`])},g(T.value.status),3),b(a,{to:`/projects/${r(w)}/files`,class:`inline-flex items-center px-3 py-1.5 text-xs font-medium bg-base border border-border hover:border-primary/50 hover:text-primary text-textMuted rounded-md transition-all`},{default:te(()=>[b(r(ce),{class:`w-3.5 h-3.5 mr-1.5`}),n[20]||=o(` 查看文件 `,-1)]),_:1},8,[`to`]),b(a,{to:`/audit?targetId=${r(w)}`,class:`inline-flex items-center px-3 py-1.5 text-xs font-medium bg-base border border-border hover:border-primary/50 hover:text-primary text-textMuted rounded-md transition-all`},{default:te(()=>[b(r(pe),{class:`w-3.5 h-3.5 mr-1.5`}),n[21]||=o(` 操作历史 `,-1)]),_:1},8,[`to`])]),s(`p`,Ge,g(T.value.id),1)])]),s(`div`,Ke,[s(`div`,qe,[s(`div`,Je,[s(`h2`,Ye,[b(r(le),{class:`w-5 h-5 mr-2 text-primary`}),n[22]||=o(` 鉴权 Token 管理 `,-1)]),n[23]||=s(`p`,{class:`text-sm text-textMuted mt-1`},`用于 CLI 或 Webhook 触发自动化部署的专属凭证。`,-1)]),s(`div`,Xe,[s(`div`,Ze,[s(`div`,Qe,[s(`input`,{type:I.value?`text`:`password`,readonly:``,value:T.value.token||`暂无 Token,请生成`,class:c([`w-full bg-base border border-border rounded-md pl-4 pr-12 py-3 text-textMain font-mono text-sm focus:outline-none focus:border-primary/50 transition-colors`,{"opacity-50 blur-[2px] select-none":!I.value&&T.value.token}])},null,10,$e),T.value.token?(f(),p(`button`,{key:0,onClick:n[1]||=e=>I.value=!I.value,class:`absolute right-3 top-1/2 -translate-y-1/2 text-textMuted hover:text-textMain text-xs font-medium px-2 py-1 rounded transition-colors`},g(I.value?`隐藏`:`显示`),1)):h(``,!0)]),s(`div`,et,[s(`button`,{onClick:Xn,disabled:!T.value.token,class:`flex items-center justify-center px-4 py-3 bg-base border border-border hover:border-primary/50 hover:text-primary text-textMain rounded-md transition-all disabled:opacity-50 disabled:cursor-not-allowed w-full sm:w-auto`},[L.value?(f(),l(r(Ce),{key:0,class:`w-4 h-4 mr-2 text-success`})):(f(),l(r(oe),{key:1,class:`w-4 h-4 mr-2`})),o(` `+g(L.value?`已复制`:`复制`),1)],8,tt),s(`button`,{onClick:sr,class:`flex items-center justify-center px-4 py-3 bg-primary/10 text-primary border border-primary/20 hover:bg-primary hover:text-white rounded-md transition-all w-full sm:w-auto font-medium shadow-[0_0_10px_rgba(59,130,246,0.1)] hover:shadow-[0_0_15px_rgba(59,130,246,0.4)]`},[b(r(fe),{class:`w-4 h-4 mr-2`}),n[24]||=o(` 重新生成 `,-1)])])]),n[25]||=s(`div`,{class:`mt-4 p-4 rounded-md bg-primary/5 border border-primary/10 text-sm`},[s(`p`,{class:`text-textMuted leading-relaxed`},[s(`strong`,{class:`text-primary font-medium`},`CLI 用法:`),o(` 将此 Token 保存到全局配置后,`),s(`code`,{class:`bg-base px-1 py-0.5 rounded font-mono text-xs text-textMain border border-border`},`kite push`),o(` 时无需再传。 `)])],-1)])]),s(`div`,nt,[s(`div`,rt,[s(`div`,null,[s(`h2`,it,[b(r(Ee),{class:`w-5 h-5 mr-2 text-primary`}),n[26]||=o(` 部署历史 `,-1)]),n[27]||=s(`p`,{class:`text-sm text-textMuted mt-1`},`最近 10 次部署。点击行查看完整日志,行末可对已归档版本一键回滚。`,-1)]),s(`div`,at,[s(`button`,{onClick:K,disabled:H.value,class:`inline-flex items-center px-3 py-1.5 text-xs font-medium bg-base border border-border hover:border-primary/50 hover:text-primary text-textMuted rounded-md transition-all disabled:opacity-50`},[b(r(fe),{class:c([`w-3.5 h-3.5 mr-1.5`,{"animate-spin":H.value}])},null,8,[`class`]),n[28]||=o(` 刷新 `,-1)],8,ot),s(`button`,{onClick:n[2]||=e=>Ie(),class:`inline-flex items-center px-3 py-1.5 text-xs font-medium bg-base border border-border hover:border-primary/50 hover:text-primary text-textMuted rounded-md transition-all`},[b(r(pe),{class:`w-3.5 h-3.5 mr-1.5`}),n[29]||=o(` 查看全部 `,-1)])])]),s(`div`,st,[H.value&&V.value.length===0?(f(),p(`div`,ct,` 加载中… `)):V.value.length===0?(f(),p(`div`,lt,` 该项目暂无部署记录。 `)):(f(),p(`ul`,ut,[(f(!0),p(_,null,d(V.value,e=>(f(),p(`li`,{key:e.id,class:`flex items-center gap-3 py-3 hover:bg-black/[0.02] dark:hover:bg-white/[0.02] -mx-2 px-2 rounded-md transition-colors`},[s(`button`,{onClick:t=>Ie(e),class:`flex-1 min-w-0 text-left`},[s(`div`,ft,[s(`span`,{class:c([`inline-flex items-center px-2 py-0.5 rounded text-[10px] font-medium border`,{"bg-success/10 border-success/20 text-success":e.status===`success`,"bg-danger/10 border-danger/20 text-danger":e.status===`failed`,"bg-primary/10 border-primary/20 text-primary":e.status===`running`}])},g(e.status),3),e.triggerSource===`rollback`?(f(),p(`span`,pt,` RB `)):h(``,!0),e.artifactPath?(f(),l(r(ie),{key:1,class:`w-3.5 h-3.5 text-success/70`,"aria-label":`已归档`})):(f(),l(r(S),{key:2,class:`w-3.5 h-3.5 text-textMuted/50`,"aria-label":`无归档`})),s(`span`,{role:`button`,tabindex:`0`,class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded bg-base border border-border font-mono text-[10px] text-textMuted hover:text-primary hover:border-primary/40 transition-colors cursor-pointer`,title:`点击复制完整 ID: ${e.id}`,onClick:j(t=>Ae(e.id,t),[`stop`,`prevent`]),onKeydown:ge(j(t=>Ae(e.id,t),[`stop`,`prevent`]),[`enter`])},[J.value===e.id?(f(),l(r(ne),{key:0,class:`w-3 h-3 text-success`})):(f(),l(r(oe),{key:1,class:`w-3 h-3`})),o(` `+g(q(e.id)),1)],40,mt),e.id===je.value?(f(),p(`span`,ht,`当前版本`)):h(``,!0),n[30]||=s(`span`,{class:`text-xs text-textMuted`},`·`,-1),s(`span`,gt,g(Le(e.startTime)),1),e.duration?(f(),p(`span`,_t,`· `+g(e.duration),1)):h(``,!0)])],8,dt),Me(e)?(f(),p(`button`,{key:0,onClick:j(t=>Pe(e),[`stop`]),class:`inline-flex items-center px-2.5 py-1 text-xs font-medium bg-warning/10 text-warning border border-warning/20 hover:bg-warning hover:text-white rounded-md transition-all`},[b(r(re),{class:`w-3 h-3 mr-1`}),n[31]||=o(` 回滚到此版本 `,-1)],8,vt)):(f(),p(`span`,{key:1,class:`inline-flex items-center px-2.5 py-1 text-xs font-medium bg-base text-textMuted/60 border border-border rounded-md cursor-not-allowed`,title:Ne(e)},` 不可回滚 `,8,yt))]))),128))]))])]),s(`div`,bt,[s(`div`,xt,[s(`h2`,St,[b(r(me),{class:`w-5 h-5 mr-2 text-primary`}),n[32]||=o(` CLI 快速部署指引 `,-1)]),n[33]||=s(`p`,{class:`text-sm text-textMuted mt-1`},`三步完成部署:安装 CLI、初始化配置、推送部署。`,-1)]),s(`div`,Ct,[s(`div`,wt,[n[38]||=s(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`部署环境 (可选)`,-1),s(`div`,Tt,[e(s(`input`,{"onUpdate:modelValue":n[3]||=e=>B.value=e,type:`text`,class:`flex-1 bg-base border border-border rounded-md px-3 py-2 text-textMain font-mono text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all`,placeholder:`留空为默认环境,或输入如 test、staging、prod`},null,512),[[O,B.value]])]),s(`p`,Et,[n[34]||=o(` 填写后下方所有指令将自动带上 `,-1),n[35]||=s(`code`,{class:`font-mono text-textMain`},`--env`,-1),n[36]||=o(` 参数,生成的配置文件为 `,-1),s(`code`,Dt,g(Qn.value),1),n[37]||=o(`。 适用于同一项目需要部署到不同环境(测试/预发/生产)的场景。 `,-1)])]),s(`div`,Ot,[n[39]||=s(`p`,{class:`text-sm font-medium text-textMain mb-2`},`1. 安装 CLI`,-1),s(`div`,kt,[s(`code`,{class:`flex-1 text-xs text-success font-mono break-all`},g(Gn)),s(`button`,{onClick:n[4]||=e=>Y(`install`,Gn),class:`text-xs text-primary hover:text-textMain`},g(R.value===`install`?`已复制`:`复制`),1)])]),s(`div`,At,[n[46]||=s(`p`,{class:`text-sm font-medium text-textMain mb-2`},`2. 初始化项目配置`,-1),s(`div`,jt,[s(`code`,Mt,g($n.value),1),s(`button`,{onClick:n[5]||=e=>Y(`init`,$n.value),class:`text-xs text-primary hover:text-textMain`},g(R.value===`init`?`已复制`:`复制`),1)]),s(`p`,Nt,[n[40]||=o(`执行后会在当前目录生成 `,-1),s(`code`,Pt,g(Qn.value),1),n[41]||=o(`,请确认生成的配置:`,-1)]),s(`pre`,Ft,g(rr.value),1),s(`div`,It,[n[42]||=s(`p`,null,[s(`code`,{class:`font-mono text-textMain`},`projectId`),o(` — 项目唯一标识,由服务端分配`)],-1),n[43]||=s(`p`,null,[s(`code`,{class:`font-mono text-textMain`},`outputDir`),o(` — 本地打包输出目录,默认 `),s(`code`,{class:`font-mono`},`./dist`)],-1),n[44]||=s(`p`,null,[s(`code`,{class:`font-mono text-textMain`},`files`),o(` — 要打包上传的文件 glob 模式列表,示例:`)],-1),s(`div`,Lt,[(f(),p(_,null,d(ir,e=>s(`div`,{key:e.label,class:`flex items-center gap-2`},[s(`span`,Rt,g(e.label),1),s(`code`,zt,`"files": `+g(JSON.stringify(e.files)),1)])),64))]),n[45]||=s(`p`,null,[s(`code`,{class:`font-mono text-textMain`},`postDeploy`),o(` — 服务端解压后执行的命令,如重启服务`)],-1)])]),s(`div`,Bt,[s(`div`,Vt,[n[47]||=s(`p`,{class:`text-sm font-medium text-textMain mb-2`},`3. 部署 — 使用已保存的配置`,-1),s(`div`,Ht,[s(`code`,Ut,g(er.value),1),s(`button`,{onClick:n[6]||=e=>Y(`push`,er.value),class:`text-xs text-primary hover:text-textMain`},g(R.value===`push`?`已复制`:`复制`),1)]),n[48]||=s(`p`,{class:`text-xs text-textMuted mt-2`},[o(`需先通过 `),s(`code`,{class:`font-mono`},`kite config:set token`),o(` 或 `),s(`code`,{class:`font-mono`},`--token-store global`),o(` 保存过 Token。`)],-1)]),s(`div`,Wt,[n[51]||=s(`p`,{class:`text-sm font-medium text-textMain mb-2`},`3. 部署 — CLI 覆盖配置`,-1),s(`div`,Gt,[n[49]||=s(`span`,{class:`text-xs text-textMuted w-20 shrink-0`},`使用全局 Token`,-1),s(`code`,Kt,g(tr.value),1),s(`button`,{onClick:n[7]||=e=>Y(`direct-push`,tr.value),class:`text-xs text-primary hover:text-textMain`},g(R.value===`direct-push`?`已复制`:`复制`),1)]),s(`div`,qt,[n[50]||=s(`span`,{class:`text-xs text-textMuted w-20 shrink-0`},`指定项目 Token`,-1),s(`code`,Jt,g(nr.value),1),s(`button`,{onClick:n[8]||=e=>Y(`direct-push-token`,nr.value),class:`text-xs text-primary hover:text-textMain`},g(R.value===`direct-push-token`?`已复制`:`复制`),1)])])]),n[52]||=u(`<div class="rounded-lg border border-primary/20 bg-primary/5 p-4 space-y-2"><p class="text-sm text-textMuted leading-relaxed"><strong class="text-primary font-medium">Token 设置方式:</strong><code class="font-mono text-xs bg-base px-1 py-0.5 rounded border border-border">kite config:set token &lt;token&gt;</code> 按项目保存, <code class="font-mono text-xs bg-base px-1 py-0.5 rounded border border-border">kite config:set token &lt;token&gt; --global</code> 设置全局 fallback。 也可在 <code class="font-mono">.env.local</code> 中写入 <code class="font-mono">KITE_DEPLOY_TOKEN=&lt;token&gt;</code>。 </p><p class="text-sm text-textMuted leading-relaxed"> 配置优先级:<strong class="text-primary">CLI 参数</strong> &gt; <strong class="text-primary">.env.local</strong> &gt; <strong class="text-primary">项目级 Token</strong> &gt; <strong class="text-primary">全局 Token</strong>。未在 CLI 传入的部署脚本,会回退到本页保存的云端默认脚本。 </p></div>`,1)])]),s(`div`,Yt,[s(`div`,Xt,[s(`h2`,Zt,[b(r(me),{class:`w-5 h-5 mr-2 text-primary`}),n[53]||=o(` 部署脚本配置 (云端默认) `,-1)]),n[54]||=s(`p`,{class:`text-sm text-textMuted mt-1`},`配置此项目在服务端接收到文件后,默认执行的 Shell 指令。可被 CLI 参数覆盖。`,-1)]),s(`div`,Qt,[s(`div`,null,[n[55]||=s(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`部署目录 (Destination Path)`,-1),e(s(`input`,{"onUpdate:modelValue":n[9]||=e=>E.value.destPath=e,type:`text`,class:`w-full bg-base border border-border rounded-md px-4 py-3 text-textMain font-mono text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all`,placeholder:`e.g. /var/www/my-project`},null,512),[[O,E.value.destPath]]),n[56]||=s(`p`,{class:`text-xs text-textMuted mt-2`},`在服务端解压和部署该项目文件的绝对路径。`,-1)]),s(`div`,null,[n[58]||=s(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`所属分类`,-1),e(s(`select`,{"onUpdate:modelValue":n[10]||=e=>E.value.categoryId=e,class:`w-full bg-base border border-border rounded-md px-4 py-3 text-textMain text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all`},[n[57]||=s(`option`,{value:``},`默认(未分类)`,-1),(f(!0),p(_,null,d(r(x).categories,e=>(f(),p(`option`,{key:e.id,value:e.id},g(e.name),9,$t))),128))],512),[[_e,E.value.categoryId]]),n[59]||=s(`p`,{class:`text-xs text-textMuted mt-2`},`用于在项目列表中按分类筛选。在「项目管理 → 管理分类」中创建更多分类。`,-1)]),s(`div`,null,[n[61]||=s(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`前置脚本 (Pre-Deploy)`,-1),s(`div`,en,[n[60]||=s(`div`,{class:`absolute left-0 top-0 bottom-0 w-8 bg-base border-r border-border rounded-l-md flex flex-col items-center py-3 text-textMuted font-mono text-xs select-none`},[s(`span`,null,`1`)],-1),e(s(`textarea`,{"onUpdate:modelValue":n[11]||=e=>E.value.preDeploy=e,class:`w-full bg-base border border-border rounded-md pl-11 pr-4 py-3 text-success font-mono text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all min-h-[100px] resize-y`,placeholder:`# e.g. npm install && npm run build`,spellcheck:`false`},null,512),[[O,E.value.preDeploy]])]),n[62]||=s(`p`,{class:`text-xs text-textMuted mt-2`},`在打包上传之前,于本地执行的构建命令(通常配置在本地 CLI,此处作为备用参考)。`,-1)]),s(`div`,null,[n[64]||=s(`label`,{class:`block text-sm font-medium text-textMain mb-2`},`后置脚本 (Post-Deploy)`,-1),s(`div`,tn,[n[63]||=s(`div`,{class:`absolute left-0 top-0 bottom-0 w-8 bg-base border-r border-border rounded-l-md flex flex-col items-center py-3 text-textMuted font-mono text-xs select-none`},[s(`span`,null,`1`)],-1),e(s(`textarea`,{"onUpdate:modelValue":n[12]||=e=>E.value.postDeploy=e,class:`w-full bg-base border border-border rounded-md pl-11 pr-4 py-3 text-success font-mono text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all min-h-[100px] resize-y`,placeholder:`# e.g. pm2 restart api-server`,spellcheck:`false`},null,512),[[O,E.value.postDeploy]])]),n[65]||=s(`p`,{class:`text-xs text-textMuted mt-2`},`服务端解压文件后,在目标目录执行的重启或服务加载命令。`,-1)]),s(`div`,nn,[s(`button`,{onClick:Re,class:`flex items-center px-6 py-2.5 bg-primary hover:bg-primary/90 text-white rounded-md transition-all font-medium shadow-[0_0_15px_rgba(59,130,246,0.3)]`},[b(r(ue),{class:`w-4 h-4 mr-2`}),n[66]||=o(` 保存配置 `,-1)])])])]),s(`div`,rn,[s(`div`,an,[s(`h2`,on,[b(r(ke),{class:`w-5 h-5 mr-2 text-primary`}),n[67]||=o(` 部署清理策略 `,-1)]),n[68]||=s(`p`,{class:`text-sm text-textMuted mt-1`},[o(`每次部署解压前,对目标目录执行的清理动作。默认 `),s(`code`,{class:`font-mono text-textMain`},`merge`),o(` 沿用旧行为(零破坏)。`)],-1)]),s(`div`,sn,[s(`div`,cn,[(f(),p(_,null,d([{value:`merge`,title:`merge`,desc:`不清理,直接覆盖。旧行为,零破坏。`,tone:`primary`},{value:`clean`,title:`clean`,desc:`清空目录,但保留 protectPaths 命中的文件,以及 .kite-* 内部目录。`,tone:`warning`},{value:`clean-all`,title:`clean-all`,desc:`清空全部内容(仅保留 .kite-*),protectPaths 也被忽略。`,tone:`danger`}],t=>s(`label`,{key:t.value,class:c([`relative flex flex-col p-4 rounded-lg border-2 cursor-pointer transition-all`,D.value.cleanMode===t.value?t.tone===`danger`?`border-danger bg-danger/5`:t.tone===`warning`?`border-yellow-400 bg-yellow-400/5`:`border-primary bg-primary/5`:`border-border hover:border-textMuted/50 bg-base`])},[e(s(`input`,{type:`radio`,"onUpdate:modelValue":n[13]||=e=>D.value.cleanMode=e,value:t.value,class:`sr-only`},null,8,ln),[[be,D.value.cleanMode]]),s(`span`,{class:c([`text-sm font-semibold font-mono`,t.tone===`danger`?`text-danger`:t.tone===`warning`?`text-yellow-400`:`text-primary`])},g(t.title),3),s(`span`,un,g(t.desc),1),D.value.cleanMode===t.value?(f(),l(r(Ce),{key:0,class:c([`absolute top-2 right-2 w-4 h-4`,t.tone===`danger`?`text-danger`:t.tone===`warning`?`text-yellow-400`:`text-primary`])},null,8,[`class`])):h(``,!0)],2)),64))]),D.value.cleanMode===`clean`?(f(),p(`div`,dn,[n[70]||=u(`<label class="block text-sm font-medium text-textMain mb-2">保护路径 (protectPaths)</label><p class="text-xs text-textMuted mb-2"> 支持 minimatch glob,命中文件不会被删除。<code class="font-mono text-textMain">.kite-*</code> 始终自动保护,无需添加。 常见示例:<code class="font-mono text-textMain">uploads/**</code>、<code class="font-mono text-textMain">.env</code>、<code class="font-mono text-textMain">config/*.json</code></p>`,2),s(`div`,fn,[e(s(`input`,{"onUpdate:modelValue":n[14]||=e=>k.value=e,type:`text`,class:`flex-1 bg-base border border-border rounded-md px-3 py-2 text-textMain font-mono text-sm focus:outline-none focus:border-primary focus:ring-1 focus:ring-primary/50 transition-all`,placeholder:`如 uploads/**`,onKeydown:ge(j(ze,[`prevent`]),[`enter`])},null,40,pn),[[O,k.value]]),s(`button`,{onClick:ze,type:`button`,class:`flex items-center px-3 bg-base border border-border hover:border-primary/50 hover:text-primary text-textMain rounded-md transition-all`},[b(r(de),{class:`w-4 h-4 mr-1`}),n[69]||=o(` 添加 `,-1)])]),D.value.protectPaths.length?(f(),p(`div`,mn,[(f(!0),p(_,null,d(D.value.protectPaths,e=>(f(),p(`span`,{key:e,class:`inline-flex items-center gap-1 text-xs px-2 py-1 rounded bg-success/10 border border-success/30 text-success font-mono`},[o(g(e)+` `,1),s(`button`,{onClick:t=>Kn(e),type:`button`,class:`text-success/70 hover:text-danger`},[b(r(Se),{class:`w-3 h-3`})],8,hn)]))),128))])):(f(),p(`p`,gn,`尚未设置保护路径。clean 模式下会清空整个部署目录(仅保留 .kite-*)。`))])):h(``,!0),D.value.cleanMode===`clean-all`?(f(),p(`div`,_n,[b(r(De),{class:`w-4 h-4 text-danger shrink-0 mt-0.5`}),n[71]||=s(`p`,{class:`text-xs text-danger leading-relaxed`},[o(` clean-all 会清空目标目录下`),s(`strong`,null,`所有`),o(`文件(仅保留 `),s(`code`,{class:`font-mono bg-base px-1 rounded`},`.kite-*`),o(`)。protectPaths 设置在此模式下被忽略。请务必通过预览确认。 `)],-1)])):h(``,!0),s(`div`,vn,[D.value.cleanMode===`merge`?h(``,!0):(f(),p(`button`,{key:0,onClick:Yn,type:`button`,class:`flex items-center px-4 py-2 text-sm bg-base border border-border hover:border-yellow-400/50 hover:text-yellow-400 text-textMain rounded-md transition-all`},[b(r(se),{class:`w-4 h-4 mr-2`}),n[72]||=o(` 预览将删除的文件 (DRY-RUN) `,-1)])),s(`button`,{onClick:Jn,disabled:A.value,class:c([`flex items-center px-6 py-2.5 text-sm font-medium rounded-md transition-all disabled:opacity-50`,D.value.cleanMode===`clean-all`?`bg-danger text-white hover:bg-danger/90`:`bg-primary text-white hover:bg-primary/90 shadow-[0_0_15px_rgba(59,130,246,0.3)]`])},[b(r(ue),{class:`w-4 h-4 mr-2`}),o(` `+g(A.value?`保存中...`:`保存清理策略`),1)],10,yn)])])]),s(`div`,bn,[s(`div`,xn,[s(`h3`,Sn,[b(r(he),{class:`w-4 h-4 mr-2`}),n[73]||=o(` 危险操作区 `,-1)]),s(`div`,{class:`mt-4 flex items-start justify-between gap-4`},[n[74]||=s(`div`,{class:`text-sm text-textMuted space-y-1`},[s(`p`,null,`删除该项目将同时清空数据库中的项目配置与全部部署历史日志,且不可恢复。`),s(`p`,{class:`text-textMuted/80`},`部署目录中的实际文件不会被删除,需要时请手动清理。`)],-1),s(`button`,{onClick:dr,class:`shrink-0 px-4 py-2 bg-danger/10 hover:bg-danger text-danger hover:text-white border border-danger/20 hover:border-danger rounded-md transition-colors text-sm font-medium`},` 删除项目 `)])])])]),lr.value?(f(),p(`div`,{key:0,class:`fixed inset-0 z-50 flex items-center justify-center bg-black/60 backdrop-blur-sm`,onClick:j(fr,[`self`])},[s(`div`,Cn,[s(`div`,wn,[s(`div`,Tn,[b(r(ve),{class:`w-5 h-5 text-danger`})]),s(`div`,En,[n[78]||=s(`h2`,{class:`text-lg font-semibold text-textMain`},`确认删除项目`,-1),s(`p`,Dn,[n[75]||=o(` 即将删除项目 `,-1),s(`span`,On,g(T.value?.name),1),n[76]||=o(` (`,-1),s(`span`,kn,g(r(w)),1),n[77]||=o(`),此操作不可恢复。 `,-1)])])]),s(`div`,An,[s(`div`,jn,[s(`p`,Mn,[b(r(Se),{class:`w-3.5 h-3.5 mr-1.5`}),n[79]||=o(` 将被永久删除的内容 `,-1)]),n[80]||=s(`ul`,{class:`text-xs text-textMain/90 space-y-1 list-disc list-inside marker:text-danger/60`},[s(`li`,null,`该项目在数据库中的配置(名称、描述、部署目录、部署脚本、Token、环境标识等)`),s(`li`,null,[o(`该项目的`),s(`span`,{class:`font-medium`},`全部部署历史日志`),o(`(部署日志面板中将不再可见)`)])],-1)]),s(`div`,Nn,[s(`p`,Pn,[b(r(Ce),{class:`w-3.5 h-3.5 mr-1.5`}),n[81]||=o(` 不会被删除的内容 `,-1)]),s(`ul`,Fn,[s(`li`,null,[n[82]||=o(` 部署目录 `,-1),s(`code`,In,g(T.value?.destPath||`—`),1),n[83]||=o(` 下的所有实际文件 `,-1)]),n[84]||=s(`li`,null,`其他项目的数据、全局设置、Admin Token`,-1),n[85]||=s(`li`,null,[o(`项目源码中的 `),s(`code`,{class:`font-mono text-textMain bg-base px-1 py-0.5 rounded text-[11px]`},`kite.config*.json`),o(` / `),s(`code`,{class:`font-mono text-textMain bg-base px-1 py-0.5 rounded text-[11px]`},`.env.local`),o(` 等本地配置`)],-1)])])]),s(`div`,Ln,[s(`label`,Rn,[n[86]||=o(` 请输入项目名称 `,-1),s(`span`,zn,g($.value),1),n[87]||=o(` 以确认删除 `,-1)]),e(s(`input`,{"onUpdate:modelValue":n[15]||=e=>X.value=e,type:`text`,disabled:Z.value,placeholder:$.value,class:`w-full bg-base border border-border rounded-md px-3 py-2 text-textMain font-mono focus:outline-none focus:border-danger focus:ring-1 focus:ring-danger/50 transition-all text-sm disabled:opacity-60`,onKeydown:ge(j(pr,[`prevent`]),[`enter`])},null,40,Bn),[[O,X.value]])]),Q.value?(f(),p(`p`,Vn,g(Q.value),1)):h(``,!0),s(`div`,Hn,[s(`button`,{onClick:fr,disabled:Z.value,class:`px-4 py-2 text-sm font-medium text-textMuted hover:text-textMain dark:hover:bg-white/5 hover:bg-black/5 rounded-md transition-colors disabled:opacity-50 disabled:cursor-not-allowed`},` 取消 `,8,Un),s(`button`,{onClick:pr,disabled:!ur.value,class:`px-4 py-2 text-sm font-medium bg-danger text-white rounded-md hover:bg-danger/90 disabled:opacity-50 disabled:cursor-not-allowed transition-colors flex items-center`},[Z.value?(f(),l(r(fe),{key:0,class:`w-4 h-4 mr-2 animate-spin`})):(f(),l(r(he),{key:1,class:`w-4 h-4 mr-2`})),o(` `+g(Z.value?`正在删除...`:`永久删除`),1)],8,Wn)])])])):h(``,!0),b(Te,{open:ar.value,"onUpdate:open":n[16]||=e=>ar.value=e,tone:`warning`,title:`重新生成项目 Token?`,message:`旧 Token 将立即失效。所有正在使用旧 Token 的 CLI / Webhook 调用都会被拒绝,请记得同步更新。`,"confirm-text":`重新生成`,"cancel-text":`取消`,loading:or.value,onConfirm:cr},null,8,[`open`,`loading`]),b(Be,{open:Oe.value,"onUpdate:open":n[17]||=e=>Oe.value=e,loading:N.value,error:P.value,preview:F.value,mode:D.value.cleanMode===`merge`?`clean`:D.value.cleanMode,"protect-paths":D.value.protectPaths},null,8,[`open`,`loading`,`error`,`preview`,`mode`,`protect-paths`]),b(Te,{open:M.value,"onUpdate:open":n[18]||=e=>M.value=e,tone:`danger`,title:`确认启用 clean-all 模式?`,message:`后续每次部署都会清空部署目录下除 .kite-* 之外的全部内容,protectPaths 在此模式下被忽略。请输入项目名以确认。`,"confirm-text":`启用 clean-all`,"cancel-text":`取消`,"require-text":$.value,"require-text-hint":`请输入项目名 ${$.value} 以确认`,loading:A.value,onConfirm:qn},null,8,[`open`,`require-text`,`require-text-hint`,`loading`]),b(Te,{open:U.value,"onUpdate:open":n[19]||=e=>U.value=e,tone:`warning`,title:`确认回滚到此版本?`,message:G.value?`将以归档 ${q(G.value.id)} 重新部署到项目 ${G.value.projectName}。会按当前项目的 cleanMode / protectPaths 执行清理后再解压,运行时数据按保护规则保留。`:``,"confirm-text":`确认回滚`,"cancel-text":`取消`,loading:W.value,onConfirm:Fe},null,8,[`open`,`message`,`loading`])])):h(``,!0)}}});export{Kn as default};
@@ -0,0 +1 @@
1
+ .fade-enter-active[data-v-e5333666],.fade-leave-active[data-v-e5333666]{transition:opacity .15s}.fade-enter-from[data-v-e5333666],.fade-leave-to[data-v-e5333666]{opacity:0}
@@ -0,0 +1 @@
1
+ .fade-enter-active[data-v-72963037],.fade-leave-active[data-v-72963037]{transition:opacity .15s}.fade-enter-from[data-v-72963037],.fade-leave-to[data-v-72963037]{opacity:0}.fade-enter-active[data-v-1f9f478f],.fade-leave-active[data-v-1f9f478f]{transition:opacity .12s}.fade-enter-from[data-v-1f9f478f],.fade-leave-to[data-v-1f9f478f]{opacity:0}