use openlark_core::{
api::{ApiRequest, ApiResponseTrait, ResponseFormat},
config::Config,
http::Transport,
SDKResult,
};
use serde::{Deserialize, Serialize};
use crate::common::{api_endpoints::DriveApi, api_utils::*};
#[derive(Debug, Clone)]
pub struct BatchGetTmpDownloadUrlRequest {
config: Config,
pub file_tokens: Vec<String>,
pub extra: Option<String>,
}
impl BatchGetTmpDownloadUrlRequest {
pub fn new(config: Config, file_tokens: Vec<String>) -> Self {
Self {
config,
file_tokens,
extra: None,
}
}
pub fn add_file_token(mut self, file_token: impl Into<String>) -> Self {
self.file_tokens.push(file_token.into());
self
}
pub fn extra(mut self, extra: impl Into<String>) -> Self {
self.extra = Some(extra.into());
self
}
pub async fn execute(self) -> SDKResult<BatchGetTmpDownloadUrlResponse> {
self.execute_with_options(openlark_core::req_option::RequestOption::default())
.await
}
pub async fn execute_with_options(
self,
option: openlark_core::req_option::RequestOption,
) -> SDKResult<BatchGetTmpDownloadUrlResponse> {
if self.file_tokens.is_empty() {
return Err(openlark_core::error::validation_error(
"file_tokens",
"file_tokens 不能为空",
));
}
if self.file_tokens.len() > 5 {
return Err(openlark_core::error::validation_error(
"file_tokens",
"file_tokens 一次最多传 5 个",
));
}
let api_endpoint = DriveApi::GetMediaTempDownloadUrls;
let mut query_pairs: Vec<String> = Vec::new();
for token in &self.file_tokens {
if token.is_empty() {
return Err(openlark_core::error::validation_error(
"file_tokens",
"file_tokens 不能包含空值",
));
}
query_pairs.push(format!("file_tokens={}", urlencoding::encode(token)));
}
if let Some(extra) = &self.extra {
query_pairs.push(format!("extra={}", urlencoding::encode(extra)));
}
let url = format!("{}?{}", api_endpoint.to_url(), query_pairs.join("&"));
let request = ApiRequest::<BatchGetTmpDownloadUrlResponse>::get(url);
let response = Transport::request(request, &self.config, Some(option)).await?;
extract_response_data(response, "获取")
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct TmpDownloadUrlInfo {
pub file_token: String,
pub tmp_download_url: String,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct BatchGetTmpDownloadUrlResponse {
#[serde(default)]
pub tmp_download_urls: Vec<TmpDownloadUrlInfo>,
}
impl ApiResponseTrait for BatchGetTmpDownloadUrlResponse {
fn data_format() -> ResponseFormat {
ResponseFormat::Data
}
}
#[cfg(test)]
mod tests {
use super::*;
use openlark_core::testing::prelude::test_runtime;
#[test]
fn test_batch_get_tmp_download_url_request_builder() {
let config = Config::default();
let request = BatchGetTmpDownloadUrlRequest::new(
config,
vec!["file_token_1".to_string(), "file_token_2".to_string()],
)
.extra("extra");
assert_eq!(request.file_tokens.len(), 2);
assert_eq!(request.file_tokens[0], "file_token_1");
assert_eq!(request.extra, Some("extra".to_string()));
}
#[test]
fn test_batch_get_tmp_download_url_request_add_token() {
let config = Config::default();
let request = BatchGetTmpDownloadUrlRequest::new(config, vec!["file_token_1".to_string()])
.add_file_token("file_token_2");
assert_eq!(request.file_tokens.len(), 2);
assert_eq!(request.file_tokens[1], "file_token_2");
}
#[test]
fn test_response_trait() {
assert_eq!(
BatchGetTmpDownloadUrlResponse::data_format(),
ResponseFormat::Data
);
}
#[test]
fn test_empty_file_tokens_validation() {
let config = Config::default();
let request = BatchGetTmpDownloadUrlRequest::new(config, vec![]);
let result = std::thread::spawn(move || {
let rt = test_runtime();
rt.block_on(async move {
let _ = request.execute().await;
})
})
.join();
assert!(result.is_ok());
}
#[test]
fn test_file_tokens_count_validation() {
let config = Config::default();
let tokens_5: Vec<String> = ["a"; 5].iter().map(|s| s.to_string()).collect();
let request1 = BatchGetTmpDownloadUrlRequest::new(config.clone(), tokens_5);
assert_eq!(request1.file_tokens.len(), 5);
let tokens_6: Vec<String> = ["a"; 6].iter().map(|s| s.to_string()).collect();
let request2 = BatchGetTmpDownloadUrlRequest::new(config, tokens_6);
let result = std::thread::spawn(move || {
let rt = test_runtime();
rt.block_on(async move {
let _ = request2.execute().await;
})
})
.join();
assert!(result.is_ok());
}
#[test]
fn test_empty_token_in_list_validation() {
let config = Config::default();
let request = BatchGetTmpDownloadUrlRequest::new(
config,
vec!["valid_token".to_string(), "".to_string()],
);
let result = std::thread::spawn(move || {
let rt = test_runtime();
rt.block_on(async move {
let _ = request.execute().await;
})
})
.join();
assert!(result.is_ok());
}
#[test]
fn test_single_token() {
let config = Config::default();
let request = BatchGetTmpDownloadUrlRequest::new(config, vec!["single_token".to_string()]);
assert_eq!(request.file_tokens.len(), 1);
assert_eq!(request.file_tokens[0], "single_token");
}
#[test]
fn test_extra_optional() {
let config = Config::default();
let request1 =
BatchGetTmpDownloadUrlRequest::new(config.clone(), vec!["token".to_string()]);
assert!(request1.extra.is_none());
let request2 = BatchGetTmpDownloadUrlRequest::new(config, vec!["token".to_string()])
.extra("extra_param");
assert_eq!(request2.extra, Some("extra_param".to_string()));
}
}