mirror of
https://github.com/tauri-apps/plugins-workspace.git
synced 2026-04-21 11:26:15 +02:00
4a5ab18a22
* fix(upload): return type on POST is now string A POST to a webserver can not always be expected to be a JSON response. Success is now determined by the HTTP return code. Upon success the body content is returned as a string. * feat: add content-length on POST Not all embedded devices are acceptable to receiving unspecified amounts of data. Appending the content-length up front helps this devices succeed. * fix: return values unified The return values was not used. On POST the HTTP error code is returned as an enum. * fix: upload, return value as string * Update plugins/upload/src/lib.rs Co-authored-by: Lucas Fernandes Nogueira <lucas@tauri.app> * Update plugins/upload/src/lib.rs Co-authored-by: Lucas Fernandes Nogueira <lucas@tauri.app> * fix: added covector changelog file --------- Co-authored-by: Lucas Fernandes Nogueira <lucas@tauri.app>
150 lines
3.9 KiB
Rust
150 lines
3.9 KiB
Rust
// Copyright 2021 Tauri Programme within The Commons Conservancy
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
use futures_util::TryStreamExt;
|
|
use serde::{ser::Serializer, Serialize};
|
|
use tauri::{
|
|
command,
|
|
plugin::{Builder as PluginBuilder, TauriPlugin},
|
|
Runtime, Window,
|
|
};
|
|
use tokio::{
|
|
fs::File,
|
|
io::{AsyncWriteExt, BufWriter},
|
|
};
|
|
use tokio_util::codec::{BytesCodec, FramedRead};
|
|
|
|
use read_progress_stream::ReadProgressStream;
|
|
|
|
use std::{collections::HashMap, sync::Mutex};
|
|
|
|
type Result<T> = std::result::Result<T, Error>;
|
|
|
|
#[derive(Debug, thiserror::Error)]
|
|
pub enum Error {
|
|
#[error(transparent)]
|
|
Io(#[from] std::io::Error),
|
|
#[error(transparent)]
|
|
Request(#[from] reqwest::Error),
|
|
#[error("{0}")]
|
|
ContentLength(String),
|
|
#[error("request failed with status code {0}: {1}")]
|
|
HttpErrorCode(u16, String),
|
|
}
|
|
|
|
impl Serialize for Error {
|
|
fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
|
|
where
|
|
S: Serializer,
|
|
{
|
|
serializer.serialize_str(self.to_string().as_ref())
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Serialize)]
|
|
struct ProgressPayload {
|
|
id: u32,
|
|
progress: u64,
|
|
total: u64,
|
|
}
|
|
|
|
#[command]
|
|
async fn download<R: Runtime>(
|
|
window: Window<R>,
|
|
id: u32,
|
|
url: &str,
|
|
file_path: &str,
|
|
headers: HashMap<String, String>,
|
|
) -> Result<u32> {
|
|
let client = reqwest::Client::new();
|
|
|
|
let mut request = client.get(url);
|
|
// Loop trought the headers keys and values
|
|
// and add them to the request object.
|
|
for (key, value) in headers {
|
|
request = request.header(&key, value);
|
|
}
|
|
|
|
let response = request.send().await?;
|
|
let total = response.content_length().unwrap_or(0);
|
|
|
|
let mut file = BufWriter::new(File::create(file_path).await?);
|
|
let mut stream = response.bytes_stream();
|
|
|
|
while let Some(chunk) = stream.try_next().await? {
|
|
file.write_all(&chunk).await?;
|
|
let _ = window.emit(
|
|
"download://progress",
|
|
ProgressPayload {
|
|
id,
|
|
progress: chunk.len() as u64,
|
|
total,
|
|
},
|
|
);
|
|
}
|
|
file.flush().await?;
|
|
|
|
Ok(id)
|
|
}
|
|
|
|
#[command]
|
|
async fn upload<R: Runtime>(
|
|
window: Window<R>,
|
|
id: u32,
|
|
url: &str,
|
|
file_path: &str,
|
|
headers: HashMap<String, String>,
|
|
) -> Result<String> {
|
|
// Read the file
|
|
let file = File::open(file_path).await?;
|
|
let file_len = file.metadata().await.unwrap().len();
|
|
|
|
// Create the request and attach the file to the body
|
|
let client = reqwest::Client::new();
|
|
let mut request = client
|
|
.post(url)
|
|
.header(reqwest::header::CONTENT_LENGTH, file_len)
|
|
.body(file_to_body(id, window, file));
|
|
|
|
// Loop trought the headers keys and values
|
|
// and add them to the request object.
|
|
for (key, value) in headers {
|
|
request = request.header(&key, value);
|
|
}
|
|
|
|
let response = request.send().await?;
|
|
if response.status().is_success() {
|
|
response.text().await.map_err(Into::into)
|
|
} else {
|
|
Err(Error::HttpErrorCode(
|
|
response.status().as_u16(),
|
|
response.text().await.unwrap_or_default(),
|
|
))
|
|
}
|
|
}
|
|
|
|
fn file_to_body<R: Runtime>(id: u32, window: Window<R>, file: File) -> reqwest::Body {
|
|
let stream = FramedRead::new(file, BytesCodec::new()).map_ok(|r| r.freeze());
|
|
let window = Mutex::new(window);
|
|
reqwest::Body::wrap_stream(ReadProgressStream::new(
|
|
stream,
|
|
Box::new(move |progress, total| {
|
|
let _ = window.lock().unwrap().emit(
|
|
"upload://progress",
|
|
ProgressPayload {
|
|
id,
|
|
progress,
|
|
total,
|
|
},
|
|
);
|
|
}),
|
|
))
|
|
}
|
|
|
|
pub fn init<R: Runtime>() -> TauriPlugin<R> {
|
|
PluginBuilder::new("upload")
|
|
.invoke_handler(tauri::generate_handler![download, upload])
|
|
.build()
|
|
}
|