// Copyright 2019-2023 Tauri Programme within The Commons Conservancy // SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: MIT //! [![](https://github.com/tauri-apps/plugins-workspace/raw/v2/plugins/upload/banner.png)](https://github.com/tauri-apps/plugins-workspace/tree/v2/plugins/upload) //! //! Upload files from disk to a remote server over HTTP. //! //! Download files from a remote HTTP server to disk. #![doc( html_logo_url = "https://github.com/tauri-apps/tauri/raw/dev/app-icon.png", html_favicon_url = "https://github.com/tauri-apps/tauri/raw/dev/app-icon.png" )] use futures_util::TryStreamExt; use serde::{ser::Serializer, Serialize}; use tauri::{ command, ipc::Channel, plugin::{Builder as PluginBuilder, TauriPlugin}, Runtime, }; use tokio::{ fs::File, io::{AsyncWriteExt, BufWriter}, }; use tokio_util::codec::{BytesCodec, FramedRead}; use read_progress_stream::ReadProgressStream; use std::collections::HashMap; type Result = std::result::Result; #[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(&self, serializer: S) -> std::result::Result where S: Serializer, { serializer.serialize_str(self.to_string().as_ref()) } } #[derive(Clone, Serialize)] struct ProgressPayload { progress: u64, total: u64, } #[command] async fn download( url: &str, file_path: &str, headers: HashMap, on_progress: Channel, ) -> Result<()> { 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 _ = on_progress.send(&ProgressPayload { progress: chunk.len() as u64, total, }); } file.flush().await?; Ok(()) } #[command] async fn upload( url: &str, file_path: &str, headers: HashMap, on_progress: Channel, ) -> Result { // 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(on_progress, 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(channel: Channel, file: File) -> reqwest::Body { let stream = FramedRead::new(file, BytesCodec::new()).map_ok(|r| r.freeze()); reqwest::Body::wrap_stream(ReadProgressStream::new( stream, Box::new(move |progress, total| { let _ = channel.send(ProgressPayload { progress, total }); }), )) } pub fn init() -> TauriPlugin { PluginBuilder::new("upload") .invoke_handler(tauri::generate_handler![download, upload]) .build() }