mirror of
				https://git.proxmox.com/git/proxmox-backup
				synced 2025-10-26 19:44:31 +00:00 
			
		
		
		
	make tasklog downloadable in the backup server backend
The read_tasklog API call now stream the whole log file if the query parameter 'download' is set to true. If the limit parameter is set to 0, all lines in the tasklog will be returned in json format. To make a file stream and a json response in the same API call work, I had to use one of the lower level apimethod types from the proxmox-router. Therefore, the routing declarations and parameter schemas have been changed accordingly. Signed-off-by: Daniel Tschlatscher <d.tschlatscher@proxmox.com>
This commit is contained in:
		
							parent
							
								
									9a087ce7a2
								
							
						
					
					
						commit
						67a5999aa6
					
				| @ -2,10 +2,18 @@ use std::fs::File; | |||||||
| use std::io::{BufRead, BufReader}; | use std::io::{BufRead, BufReader}; | ||||||
| 
 | 
 | ||||||
| use anyhow::{bail, Error}; | use anyhow::{bail, Error}; | ||||||
|  | use futures::FutureExt; | ||||||
|  | use http::request::Parts; | ||||||
|  | use http::{header, Response, StatusCode}; | ||||||
|  | use hyper::Body; | ||||||
|  | use proxmox_async::stream::AsyncReaderStream; | ||||||
| use serde_json::{json, Value}; | use serde_json::{json, Value}; | ||||||
| 
 | 
 | ||||||
| use proxmox_router::{list_subdirs_api_method, Permission, Router, RpcEnvironment, SubdirMap}; | use proxmox_router::{ | ||||||
| use proxmox_schema::api; |     list_subdirs_api_method, ApiHandler, ApiMethod, ApiResponseFuture, Permission, Router, | ||||||
|  |     RpcEnvironment, SubdirMap, | ||||||
|  | }; | ||||||
|  | use proxmox_schema::{api, BooleanSchema, IntegerSchema, ObjectSchema, Schema}; | ||||||
| use proxmox_sys::sortable; | use proxmox_sys::sortable; | ||||||
| 
 | 
 | ||||||
