rqbit/desktop/src-tauri/src/main.rs

194 lines
5.4 KiB
Rust
Raw Normal View History

2023-12-02 22:24:36 +00:00
// Prevents additional console window on Windows in release, DO NOT REMOVE!!
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
2023-12-03 09:49:10 +00:00
use anyhow::Context;
2023-12-02 22:24:36 +00:00
use http::StatusCode;
use librqbit::{
api::{
2023-12-03 12:14:50 +00:00
ApiAddTorrentResponse, EmptyJsonResponse, TorrentDetailsResponse, TorrentListResponse,
TorrentStats,
2023-12-02 22:24:36 +00:00
},
2023-12-06 12:14:26 +00:00
librqbit_spawn, AddTorrent, AddTorrentOptions, Api, ApiError, Session, SessionOptions,
2023-12-02 22:24:36 +00:00
};
2023-12-06 12:14:26 +00:00
use tracing::error_span;
2023-12-02 22:24:36 +00:00
struct State {
api: Api,
}
#[tauri::command]
fn torrents_list(state: tauri::State<State>) -> TorrentListResponse {
state.api.api_torrent_list()
}
#[tauri::command]
async fn torrent_create_from_url(
state: tauri::State<'_, State>,
url: String,
opts: Option<AddTorrentOptions>,
) -> Result<ApiAddTorrentResponse, ApiError> {
state
.api
2023-12-03 12:14:50 +00:00
.api_add_torrent(AddTorrent::Url(url.into()), opts)
2023-12-02 22:24:36 +00:00
.await
}
#[tauri::command]
async fn torrent_create_from_base64_file(
state: tauri::State<'_, State>,
contents: String,
opts: Option<AddTorrentOptions>,
) -> Result<ApiAddTorrentResponse, ApiError> {
use base64::{engine::general_purpose, Engine as _};
2023-12-03 09:49:10 +00:00
let bytes = general_purpose::STANDARD
2023-12-02 22:24:36 +00:00
.decode(&contents)
2023-12-03 09:49:10 +00:00
.context("invalid base64")
.map_err(|e| ApiError::new_from_anyhow(StatusCode::BAD_REQUEST, e))?;
2023-12-02 22:24:36 +00:00
state
.api
2023-12-03 12:14:50 +00:00
.api_add_torrent(AddTorrent::TorrentFileBytes(bytes.into()), opts)
2023-12-02 22:24:36 +00:00
.await
}
#[tauri::command]
async fn torrent_details(
state: tauri::State<'_, State>,
id: usize,
) -> Result<TorrentDetailsResponse, ApiError> {
state.api.api_torrent_details(id)
}
#[tauri::command]
async fn torrent_stats(
state: tauri::State<'_, State>,
id: usize,
) -> Result<TorrentStats, ApiError> {
state.api.api_stats_v1(id)
}
#[tauri::command]
async fn torrent_action_delete(
state: tauri::State<'_, State>,
id: usize,
) -> Result<EmptyJsonResponse, ApiError> {
state.api.api_torrent_action_delete(id)
}
#[tauri::command]
async fn torrent_action_pause(
state: tauri::State<'_, State>,
id: usize,
) -> Result<EmptyJsonResponse, ApiError> {
state.api.api_torrent_action_pause(id)
}
#[tauri::command]
async fn torrent_action_forget(
state: tauri::State<'_, State>,
id: usize,
) -> Result<EmptyJsonResponse, ApiError> {
state.api.api_torrent_action_forget(id)
}
#[tauri::command]
async fn torrent_action_start(
state: tauri::State<'_, State>,
id: usize,
) -> Result<EmptyJsonResponse, ApiError> {
state.api.api_torrent_action_start(id)
}
2023-12-06 12:14:26 +00:00
#[tauri::command]
fn get_version() -> &'static str {
env!("CARGO_PKG_VERSION")
}
fn init_logging() -> tokio::sync::mpsc::UnboundedSender<String> {
use tracing_subscriber::{fmt, prelude::*, EnvFilter};
let (stderr_filter, reload_stderr_filter) =
tracing_subscriber::reload::Layer::new(EnvFilter::builder().parse("info").unwrap());
let layered = tracing_subscriber::registry().with(fmt::layer().with_filter(stderr_filter));
layered.init();
let (reload_tx, mut reload_rx) = tokio::sync::mpsc::unbounded_channel::<String>();
librqbit_spawn(
"fmt_filter_reloader",
error_span!("fmt_filter_reloader"),
async move {
while let Some(rust_log) = reload_rx.recv().await {
let stderr_env_filter = match EnvFilter::builder().parse(&rust_log) {
Ok(f) => f,
Err(e) => {
eprintln!("can't parse env filter {:?}: {:#?}", rust_log, e);
continue;
}
};
eprintln!("setting RUST_LOG to {:?}", rust_log);
let _ = reload_stderr_filter.reload(stderr_env_filter);
}
Ok(())
},
);
reload_tx
}
2023-12-02 22:24:36 +00:00
async fn start_session() {
2023-12-06 12:14:26 +00:00
let rust_log_reload_tx = init_logging();
2023-12-02 22:24:36 +00:00
tauri::async_runtime::set(tokio::runtime::Handle::current());
let download_folder = directories::UserDirs::new()
.expect("directories::UserDirs::new()")
.download_dir()
.expect("download_dir()")
.to_path_buf();
2023-12-03 12:14:50 +00:00
let session = Session::new_with_opts(
2023-12-02 22:24:36 +00:00
download_folder,
2023-12-03 12:14:50 +00:00
SessionOptions {
2023-12-02 22:24:36 +00:00
disable_dht: false,
disable_dht_persistence: false,
persistence: true,
2023-12-06 01:28:43 +00:00
listen_port_range: Some(4240..4260),
2023-12-02 22:24:36 +00:00
..Default::default()
},
)
.await
.expect("couldn't set up librqbit session");
2023-12-06 12:14:26 +00:00
let api = Api::new(session.clone(), None);
librqbit_spawn(
"http api",
error_span!("http_api"),
librqbit::http_api::HttpApi::new(session, Some(rust_log_reload_tx))
2023-12-06 12:20:19 +00:00
.make_http_api_and_run("127.0.0.1:3030".parse().unwrap(), false),
2023-12-06 12:14:26 +00:00
);
2023-12-02 22:24:36 +00:00
tauri::Builder::default()
.manage(State { api })
.invoke_handler(tauri::generate_handler![
torrents_list,
torrent_details,
torrent_stats,
torrent_create_from_url,
torrent_action_delete,
torrent_action_pause,
torrent_action_forget,
torrent_action_start,
torrent_create_from_base64_file,
2023-12-06 12:14:26 +00:00
get_version
2023-12-02 22:24:36 +00:00
])
.run(tauri::generate_context!())
.expect("error while running tauri application");
}
fn main() {
tokio::runtime::Builder::new_multi_thread()
.enable_all()
.build()
.expect("couldn't set up tokio runtime")
.block_on(start_session())
}