MarkBase架构升级:Multi-Volume Virtual Tree + Dual-View Management + Git Remote修正
核心功能: - ✅ 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:
75
markbase-core/src/sftp/audit.rs
Normal file
75
markbase-core/src/sftp/audit.rs
Normal file
@@ -0,0 +1,75 @@
|
||||
use chrono::Utc;
|
||||
use std::fs::{File, OpenOptions};
|
||||
use std::io::Write;
|
||||
use std::path::PathBuf;
|
||||
use std::sync::Mutex;
|
||||
|
||||
pub struct AuditLog {
|
||||
log_file: Mutex<File>,
|
||||
log_path: PathBuf,
|
||||
}
|
||||
|
||||
impl AuditLog {
|
||||
pub fn new(log_path: &str) -> anyhow::Result<Self> {
|
||||
let path = PathBuf::from(log_path);
|
||||
|
||||
// 确保日志目录存在
|
||||
if let Some(parent) = path.parent() {
|
||||
std::fs::create_dir_all(parent)?;
|
||||
}
|
||||
|
||||
// 打开日志文件(追加模式)
|
||||
let file = OpenOptions::new().create(true).append(true).open(&path)?;
|
||||
|
||||
Ok(Self {
|
||||
log_file: Mutex::new(file),
|
||||
log_path: path,
|
||||
})
|
||||
}
|
||||
|
||||
pub fn log_operation(&self, user_id: &str, operation: &str, path: &str, result: &str) {
|
||||
let timestamp = Utc::now().format("%Y-%m-%dT%H:%M:%SZ");
|
||||
let entry = format!(
|
||||
"[{}] user={} operation={} path=\"{}\" result={}\n",
|
||||
timestamp, user_id, operation, path, result
|
||||
);
|
||||
|
||||
// 写入日志文件
|
||||
if let Ok(mut file) = self.log_file.lock() {
|
||||
if let Err(e) = file.write_all(entry.as_bytes()) {
|
||||
log::error!("Failed to write audit log: {}", e);
|
||||
}
|
||||
}
|
||||
|
||||
// 同时输出到标准日志
|
||||
log::info!(
|
||||
"Audit: user={} operation={} path=\"{}\" result={}",
|
||||
user_id,
|
||||
operation,
|
||||
path,
|
||||
result
|
||||
);
|
||||
}
|
||||
|
||||
pub fn log_error(&self, user_id: &str, operation: &str, path: &str, error: &str) {
|
||||
self.log_operation(user_id, operation, path, &format!("error: {}", error));
|
||||
}
|
||||
|
||||
pub fn log_success(&self, user_id: &str, operation: &str, path: &str) {
|
||||
self.log_operation(user_id, operation, path, "success");
|
||||
}
|
||||
|
||||
pub fn get_log_path(&self) -> &PathBuf {
|
||||
&self.log_path
|
||||
}
|
||||
}
|
||||
|
||||
impl Clone for AuditLog {
|
||||
fn clone(&self) -> Self {
|
||||
// Clone时重新打开文件
|
||||
Self::new(&self.log_path.to_string_lossy()).unwrap_or_else(|_| {
|
||||
// 如果失败,使用临时路径
|
||||
Self::new("/tmp/sftp_audit_fallback.log").unwrap()
|
||||
})
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user