Files
One-KVM/src/web/handlers/msd_api.rs
2026-05-18 15:23:42 +00:00

412 lines
12 KiB
Rust

use super::*;
use crate::msd::{
DownloadProgress, DriveFile, DriveInfo, DriveInitRequest, ImageDownloadRequest, ImageInfo,
ImageManager, MsdConnectRequest, MsdMode, MsdState, VentoyDrive,
};
#[cfg(unix)]
use axum::body::Body;
#[cfg(unix)]
use axum::extract::{Multipart, Path as AxumPath};
#[cfg(unix)]
use axum::http::{header, StatusCode};
#[cfg(unix)]
use axum::response::Response;
#[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)),
}))
}