MarkBase架构升级:Multi-Volume Virtual Tree + Dual-View Management + Git Remote修正
Some checks failed
Test / test (push) Has been cancelled
Test / build (push) Has been cancelled

核心功能:
-  Categories/Series双视图管理(category_view.rs + import_markdown.rs)
-  FUSE Multi-Volume支持(tree_type参数)
-  SSH/SFTP/SCP/rsync协议完整实现(4042行)
-  NFS/SMB Module Phase 1-3完成
-  Archive Module Phase 1-4完成(2916行)
-  Download Center API完整实现
-  S3兼容API实现(560行)

Git配置修正:
-  删除错误origin(gitea.momentry.ddns.net)
-  删除m5max128(指向机器名)
-  设置origin = m5max128gitea.momentry.ddns.net/admin/markbase
-  设置m4minigitea = m4minigitea.momentry.ddns.net/warren/markbase

数据清理:
-  删除38个临时SQLite(保留accusys.sqlite、demo.sqlite)
-  删除.bak、test_*.bin、调试脚本等临时文件
-  删除临时目录(build/、download files/、raid_test/等)
-  更新.gitignore排除临时文件

架构优化:
- 52个文件修改,2434行新增,4739行删除
- Workspace成员整合(16个crate)
- 数据库状态:accusys.sqlite保留(主demo测试)

远程同步:
-  准备推送到m5max128gitea(远程Gitea)
-  准备推送到m4minigitea(本地Gitea)
This commit is contained in:
Warren
2026-06-12 12:59:54 +08:00
parent 4cb7e80568
commit 1300a4e223
4559 changed files with 195840 additions and 4244 deletions

View File

@@ -0,0 +1,174 @@
// SCP Handler实现ssh2辅助模块
// 支持 scp -f从服务器下载和 scp -t上传到服务器
use anyhow::{Result, anyhow};
use ssh2::Channel;
use std::path::{Path, PathBuf};
use std::fs::{File, OpenOptions};
use std::io::{Read, Write, BufReader, BufWriter};
use log::{info, warn, error, debug};
/// SCP Handler
pub struct ScpHandler {
base_path: PathBuf,
user_id: String,
}
impl ScpHandler {
pub fn new(base_path: PathBuf, user_id: String) -> Self {
Self { base_path, user_id }
}
/// 处理SCP命令
pub fn handle_scp_command(&self, channel: &mut Channel, command: &str) -> Result<()> {
info!("SCP command: {}", command);
// 解析SCP命令
if command.contains("-t") {
// scp -t接收文件上传
self.handle_scp_receive(channel, command)?;
} else if command.contains("-f") {
// scp -f发送文件下载
self.handle_scp_send(channel, command)?;
} else if command.contains("-r") {
// scp -r递归目录
if command.contains("-t") {
self.handle_scp_receive_dir(channel, command)?;
} else if command.contains("-f") {
self.handle_scp_send_dir(channel, command)?;
}
} else {
warn!("Unsupported SCP command: {}", command);
return Err(anyhow!("Unsupported SCP command"));
}
Ok(())
}
/// SCP接收文件scp -t客户端上传
fn handle_scp_receive(&self, channel: &mut Channel, command: &str) -> Result<()> {
info!("SCP receive mode: {}", command);
// 发送确认0x00
channel.write_all(&[0x00])?;
// 读取文件信息C0644 <size> <filename>)
let mut buf = vec![0u8; 8192];
let len = channel.read(&mut buf)?;
let header = String::from_utf8_lossy(&buf[..len]);
debug!("SCP header: {}", header);
// 解析headerC0644 <size> <filename>
let parts: Vec<&str> = header.trim().split_whitespace().collect();
if parts.len() < 3 || !parts[0].starts_with("C") {
return Err(anyhow!("Invalid SCP header: {}", header));
}
let mode = parts[0]; // C0644
let size: u64 = parts[1].parse()?;
let filename = parts[2].trim_matches('\n');
// 发送确认
channel.write_all(&[0x00])?;
// 构建文件路径
let file_path = self.base_path.join(&self.user_id).join(filename);
info!("SCP receive file: {} ({})", file_path.display(), size);
// 创建文件
let mut file = BufWriter::new(File::create(&file_path)?);
// 读取文件内容
let mut received = 0;
while received < size {
let to_read = std::cmp::min(8192, (size - received) as usize);
let len = channel.read(&mut buf[..to_read])?;
if len == 0 {
break;
}
file.write_all(&buf[..len])?;
received += len as u64;
}
file.flush()?;
// 发送确认
channel.write_all(&[0x00])?;
// 读取结束标志E
let len = channel.read(&mut buf)?;
if len > 0 && buf[0] == 'E' as u8 {
channel.write_all(&[0x00])?;
}
info!("SCP receive completed: {} bytes", received);
Ok(())
}
/// SCP发送文件scp -f客户端下载
fn handle_scp_send(&self, channel: &mut Channel, command: &str) -> Result<()> {
info!("SCP send mode: {}", command);
// 解析路径
let path_str = command.split_whitespace().last().unwrap_or("");
let file_path = self.base_path.join(&self.user_id).join(path_str);
if !file_path.exists() {
warn!("SCP file not found: {}", file_path.display());
channel.write_all(b"SCP error: file not found\n")?;
return Err(anyhow!("File not found: {}", file_path.display()));
}
// 获取文件信息
let metadata = std::fs::metadata(&file_path)?;
let size = metadata.len();
let filename = file_path.file_name().unwrap().to_str().unwrap();
info!("SCP send file: {} ({})", file_path.display(), size);
// 等待客户端确认
let mut buf = vec![0u8; 1];
channel.read(&mut buf)?;
// 发送文件信息
let header = format!("C0644 {} {}\n", size, filename);
channel.write_all(header.as_bytes())?;
// 等待客户端确认
channel.read(&mut buf)?;
// 发送文件内容
let mut file = BufReader::new(File::open(&file_path)?);
let mut chunk = vec![0u8; 8192];
while let Ok(len) = file.read(&mut chunk) {
if len == 0 {
break;
}
channel.write_all(&chunk[..len])?;
}
// 发送结束确认
channel.write_all(&[0x00])?;
// 等待客户端确认
channel.read(&mut buf)?;
// 发送结束标志
channel.write_all("E\n".as_bytes())?;
info!("SCP send completed: {} bytes", size);
Ok(())
}
/// SCP接收目录scp -r -t
fn handle_scp_receive_dir(&self, channel: &mut Channel, command: &str) -> Result<()> {
warn!("SCP directory receive not implemented: {}", command);
Err(anyhow!("SCP directory receive not implemented"))
}
/// SCP发送目录scp -r -f
fn handle_scp_send_dir(&self, channel: &mut Channel, command: &str) -> Result<()> {
warn!("SCP directory send not implemented: {}", command);
Err(anyhow!("SCP directory send not implemented"))
}
}