local-cmd-runner 1.0.5 → 1.0.6

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.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "local-cmd-runner",
3
- "version": "1.0.5",
3
+ "version": "1.0.6",
4
4
  "description": "Run local shell commands from a web page",
5
5
  "homepage": "https://github.com/codewoow/local_cmd#readme",
6
6
  "bugs": {
package/public/app.js CHANGED
@@ -146,6 +146,9 @@ const fmCurrentPath = document.getElementById('fmCurrentPath');
146
146
  const fmUploadBtn = document.getElementById('fmUploadBtn');
147
147
  const fmUploadInput = document.getElementById('fmUploadInput');
148
148
  const fmRefreshBtn = document.getElementById('fmRefreshBtn');
149
+ const fmUploadProgress = document.getElementById('fmUploadProgress');
150
+ const fmProgressBar = document.getElementById('fmProgressBar');
151
+ const fmProgressText = document.getElementById('fmProgressText');
149
152
 
150
153
  const previewModal = document.getElementById('previewModal');
151
154
  const previewTitle = document.getElementById('previewTitle');
@@ -243,27 +246,61 @@ function triggerDirectoryUpload(targetPath) {
243
246
  if (fmRefreshBtn) {
244
247
  fmRefreshBtn.onclick = () => loadTree(currentRelPath);
245
248
  fmUploadBtn.onclick = () => triggerDirectoryUpload(currentRelPath);
246
- fmUploadInput.onchange = async (e) => {
249
+ fmUploadInput.onchange = (e) => {
247
250
  const file = e.target.files[0];
248
251
  if (!file) return;
252
+
253
+ if (file.size > 5 * 1024 * 1024) {
254
+ alert('上传失败:文件大小不能超过 5MB!');
255
+ fmUploadInput.value = '';
256
+ return;
257
+ }
258
+
249
259
  const formData = new FormData();
250
260
  formData.append('file', file);
251
261
  formData.append('path', pendingUploadPath);
252
262
 
253
- try {
254
- fmUploadBtn.textContent = '...';
255
- const res = await fetch('/cmd/api/upload', {
256
- method: 'POST',
257
- body: formData
258
- });
259
- if (!res.ok) throw new Error('上传时服务器报错');
263
+ fmUploadBtn.textContent = '...';
264
+ if (fmUploadProgress) {
265
+ fmUploadProgress.classList.remove('hidden');
266
+ fmProgressBar.style.width = '0%';
267
+ fmProgressText.textContent = '0%';
268
+ }
269
+
270
+ const xhr = new XMLHttpRequest();
271
+ xhr.open('POST', '/cmd/api/upload');
272
+
273
+ xhr.upload.onprogress = (event) => {
274
+ if (event.lengthComputable && fmUploadProgress) {
275
+ const percentComplete = Math.floor((event.loaded / event.total) * 100);
276
+ fmProgressBar.style.width = percentComplete + '%';
277
+ fmProgressText.textContent = percentComplete + '%';
278
+ }
279
+ };
280
+
281
+ xhr.onload = () => {
260
282
  fmUploadInput.value = '';
261
- loadTree(currentRelPath);
262
- } catch (err) {
263
- alert('上传失败: ' + err.message);
264
- } finally {
265
283
  fmUploadBtn.textContent = '↑';
266
- }
284
+ if (fmUploadProgress) fmUploadProgress.classList.add('hidden');
285
+
286
+ if (xhr.status >= 200 && xhr.status < 300) {
287
+ // Success
288
+ loadTree(currentRelPath);
289
+ } else {
290
+ let errMsg = '上传时服务器报错';
291
+ try { errMsg = JSON.parse(xhr.responseText).error || errMsg; } catch (e) {}
292
+ alert('上传失败: ' + errMsg);
293
+ }
294
+ };
295
+
296
+ xhr.onerror = () => {
297
+ fmUploadInput.value = '';
298
+ fmUploadBtn.textContent = '↑';
299
+ if (fmUploadProgress) fmUploadProgress.classList.add('hidden');
300
+ alert('上传失败: 网络错误');
301
+ };
302
+
303
+ xhr.send(formData);
267
304
  };
268
305
  }
269
306
 
package/public/index.html CHANGED
@@ -58,6 +58,12 @@
58
58
  </div>
59
59
  </div>
60
60
  <div class="fm-path" id="fmCurrentPath">/root/.openclaw</div>
61
+ <div id="fmUploadProgress" class="fm-upload-progress hidden">
62
+ <div class="fm-progress-bar-wrap">
63
+ <div class="fm-progress-bar" id="fmProgressBar"></div>
64
+ </div>
65
+ <div class="fm-progress-text" id="fmProgressText">0%</div>
66
+ </div>
61
67
  <div class="fm-body" id="fmTree">
62
68
  <div class="fm-loading">加载中...</div>
63
69
  </div>
@@ -68,7 +74,8 @@
68
74
  <div id="previewModal" class="modal-overlay hidden">
69
75
  <div class="modal-content preview-modal-content">
70
76
  <h3 id="previewTitle" style="word-break: break-all;">文件预览</h3>
71
- <div class="preview-body" id="previewBody" style="display: flex; flex-direction: column; align-items: center; justify-content: center;"></div>
77
+ <div class="preview-body" id="previewBody"
78
+ style="display: flex; flex-direction: column; align-items: center; justify-content: center;"></div>
72
79
  <div class="modal-actions">
73
80
  <button id="previewClose" class="btn-primary">关闭</button>
74
81
  </div>
package/public/style.css CHANGED
@@ -470,3 +470,40 @@ input[type="text"]:focus {
470
470
  white-space: pre-wrap;
471
471
  word-break: break-all;
472
472
  }
473
+
474
+ /* File Manager Progress */
475
+ .fm-upload-progress {
476
+ display: flex;
477
+ align-items: center;
478
+ padding: 8px 16px;
479
+ background: rgba(59, 130, 246, 0.1);
480
+ border-bottom: 1px solid rgba(255, 255, 255, 0.05);
481
+ gap: 12px;
482
+ }
483
+
484
+ .fm-upload-progress.hidden {
485
+ display: none;
486
+ }
487
+
488
+ .fm-progress-bar-wrap {
489
+ flex: 1;
490
+ background: rgba(0, 0, 0, 0.3);
491
+ border-radius: 4px;
492
+ height: 8px;
493
+ overflow: hidden;
494
+ }
495
+
496
+ .fm-progress-bar {
497
+ background: var(--accent);
498
+ width: 0%;
499
+ height: 100%;
500
+ transition: width 0.1s linear;
501
+ }
502
+
503
+ .fm-progress-text {
504
+ font-size: 0.8rem;
505
+ color: var(--text-primary);
506
+ min-width: 36px;
507
+ text-align: right;
508
+ font-variant-numeric: tabular-nums;
509
+ }
package/server.js CHANGED
@@ -79,16 +79,29 @@ app.get('/cmd/api/download', (req, res) => {
79
79
  }
80
80
  });
81
81
 
82
- const upload = multer({ dest: os.tmpdir() });
83
- app.post('/cmd/api/upload', upload.single('file'), async (req, res) => {
84
- try {
85
- const targetDir = getSecurePath(req.body.path || '');
86
- const targetPath = path.join(targetDir, req.file.originalname);
87
- await fs.promises.rename(req.file.path, targetPath);
88
- res.json({ success: true });
89
- } catch (err) {
90
- res.status(500).json({ error: err.message });
91
- }
82
+ const upload = multer({
83
+ dest: os.tmpdir(),
84
+ limits: { fileSize: 5 * 1024 * 1024 }
85
+ }).single('file');
86
+
87
+ app.post('/cmd/api/upload', (req, res) => {
88
+ upload(req, res, async (err) => {
89
+ if (err instanceof multer.MulterError && err.code === 'LIMIT_FILE_SIZE') {
90
+ return res.status(400).json({ error: '文件大小超出 5MB 限制!' });
91
+ } else if (err) {
92
+ return res.status(500).json({ error: err.message });
93
+ }
94
+
95
+ try {
96
+ if (!req.file) throw new Error('未提供文件');
97
+ const targetDir = getSecurePath(req.body.path || '');
98
+ const targetPath = path.join(targetDir, req.file.originalname);
99
+ await fs.promises.rename(req.file.path, targetPath);
100
+ res.json({ success: true });
101
+ } catch (e) {
102
+ res.status(500).json({ error: e.message });
103
+ }
104
+ });
92
105
  });
93
106
 
94
107
  io.on('connection', (socket) => {