| use pbs_api_types::{ | use pbs_api_types::{ | ||||||
| @ -19,6 +27,27 @@ use crate::api2::pull::check_pull_privs; | |||||||
| use pbs_config::CachedUserInfo; | use pbs_config::CachedUserInfo; | ||||||
| use proxmox_rest_server::{upid_log_path, upid_read_status, TaskListInfoIterator, TaskState}; | use proxmox_rest_server::{upid_log_path, upid_read_status, TaskListInfoIterator, TaskState}; | ||||||
| 
 | 
 | ||||||
|  | pub const START_PARAM_SCHEMA: Schema = | ||||||
|  |     IntegerSchema::new("Start at this line when reading the tasklog") | ||||||
|  |         .minimum(0) | ||||||
|  |         .default(0) | ||||||
|  |         .schema(); | ||||||
|  | 
 | ||||||
|  | pub const LIMIT_PARAM_SCHEMA: Schema = | ||||||
|  |     IntegerSchema::new("The amount of lines to read from the tasklog. Setting this parameter to 0 will return all lines until the end of the file.") | ||||||
|  |         .minimum(0) | ||||||
|  |         .default(50) | ||||||
|  |         .schema(); | ||||||
|  | 
 | ||||||
|  | pub const DOWNLOAD_PARAM_SCHEMA: Schema = | ||||||
|  |     BooleanSchema::new("Whether the tasklog file should be downloaded. This parameter can't be used in conjunction with other parameters") | ||||||
|  |         .default(false) | ||||||
|  |         .schema(); | ||||||
|  | 
 | ||||||
|  | pub const TEST_STATUS_PARAM_SCHEMA: Schema = | ||||||
|  |     BooleanSchema::new("Test task status, and set result attribute \"active\" accordingly.") | ||||||
|  |         .schema(); | ||||||
|  | 
 | ||||||
| // matches respective job execution privileges
 | // matches respective job execution privileges
 | ||||||
| fn check_job_privs(auth_id: &Authid, user_info: &CachedUserInfo, upid: &UPID) -> Result<(), Error> { | fn check_job_privs(auth_id: &Authid, user_info: &CachedUserInfo, upid: &UPID) -> Result<(), Error> { | ||||||
|     match (upid.worker_type.as_str(), &upid.worker_id) { |     match (upid.worker_type.as_str(), &upid.worker_id) { | ||||||
| @ -268,59 +297,67 @@ fn extract_upid(param: &Value) -> Result<UPID, Error> { | |||||||
|     upid_str.parse::<UPID>() |     upid_str.parse::<UPID>() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[api(
 | #[sortable] | ||||||
|     input: { | pub const API_METHOD_READ_TASK_LOG: ApiMethod = ApiMethod::new( | ||||||
|         properties: { |     &ApiHandler::AsyncHttp(&read_task_log), | ||||||
|             node: { |     &ObjectSchema::new( | ||||||
|                 schema: NODE_SCHEMA, |         "Read the task log", | ||||||
|             }, |         &sorted!([ | ||||||
|             upid: { |             ("node", false, &NODE_SCHEMA), | ||||||
|                 schema: UPID_SCHEMA, |             ("upid", false, &UPID_SCHEMA), | ||||||
|             }, |             ("start", true, &START_PARAM_SCHEMA), | ||||||
|             "test-status": { |             ("limit", true, &LIMIT_PARAM_SCHEMA), | ||||||
|                 type: bool, |             ("download", true, &DOWNLOAD_PARAM_SCHEMA), | ||||||
|                 optional: true, |             ("test-status", true, &TEST_STATUS_PARAM_SCHEMA) | ||||||
|                 description: "Test task status, and set result attribute \"active\" accordingly.", |         ]), | ||||||
|             }, |     ), | ||||||
|             start: { | ) | ||||||
|                 type: u64, | .access( | ||||||
|                 optional: true, |     Some("Users can access their own tasks, or need Sys.Audit on /system/tasks."), | ||||||
|                 description: "Start at this line.", |     &Permission::Anybody, | ||||||
|                 default: 0, | ); | ||||||
|             }, | fn read_task_log( | ||||||
|             limit: { |     _parts: Parts, | ||||||
|                 type: u64, |     _req_body: Body, | ||||||
|                 optional: true, |     param: Value, | ||||||
|                 description: "Only list this amount of lines.", |     _info: &ApiMethod, | ||||||
|                 default: 50, |     rpcenv: Box<dyn RpcEnvironment>, | ||||||
|             }, | ) -> ApiResponseFuture { | ||||||
|         }, |     async move { | ||||||
|     }, |         let upid: UPID = extract_upid(¶m)?; | ||||||
|     access: { |  | ||||||
|         description: "Users can access their own tasks, or need Sys.Audit on /system/tasks.", |  | ||||||
|         permission: &Permission::Anybody, |  | ||||||
|     }, |  | ||||||
| )] |  | ||||||
| /// Read task log.
 |  | ||||||
| async fn read_task_log(param: Value, rpcenv: &mut dyn RpcEnvironment) -> Result<Value, Error> { |  | ||||||
|     let upid = extract_upid(¶m)?; |  | ||||||
| 
 |  | ||||||
|         let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; |         let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; | ||||||
| 
 |  | ||||||
|         check_task_access(&auth_id, &upid)?; |         check_task_access(&auth_id, &upid)?; | ||||||
| 
 | 
 | ||||||
|     let test_status = param["test-status"].as_bool().unwrap_or(false); |         let download = param["download"].as_bool().unwrap_or(false); | ||||||
|  |         let path = upid_log_path(&upid)?; | ||||||
| 
 | 
 | ||||||
|  |         if download { | ||||||
|  |             if !param["start"].is_null() | ||||||
|  |                 || !param["limit"].is_null() | ||||||
|  |                 || !param["test-status"].is_null() | ||||||
|  |             { | ||||||
|  |                 bail!("Parameter 'download' cannot be used with other parameters"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             let header_disp = format!("attachment; filename={}", &upid.to_string()); | ||||||
|  |             let stream = AsyncReaderStream::new(tokio::fs::File::open(path).await?); | ||||||
|  | 
 | ||||||
|  |             Ok(Response::builder() | ||||||
|  |                 .status(StatusCode::OK) | ||||||
|  |                 .header(header::CONTENT_TYPE, "text/plain") | ||||||
|  |                 .header(header::CONTENT_DISPOSITION, &header_disp) | ||||||
|  |                 .body(Body::wrap_stream(stream)) | ||||||
|  |                 .unwrap()) | ||||||
|  |         } else { | ||||||
|             let start = param["start"].as_u64().unwrap_or(0); |             let start = param["start"].as_u64().unwrap_or(0); | ||||||
|             let mut limit = param["limit"].as_u64().unwrap_or(50); |             let mut limit = param["limit"].as_u64().unwrap_or(50); | ||||||
| 
 |             let test_status = param["test-status"].as_bool().unwrap_or(false); | ||||||
|     let mut count: u64 = 0; |  | ||||||
| 
 |  | ||||||
|     let path = upid_log_path(&upid)?; |  | ||||||
| 
 | 
 | ||||||
|             let file = File::open(path)?; |             let file = File::open(path)?; | ||||||
| 
 | 
 | ||||||
|  |             let mut count: u64 = 0; | ||||||
|             let mut lines: Vec<Value> = vec![]; |             let mut lines: Vec<Value> = vec![]; | ||||||
|  |             let read_until_end = if limit == 0 { true } else { false }; | ||||||
| 
 | 
 | ||||||
|             for line in BufReader::new(file).lines() { |             for line in BufReader::new(file).lines() { | ||||||
|                 match line { |                 match line { | ||||||
| @ -329,13 +366,14 @@ async fn read_task_log(param: Value, rpcenv: &mut dyn RpcEnvironment) -> Result< | |||||||
|                         if count < start { |                         if count < start { | ||||||
|                             continue; |                             continue; | ||||||
|                         }; |                         }; | ||||||
|  |                         if !read_until_end { | ||||||
|                             if limit == 0 { |                             if limit == 0 { | ||||||
|                                 continue; |                                 continue; | ||||||
|                             }; |                             }; | ||||||
|  |                             limit -= 1; | ||||||
|  |                         } | ||||||
| 
 | 
 | ||||||
|                         lines.push(json!({ "n": count, "t": line })); |                         lines.push(json!({ "n": count, "t": line })); | ||||||
| 
 |  | ||||||
|                 limit -= 1; |  | ||||||
|                     } |                     } | ||||||
|                     Err(err) => { |                     Err(err) => { | ||||||
|                         log::error!("reading task log failed: {}", err); |                         log::error!("reading task log failed: {}", err); | ||||||
| @ -344,14 +382,25 @@ async fn read_task_log(param: Value, rpcenv: &mut dyn RpcEnvironment) -> Result< | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|     rpcenv["total"] = Value::from(count); |             let mut json = json!({ | ||||||
|  |                 "data": lines, | ||||||
|  |                 "total": count, | ||||||
|  |                 "success": 1, | ||||||
|  |             }); | ||||||
| 
 | 
 | ||||||
|             if test_status { |             if test_status { | ||||||
|                 let active = proxmox_rest_server::worker_is_active(&upid).await?; |                 let active = proxmox_rest_server::worker_is_active(&upid).await?; | ||||||
|         rpcenv["active"] = Value::from(active); |                 json["test-status"] = Value::from(active); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|     Ok(json!(lines)) |             Ok(Response::builder() | ||||||
|  |                 .status(StatusCode::OK) | ||||||
|  |                 .header(header::CONTENT_TYPE, "application/json") | ||||||
|  |                 .body(Body::from(json.to_string())) | ||||||
|  |                 .unwrap()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     .boxed() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[api(
 | #[api(
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Daniel Tschlatscher
						Daniel Tschlatscher