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))
|
|
|
|
|
.make_http_api_and_run("127.0.0.1:3000".parse().unwrap(), false),
|
|
|
|
|
);
|
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())
|
|
|
|
|
}
|