mirror of
https://github.com/mofeng-git/One-KVM.git
synced 2026-06-14 03:32:00 +08:00
feat: 初步增加 Windows 支持
This commit is contained in:
405
src/web/handlers/msd_api.rs
Normal file
405
src/web/handlers/msd_api.rs
Normal file
@@ -0,0 +1,405 @@
|
||||
use super::*;
|
||||
|
||||
use crate::msd::{
|
||||
DownloadProgress, DriveFile, DriveInfo, DriveInitRequest, ImageDownloadRequest, ImageInfo,
|
||||
ImageManager, MsdConnectRequest, MsdMode, MsdState, VentoyDrive,
|
||||
};
|
||||
#[cfg(unix)]
|
||||
use axum::extract::{Multipart, Path as AxumPath};
|
||||
#[cfg(unix)]
|
||||
use std::collections::HashMap;
|
||||
|
||||
/// MSD status response
|
||||
#[cfg(unix)]
|
||||
#[derive(Serialize)]
|
||||
pub struct MsdStatus {
|
||||
pub available: bool,
|
||||
pub state: MsdState,
|
||||
}
|
||||
|
||||
/// Get MSD status
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_status(State(state): State<Arc<AppState>>) -> Result<Json<MsdStatus>> {
|
||||
let msd_guard = state.msd.read().await;
|
||||
match msd_guard.as_ref() {
|
||||
Some(controller) => {
|
||||
let msd_state = controller.state().await;
|
||||
Ok(Json(MsdStatus {
|
||||
available: true,
|
||||
state: msd_state,
|
||||
}))
|
||||
}
|
||||
None => Ok(Json(MsdStatus {
|
||||
available: false,
|
||||
state: MsdState::default(),
|
||||
})),
|
||||
}
|
||||
}
|
||||
|
||||
/// List all available images
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_images_list(State(state): State<Arc<AppState>>) -> Result<Json<Vec<ImageInfo>>> {
|
||||
let config = state.config.get();
|
||||
let images_path = config.msd.images_dir();
|
||||
let manager = ImageManager::new(images_path);
|
||||
|
||||
let images = manager.list()?;
|
||||
Ok(Json(images))
|
||||
}
|
||||
|
||||
/// Upload new image (streaming - memory efficient for large files)
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_image_upload(
|
||||
State(state): State<Arc<AppState>>,
|
||||
mut multipart: Multipart,
|
||||
) -> Result<Json<ImageInfo>> {
|
||||
let config = state.config.get();
|
||||
let images_path = config.msd.images_dir();
|
||||
let manager = ImageManager::new(images_path);
|
||||
|
||||
while let Some(field) = multipart
|
||||
.next_field()
|
||||
.await
|
||||
.map_err(|e| AppError::Internal(format!("Multipart error: {}", e)))?
|
||||
{
|
||||
let name = field.name().unwrap_or("file").to_string();
|
||||
if name == "file" {
|
||||
let filename = field
|
||||
.file_name()
|
||||
.ok_or_else(|| AppError::BadRequest("Missing filename".to_string()))?
|
||||
.to_string();
|
||||
|
||||
// Use streaming upload - chunks are written directly to disk
|
||||
// This avoids loading the entire file into memory
|
||||
let image = manager
|
||||
.create_from_multipart_field(&filename, field)
|
||||
.await?;
|
||||
return Ok(Json(image));
|
||||
}
|
||||
}
|
||||
|
||||
Err(AppError::BadRequest("No file provided".to_string()))
|
||||
}
|
||||
|
||||
/// Get image by ID
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_image_get(
|
||||
State(state): State<Arc<AppState>>,
|
||||
AxumPath(id): AxumPath<String>,
|
||||
) -> Result<Json<ImageInfo>> {
|
||||
let config = state.config.get();
|
||||
let images_path = config.msd.images_dir();
|
||||
let manager = ImageManager::new(images_path);
|
||||
|
||||
let image = manager.get(&id)?;
|
||||
Ok(Json(image))
|
||||
}
|
||||
|
||||
/// Delete image by ID
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_image_delete(
|
||||
State(state): State<Arc<AppState>>,
|
||||
AxumPath(id): AxumPath<String>,
|
||||
) -> Result<Json<LoginResponse>> {
|
||||
let config = state.config.get();
|
||||
let images_path = config.msd.images_dir();
|
||||
let manager = ImageManager::new(images_path);
|
||||
|
||||
manager.delete(&id)?;
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some("Image deleted".to_string()),
|
||||
}))
|
||||
}
|
||||
|
||||
/// Download image from URL
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_image_download(
|
||||
State(state): State<Arc<AppState>>,
|
||||
Json(req): Json<ImageDownloadRequest>,
|
||||
) -> Result<Json<DownloadProgress>> {
|
||||
let msd_guard = state.msd.read().await;
|
||||
let controller = msd_guard
|
||||
.as_ref()
|
||||
.ok_or_else(|| AppError::Internal("MSD not initialized".to_string()))?;
|
||||
|
||||
let progress = controller.download_image(req.url, req.filename).await?;
|
||||
|
||||
Ok(Json(progress))
|
||||
}
|
||||
|
||||
/// Cancel image download
|
||||
#[cfg(unix)]
|
||||
#[derive(serde::Deserialize)]
|
||||
pub struct CancelDownloadRequest {
|
||||
pub download_id: String,
|
||||
}
|
||||
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_image_download_cancel(
|
||||
State(state): State<Arc<AppState>>,
|
||||
Json(req): Json<CancelDownloadRequest>,
|
||||
) -> Result<Json<LoginResponse>> {
|
||||
let msd_guard = state.msd.read().await;
|
||||
let controller = msd_guard
|
||||
.as_ref()
|
||||
.ok_or_else(|| AppError::Internal("MSD not initialized".to_string()))?;
|
||||
|
||||
controller.cancel_download(&req.download_id).await?;
|
||||
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some("Download cancelled".to_string()),
|
||||
}))
|
||||
}
|
||||
|
||||
/// Connect MSD (image or drive)
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_connect(
|
||||
State(state): State<Arc<AppState>>,
|
||||
Json(req): Json<MsdConnectRequest>,
|
||||
) -> Result<Json<LoginResponse>> {
|
||||
let config = state.config.get();
|
||||
let mut msd_guard = state.msd.write().await;
|
||||
let controller = msd_guard
|
||||
.as_mut()
|
||||
.ok_or_else(|| AppError::Internal("MSD not initialized".to_string()))?;
|
||||
|
||||
match req.mode {
|
||||
MsdMode::Image => {
|
||||
let image_id = req.image_id.ok_or_else(|| {
|
||||
AppError::BadRequest("image_id required for image mode".to_string())
|
||||
})?;
|
||||
|
||||
// Get image info from ImageManager
|
||||
let images_path = config.msd.images_dir();
|
||||
let manager = ImageManager::new(images_path);
|
||||
let image = manager.get(&image_id)?;
|
||||
|
||||
// Get mount options from request (defaults: cdrom=false, read_only=false)
|
||||
let cdrom = req.cdrom.unwrap_or(false);
|
||||
let read_only = req.read_only.unwrap_or(false);
|
||||
|
||||
controller.connect_image(&image, cdrom, read_only).await?;
|
||||
}
|
||||
MsdMode::Drive => {
|
||||
controller.connect_drive().await?;
|
||||
}
|
||||
MsdMode::None => {
|
||||
return Err(AppError::BadRequest("Invalid mode: none".to_string()));
|
||||
}
|
||||
}
|
||||
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some("MSD connected".to_string()),
|
||||
}))
|
||||
}
|
||||
|
||||
/// Disconnect MSD
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_disconnect(State(state): State<Arc<AppState>>) -> Result<Json<LoginResponse>> {
|
||||
let mut msd_guard = state.msd.write().await;
|
||||
let controller = msd_guard
|
||||
.as_mut()
|
||||
.ok_or_else(|| AppError::Internal("MSD not initialized".to_string()))?;
|
||||
|
||||
controller.disconnect().await?;
|
||||
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some("MSD disconnected".to_string()),
|
||||
}))
|
||||
}
|
||||
|
||||
/// Get drive info
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_info(State(state): State<Arc<AppState>>) -> Result<Json<DriveInfo>> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
if !drive.exists() {
|
||||
return Err(AppError::NotFound("Drive not initialized".to_string()));
|
||||
}
|
||||
|
||||
let info = drive.info().await?;
|
||||
Ok(Json(info))
|
||||
}
|
||||
|
||||
/// Initialize Ventoy drive
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_init(
|
||||
State(state): State<Arc<AppState>>,
|
||||
Json(req): Json<DriveInitRequest>,
|
||||
) -> Result<Json<DriveInfo>> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
let info = drive.init(req.size_mb).await?;
|
||||
Ok(Json(info))
|
||||
}
|
||||
|
||||
/// Delete virtual drive
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_delete(State(state): State<Arc<AppState>>) -> Result<Json<LoginResponse>> {
|
||||
let config = state.config.get();
|
||||
|
||||
// Check if drive is currently connected
|
||||
let msd_guard = state.msd.write().await;
|
||||
if let Some(controller) = msd_guard.as_ref() {
|
||||
let msd_state = controller.state().await;
|
||||
if msd_state.connected && msd_state.mode == crate::msd::types::MsdMode::Drive {
|
||||
return Err(AppError::BadRequest(
|
||||
"Cannot delete drive while connected. Disconnect first.".to_string(),
|
||||
));
|
||||
}
|
||||
}
|
||||
drop(msd_guard);
|
||||
|
||||
// Delete the drive file
|
||||
let drive_path = config.msd.drive_path();
|
||||
if drive_path.exists() {
|
||||
std::fs::remove_file(&drive_path)
|
||||
.map_err(|e| AppError::Internal(format!("Failed to delete drive file: {}", e)))?;
|
||||
}
|
||||
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some("Virtual drive deleted".to_string()),
|
||||
}))
|
||||
}
|
||||
|
||||
/// List drive files
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_files(
|
||||
State(state): State<Arc<AppState>>,
|
||||
Query(params): Query<HashMap<String, String>>,
|
||||
) -> Result<Json<Vec<DriveFile>>> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
let dir_path = params.get("path").map(|s| s.as_str()).unwrap_or("/");
|
||||
let files = drive.list_files(dir_path).await?;
|
||||
Ok(Json(files))
|
||||
}
|
||||
|
||||
/// Upload file to drive (streaming - memory efficient for large files)
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_upload(
|
||||
State(state): State<Arc<AppState>>,
|
||||
Query(params): Query<HashMap<String, String>>,
|
||||
mut multipart: Multipart,
|
||||
) -> Result<Json<LoginResponse>> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
let target_dir = params.get("path").map(|s| s.as_str()).unwrap_or("/");
|
||||
|
||||
while let Some(field) = multipart
|
||||
.next_field()
|
||||
.await
|
||||
.map_err(|e| AppError::Internal(format!("Multipart error: {}", e)))?
|
||||
{
|
||||
let name = field.name().unwrap_or("file").to_string();
|
||||
if name == "file" {
|
||||
let filename = field
|
||||
.file_name()
|
||||
.ok_or_else(|| AppError::BadRequest("Missing filename".to_string()))?
|
||||
.to_string();
|
||||
|
||||
let file_path = if target_dir == "/" {
|
||||
format!("/{}", filename)
|
||||
} else {
|
||||
format!("{}/{}", target_dir.trim_end_matches('/'), filename)
|
||||
};
|
||||
|
||||
// Use streaming upload - chunks are written directly to disk
|
||||
// This avoids loading the entire file into memory
|
||||
drive
|
||||
.write_file_from_multipart_field(&file_path, field)
|
||||
.await?;
|
||||
|
||||
return Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some(format!("File uploaded: {}", file_path)),
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
Err(AppError::BadRequest("No file provided".to_string()))
|
||||
}
|
||||
|
||||
/// Download file from drive (streaming for large files)
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_download(
|
||||
State(state): State<Arc<AppState>>,
|
||||
AxumPath(file_path): AxumPath<String>,
|
||||
) -> Result<Response> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
// Get file stream (returns file size and channel receiver)
|
||||
let (file_size, mut rx) = drive.read_file_stream(&file_path).await?;
|
||||
|
||||
// Extract filename for Content-Disposition
|
||||
let filename = file_path.split('/').next_back().unwrap_or("download");
|
||||
|
||||
// Create a stream from the channel receiver
|
||||
let body_stream = async_stream::stream! {
|
||||
while let Some(chunk) = rx.recv().await {
|
||||
yield chunk;
|
||||
}
|
||||
};
|
||||
|
||||
Ok(Response::builder()
|
||||
.status(StatusCode::OK)
|
||||
.header(header::CONTENT_TYPE, "application/octet-stream")
|
||||
.header(header::CONTENT_LENGTH, file_size)
|
||||
.header(
|
||||
header::CONTENT_DISPOSITION,
|
||||
format!("attachment; filename=\"{}\"", filename),
|
||||
)
|
||||
.body(Body::from_stream(body_stream))
|
||||
.unwrap())
|
||||
}
|
||||
|
||||
/// Delete file from drive
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_file_delete(
|
||||
State(state): State<Arc<AppState>>,
|
||||
AxumPath(file_path): AxumPath<String>,
|
||||
) -> Result<Json<LoginResponse>> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
drive.delete(&file_path).await?;
|
||||
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some(format!("Deleted: {}", file_path)),
|
||||
}))
|
||||
}
|
||||
|
||||
/// Create directory in drive
|
||||
#[cfg(unix)]
|
||||
pub async fn msd_drive_mkdir(
|
||||
State(state): State<Arc<AppState>>,
|
||||
AxumPath(dir_path): AxumPath<String>,
|
||||
) -> Result<Json<LoginResponse>> {
|
||||
let config = state.config.get();
|
||||
let drive_path = config.msd.drive_path();
|
||||
let drive = VentoyDrive::new(drive_path);
|
||||
|
||||
drive.mkdir(&dir_path).await?;
|
||||
|
||||
Ok(Json(LoginResponse {
|
||||
success: true,
|
||||
message: Some(format!("Directory created: {}", dir_path)),
|
||||
}))
|
||||
}
|
||||
Reference in New Issue
Block a user