mirror of
https://github.com/Drop-OSS/drop-app.git
synced 2025-11-10 04:22:13 +10:00
Device code authorization (#83)
* feat: device code authorization * Fix for setup executable unable to be launched (#81) * Fix for redownload invalid chunks (#84) * feat: Redownloading invalid chunks Signed-off-by: quexeky <git@quexeky.dev> * fix: clippy * fix: clippy x2 --------- Signed-off-by: quexeky <git@quexeky.dev> Co-authored-by: quexeky <git@quexeky.dev> * chore: Run clippy fix pedantic Signed-off-by: quexeky <git@quexeky.dev> * feat: add better error handling * fix: clippy --------- Signed-off-by: quexeky <git@quexeky.dev> Co-authored-by: quexeky <git@quexeky.dev>
This commit is contained in:
@ -13,11 +13,7 @@
|
|||||||
<div class="max-w-lg">
|
<div class="max-w-lg">
|
||||||
<slot />
|
<slot />
|
||||||
<div class="mt-10">
|
<div class="mt-10">
|
||||||
<button
|
<div>
|
||||||
@click="() => authWrapper_wrapper()"
|
|
||||||
:disabled="loading"
|
|
||||||
class="text-sm text-left font-semibold leading-7 text-blue-600"
|
|
||||||
>
|
|
||||||
<div v-if="loading" role="status">
|
<div v-if="loading" role="status">
|
||||||
<svg
|
<svg
|
||||||
aria-hidden="true"
|
aria-hidden="true"
|
||||||
@ -37,10 +33,19 @@
|
|||||||
</svg>
|
</svg>
|
||||||
<span class="sr-only">Loading...</span>
|
<span class="sr-only">Loading...</span>
|
||||||
</div>
|
</div>
|
||||||
<span v-else>
|
<span class="inline-flex gap-x-8 items-center" v-else>
|
||||||
Sign in with your browser <span aria-hidden="true">→</span>
|
<button
|
||||||
|
@click="() => authWrapper_wrapper()"
|
||||||
|
:disabled="loading"
|
||||||
|
class="px-3 py-1 inline-flex items-center gap-x-2 bg-zinc-700 rounded text-sm text-left font-semibold leading-7 text-white"
|
||||||
|
>
|
||||||
|
Sign in with your browser <ArrowTopRightOnSquareIcon class="size-4" />
|
||||||
|
</button>
|
||||||
|
<NuxtLink href="/auth/code" class="text-zinc-100 text-sm hover:text-zinc-300">
|
||||||
|
Use a code →
|
||||||
|
</NuxtLink>
|
||||||
</span>
|
</span>
|
||||||
</button>
|
</div>
|
||||||
|
|
||||||
<div class="mt-5" v-if="offerManual">
|
<div class="mt-5" v-if="offerManual">
|
||||||
<h1 class="text-zinc-100 font-semibold">Having trouble?</h1>
|
<h1 class="text-zinc-100 font-semibold">Having trouble?</h1>
|
||||||
@ -121,6 +126,7 @@
|
|||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { XCircleIcon } from "@heroicons/vue/16/solid";
|
import { XCircleIcon } from "@heroicons/vue/16/solid";
|
||||||
|
import { ArrowTopRightOnSquareIcon } from "@heroicons/vue/20/solid";
|
||||||
import { invoke } from "@tauri-apps/api/core";
|
import { invoke } from "@tauri-apps/api/core";
|
||||||
|
|
||||||
const loading = ref(false);
|
const loading = ref(false);
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
class="h-10 cursor-pointer flex flex-row items-center justify-between bg-zinc-950"
|
class="h-16 cursor-pointer flex flex-row items-center justify-between bg-zinc-950"
|
||||||
>
|
>
|
||||||
<div class="px-5 py-3 grow" @mousedown="() => window.startDragging()">
|
<div class="px-5 py-3 grow" @mousedown="() => window.startDragging()">
|
||||||
<Wordmark class="mt-1" />
|
<Wordmark class="mt-1" />
|
||||||
|
|||||||
37
pages/auth/code.vue
Normal file
37
pages/auth/code.vue
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
<template>
|
||||||
|
<div class="min-h-full w-full flex items-center justify-center">
|
||||||
|
<div class="flex flex-col items-center">
|
||||||
|
<div class="text-center">
|
||||||
|
<h1 class="text-3xl font-semibold font-display leading-6 text-zinc-100">
|
||||||
|
Device authorization
|
||||||
|
</h1>
|
||||||
|
<div class="mt-4">
|
||||||
|
<p class="text-sm text-zinc-400 max-w-md mx-auto">
|
||||||
|
Open Drop on another one of your devices, and use your account
|
||||||
|
dropdown to "Authorize client", and enter the code below.
|
||||||
|
</p>
|
||||||
|
<div
|
||||||
|
class="mt-8 flex items-center justify-center gap-x-5 text-8xl font-bold text-zinc-100"
|
||||||
|
>
|
||||||
|
<span v-for="letter in code.split('')">{{ letter }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="mt-10 flex items-center justify-center gap-x-6">
|
||||||
|
<NuxtLink href="/auth" class="text-sm font-semibold text-blue-600"
|
||||||
|
><span aria-hidden="true">←</span> Use a different method
|
||||||
|
</NuxtLink>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { invoke } from "@tauri-apps/api/core";
|
||||||
|
|
||||||
|
const code = await invoke<string>("auth_initiate_code");
|
||||||
|
|
||||||
|
definePageMeta({
|
||||||
|
layout: "mini",
|
||||||
|
});
|
||||||
|
</script>
|
||||||
54
src-tauri/Cargo.lock
generated
54
src-tauri/Cargo.lock
generated
@ -345,6 +345,22 @@ dependencies = [
|
|||||||
"syn 2.0.101",
|
"syn 2.0.101",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "async-tungstenite"
|
||||||
|
version = "0.29.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ef0f7efedeac57d9b26170f72965ecfd31473ca52ca7a64e925b0b6f5f079886"
|
||||||
|
dependencies = [
|
||||||
|
"atomic-waker",
|
||||||
|
"futures-core",
|
||||||
|
"futures-io",
|
||||||
|
"futures-task",
|
||||||
|
"futures-util",
|
||||||
|
"log",
|
||||||
|
"pin-project-lite",
|
||||||
|
"tungstenite",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "atk"
|
name = "atk"
|
||||||
version = "0.18.2"
|
version = "0.18.2"
|
||||||
@ -1280,6 +1296,7 @@ dependencies = [
|
|||||||
"droplet-rs",
|
"droplet-rs",
|
||||||
"dynfmt",
|
"dynfmt",
|
||||||
"filetime",
|
"filetime",
|
||||||
|
"futures-lite",
|
||||||
"gethostname",
|
"gethostname",
|
||||||
"hex 0.4.3",
|
"hex 0.4.3",
|
||||||
"http 1.3.1",
|
"http 1.3.1",
|
||||||
@ -1296,6 +1313,7 @@ dependencies = [
|
|||||||
"reqwest 0.12.16",
|
"reqwest 0.12.16",
|
||||||
"reqwest-middleware 0.4.2",
|
"reqwest-middleware 0.4.2",
|
||||||
"reqwest-middleware-cache",
|
"reqwest-middleware-cache",
|
||||||
|
"reqwest-websocket",
|
||||||
"rustbreak",
|
"rustbreak",
|
||||||
"rustix 0.38.44",
|
"rustix 0.38.44",
|
||||||
"schemars",
|
"schemars",
|
||||||
@ -4357,6 +4375,24 @@ dependencies = [
|
|||||||
"url",
|
"url",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "reqwest-websocket"
|
||||||
|
version = "0.5.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "9f91a811daaa8b54faeaec9d507a336897a3d243834a4965254a17d39da8b5c9"
|
||||||
|
dependencies = [
|
||||||
|
"async-tungstenite",
|
||||||
|
"bytes",
|
||||||
|
"futures-util",
|
||||||
|
"reqwest 0.12.16",
|
||||||
|
"thiserror 2.0.12",
|
||||||
|
"tokio",
|
||||||
|
"tokio-util",
|
||||||
|
"tracing",
|
||||||
|
"tungstenite",
|
||||||
|
"web-sys",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rfd"
|
name = "rfd"
|
||||||
version = "0.15.3"
|
version = "0.15.3"
|
||||||
@ -5831,6 +5867,7 @@ checksum = "66a539a9ad6d5d281510d5bd368c973d636c02dbf8a67300bfb6b950696ad7df"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"bytes",
|
"bytes",
|
||||||
"futures-core",
|
"futures-core",
|
||||||
|
"futures-io",
|
||||||
"futures-sink",
|
"futures-sink",
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
"tokio",
|
"tokio",
|
||||||
@ -6009,6 +6046,23 @@ version = "0.2.5"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b"
|
checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "tungstenite"
|
||||||
|
version = "0.26.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "4793cb5e56680ecbb1d843515b23b6de9a75eb04b66643e256a396d43be33c13"
|
||||||
|
dependencies = [
|
||||||
|
"bytes",
|
||||||
|
"data-encoding",
|
||||||
|
"http 1.3.1",
|
||||||
|
"httparse",
|
||||||
|
"log",
|
||||||
|
"rand 0.9.1",
|
||||||
|
"sha1",
|
||||||
|
"thiserror 2.0.12",
|
||||||
|
"utf-8",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "typeid"
|
name = "typeid"
|
||||||
version = "1.0.3"
|
version = "1.0.3"
|
||||||
|
|||||||
@ -68,6 +68,8 @@ known-folders = "1.2.0"
|
|||||||
native_model = { version = "0.6.1", features = ["rmp_serde_1_3"] }
|
native_model = { version = "0.6.1", features = ["rmp_serde_1_3"] }
|
||||||
tauri-plugin-opener = "2.4.0"
|
tauri-plugin-opener = "2.4.0"
|
||||||
bitcode = "0.6.6"
|
bitcode = "0.6.6"
|
||||||
|
reqwest-websocket = "0.5.0"
|
||||||
|
futures-lite = "2.6.0"
|
||||||
# tailscale = { path = "./tailscale" }
|
# tailscale = { path = "./tailscale" }
|
||||||
|
|
||||||
[dependencies.dynfmt]
|
[dependencies.dynfmt]
|
||||||
|
|||||||
@ -13,6 +13,7 @@ use super::drop_server_error::DropServerError;
|
|||||||
#[derive(Debug, SerializeDisplay)]
|
#[derive(Debug, SerializeDisplay)]
|
||||||
pub enum RemoteAccessError {
|
pub enum RemoteAccessError {
|
||||||
FetchError(Arc<reqwest::Error>),
|
FetchError(Arc<reqwest::Error>),
|
||||||
|
FetchErrorWS(Arc<reqwest_websocket::Error>),
|
||||||
ParsingError(ParseError),
|
ParsingError(ParseError),
|
||||||
InvalidEndpoint,
|
InvalidEndpoint,
|
||||||
HandshakeFailed(String),
|
HandshakeFailed(String),
|
||||||
@ -29,7 +30,10 @@ impl Display for RemoteAccessError {
|
|||||||
match self {
|
match self {
|
||||||
RemoteAccessError::FetchError(error) => {
|
RemoteAccessError::FetchError(error) => {
|
||||||
if error.is_connect() {
|
if error.is_connect() {
|
||||||
return write!(f, "Failed to connect to Drop server. Check if you access Drop through a browser, and then try again.");
|
return write!(
|
||||||
|
f,
|
||||||
|
"Failed to connect to Drop server. Check if you access Drop through a browser, and then try again."
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
write!(
|
write!(
|
||||||
@ -42,20 +46,40 @@ impl Display for RemoteAccessError {
|
|||||||
.or_else(|| Some("Unknown error".to_string()))
|
.or_else(|| Some("Unknown error".to_string()))
|
||||||
.unwrap()
|
.unwrap()
|
||||||
)
|
)
|
||||||
},
|
}
|
||||||
|
RemoteAccessError::FetchErrorWS(error) => write!(
|
||||||
|
f,
|
||||||
|
"{}: {}",
|
||||||
|
error,
|
||||||
|
error
|
||||||
|
.source()
|
||||||
|
.map(|e| e.to_string())
|
||||||
|
.or_else(|| Some("Unknown error".to_string()))
|
||||||
|
.unwrap()
|
||||||
|
),
|
||||||
RemoteAccessError::ParsingError(parse_error) => {
|
RemoteAccessError::ParsingError(parse_error) => {
|
||||||
write!(f, "{parse_error}")
|
write!(f, "{parse_error}")
|
||||||
}
|
}
|
||||||
RemoteAccessError::InvalidEndpoint => write!(f, "invalid drop endpoint"),
|
RemoteAccessError::InvalidEndpoint => write!(f, "invalid drop endpoint"),
|
||||||
RemoteAccessError::HandshakeFailed(message) => write!(f, "failed to complete handshake: {message}"),
|
RemoteAccessError::HandshakeFailed(message) => {
|
||||||
|
write!(f, "failed to complete handshake: {message}")
|
||||||
|
}
|
||||||
RemoteAccessError::GameNotFound(id) => write!(f, "could not find game on server: {id}"),
|
RemoteAccessError::GameNotFound(id) => write!(f, "could not find game on server: {id}"),
|
||||||
RemoteAccessError::InvalidResponse(error) => write!(f, "server returned an invalid response: {}, {}", error.status_code, error.status_message),
|
RemoteAccessError::InvalidResponse(error) => write!(
|
||||||
RemoteAccessError::UnparseableResponse(error) => write!(f, "server returned an invalid response: {error}"),
|
|
||||||
RemoteAccessError::ManifestDownloadFailed(status, response) => write!(
|
|
||||||
f,
|
f,
|
||||||
"failed to download game manifest: {status} {response}"
|
"server returned an invalid response: {}, {}",
|
||||||
|
error.status_code, error.status_message
|
||||||
|
),
|
||||||
|
RemoteAccessError::UnparseableResponse(error) => {
|
||||||
|
write!(f, "server returned an invalid response: {error}")
|
||||||
|
}
|
||||||
|
RemoteAccessError::ManifestDownloadFailed(status, response) => {
|
||||||
|
write!(f, "failed to download game manifest: {status} {response}")
|
||||||
|
}
|
||||||
|
RemoteAccessError::OutOfSync => write!(
|
||||||
|
f,
|
||||||
|
"server's and client's time are out of sync. Please ensure they are within at least 30 seconds of each other"
|
||||||
),
|
),
|
||||||
RemoteAccessError::OutOfSync => write!(f, "server's and client's time are out of sync. Please ensure they are within at least 30 seconds of each other"),
|
|
||||||
RemoteAccessError::Cache(error) => write!(f, "Cache Error: {error}"),
|
RemoteAccessError::Cache(error) => write!(f, "Cache Error: {error}"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -66,6 +90,11 @@ impl From<reqwest::Error> for RemoteAccessError {
|
|||||||
RemoteAccessError::FetchError(Arc::new(err))
|
RemoteAccessError::FetchError(Arc::new(err))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
impl From<reqwest_websocket::Error> for RemoteAccessError {
|
||||||
|
fn from(err: reqwest_websocket::Error) -> Self {
|
||||||
|
RemoteAccessError::FetchErrorWS(Arc::new(err))
|
||||||
|
}
|
||||||
|
}
|
||||||
impl From<ParseError> for RemoteAccessError {
|
impl From<ParseError> for RemoteAccessError {
|
||||||
fn from(err: ParseError) -> Self {
|
fn from(err: ParseError) -> Self {
|
||||||
RemoteAccessError::ParsingError(err)
|
RemoteAccessError::ParsingError(err)
|
||||||
|
|||||||
@ -12,6 +12,7 @@ mod process;
|
|||||||
mod remote;
|
mod remote;
|
||||||
|
|
||||||
use crate::process::commands::open_process_logs;
|
use crate::process::commands::open_process_logs;
|
||||||
|
use crate::remote::commands::auth_initiate_code;
|
||||||
use crate::{database::db::DatabaseImpls, games::downloads::commands::resume_download};
|
use crate::{database::db::DatabaseImpls, games::downloads::commands::resume_download};
|
||||||
use bitcode::{Decode, Encode};
|
use bitcode::{Decode, Encode};
|
||||||
use client::commands::fetch_state;
|
use client::commands::fetch_state;
|
||||||
@ -267,6 +268,7 @@ pub fn run() {
|
|||||||
fetch_settings,
|
fetch_settings,
|
||||||
// Auth
|
// Auth
|
||||||
auth_initiate,
|
auth_initiate,
|
||||||
|
auth_initiate_code,
|
||||||
retry_connect,
|
retry_connect,
|
||||||
manual_recieve_handshake,
|
manual_recieve_handshake,
|
||||||
sign_out,
|
sign_out,
|
||||||
|
|||||||
@ -9,12 +9,12 @@ use tauri::{AppHandle, Emitter, Manager};
|
|||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
AppState, AppStatus, User,
|
||||||
database::{
|
database::{
|
||||||
db::{borrow_db_checked, borrow_db_mut_checked},
|
db::{borrow_db_checked, borrow_db_mut_checked},
|
||||||
models::data::DatabaseAuth,
|
models::data::DatabaseAuth,
|
||||||
},
|
},
|
||||||
error::{drop_server_error::DropServerError, remote_access_error::RemoteAccessError},
|
error::{drop_server_error::DropServerError, remote_access_error::RemoteAccessError},
|
||||||
AppState, AppStatus, User,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
@ -32,6 +32,7 @@ struct InitiateRequestBody {
|
|||||||
name: String,
|
name: String,
|
||||||
platform: String,
|
platform: String,
|
||||||
capabilities: HashMap<String, CapabilityConfiguration>,
|
capabilities: HashMap<String, CapabilityConfiguration>,
|
||||||
|
mode: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
@ -166,7 +167,7 @@ pub fn recieve_handshake(app: AppHandle, path: String) {
|
|||||||
app.emit("auth/finished", ()).unwrap();
|
app.emit("auth/finished", ()).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn auth_initiate_logic() -> Result<(), RemoteAccessError> {
|
pub fn auth_initiate_logic(mode: String) -> Result<String, RemoteAccessError> {
|
||||||
let base_url = {
|
let base_url = {
|
||||||
let db_lock = borrow_db_checked();
|
let db_lock = borrow_db_checked();
|
||||||
Url::parse(&db_lock.base_url.clone())?
|
Url::parse(&db_lock.base_url.clone())?
|
||||||
@ -182,6 +183,7 @@ pub fn auth_initiate_logic() -> Result<(), RemoteAccessError> {
|
|||||||
("peerAPI".to_owned(), CapabilityConfiguration {}),
|
("peerAPI".to_owned(), CapabilityConfiguration {}),
|
||||||
("cloudSaves".to_owned(), CapabilityConfiguration {}),
|
("cloudSaves".to_owned(), CapabilityConfiguration {}),
|
||||||
]),
|
]),
|
||||||
|
mode,
|
||||||
};
|
};
|
||||||
|
|
||||||
let client = reqwest::blocking::Client::new();
|
let client = reqwest::blocking::Client::new();
|
||||||
@ -194,13 +196,9 @@ pub fn auth_initiate_logic() -> Result<(), RemoteAccessError> {
|
|||||||
return Err(RemoteAccessError::HandshakeFailed(data.status_message));
|
return Err(RemoteAccessError::HandshakeFailed(data.status_message));
|
||||||
}
|
}
|
||||||
|
|
||||||
let redir_url = response.text()?;
|
let response = response.text()?;
|
||||||
let complete_redir_url = base_url.join(&redir_url)?;
|
|
||||||
|
|
||||||
debug!("opening web browser to continue authentication");
|
Ok(response)
|
||||||
webbrowser::open(complete_redir_url.as_ref()).unwrap();
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn setup() -> (AppStatus, Option<User>) {
|
pub fn setup() -> (AppStatus, Option<User>) {
|
||||||
|
|||||||
@ -1,15 +1,18 @@
|
|||||||
use std::sync::Mutex;
|
use std::sync::Mutex;
|
||||||
|
|
||||||
use log::debug;
|
use futures_lite::StreamExt;
|
||||||
|
use log::{debug, warn};
|
||||||
use reqwest::blocking::Client;
|
use reqwest::blocking::Client;
|
||||||
|
use reqwest_websocket::{Message, RequestBuilderExt};
|
||||||
|
use serde::Deserialize;
|
||||||
use tauri::{AppHandle, Emitter, Manager};
|
use tauri::{AppHandle, Emitter, Manager};
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
AppState, AppStatus,
|
||||||
database::db::{borrow_db_checked, borrow_db_mut_checked},
|
database::db::{borrow_db_checked, borrow_db_mut_checked},
|
||||||
error::remote_access_error::RemoteAccessError,
|
error::remote_access_error::RemoteAccessError,
|
||||||
remote::{auth::generate_authorization_header, requests::make_request},
|
remote::{auth::generate_authorization_header, requests::make_request},
|
||||||
AppState, AppStatus,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
@ -91,7 +94,78 @@ pub fn retry_connect(state: tauri::State<'_, Mutex<AppState>>) {
|
|||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub fn auth_initiate() -> Result<(), RemoteAccessError> {
|
pub fn auth_initiate() -> Result<(), RemoteAccessError> {
|
||||||
auth_initiate_logic()
|
let base_url = {
|
||||||
|
let db_lock = borrow_db_checked();
|
||||||
|
Url::parse(&db_lock.base_url.clone())?
|
||||||
|
};
|
||||||
|
|
||||||
|
let redir_url = auth_initiate_logic("callback".to_string())?;
|
||||||
|
let complete_redir_url = base_url.join(&redir_url)?;
|
||||||
|
|
||||||
|
debug!("opening web browser to continue authentication");
|
||||||
|
webbrowser::open(complete_redir_url.as_ref()).unwrap();
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct CodeWebsocketResponse {
|
||||||
|
#[serde(rename = "type")]
|
||||||
|
response_type: String,
|
||||||
|
value: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tauri::command]
|
||||||
|
pub fn auth_initiate_code(app: AppHandle) -> Result<String, RemoteAccessError> {
|
||||||
|
let base_url = {
|
||||||
|
let db_lock = borrow_db_checked();
|
||||||
|
Url::parse(&db_lock.base_url.clone())?
|
||||||
|
};
|
||||||
|
|
||||||
|
let code = auth_initiate_logic("code".to_string())?;
|
||||||
|
let header_code = code.clone();
|
||||||
|
|
||||||
|
tauri::async_runtime::spawn(async move {
|
||||||
|
let load = async || -> Result<(), RemoteAccessError> {
|
||||||
|
let ws_url = base_url.join("/api/v1/client/auth/code/ws")?;
|
||||||
|
let response = reqwest::Client::default()
|
||||||
|
.get(ws_url)
|
||||||
|
.header("Authorization", header_code)
|
||||||
|
.upgrade()
|
||||||
|
.send()
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let mut websocket = response.into_websocket().await?;
|
||||||
|
|
||||||
|
while let Some(token) = websocket.try_next().await? {
|
||||||
|
if let Message::Text(response) = token {
|
||||||
|
let response = serde_json::from_str::<CodeWebsocketResponse>(&response)
|
||||||
|
.map_err(|e| RemoteAccessError::UnparseableResponse(e.to_string()))?;
|
||||||
|
match response.response_type.as_str() {
|
||||||
|
"token" => {
|
||||||
|
let recieve_app = app.clone();
|
||||||
|
tauri::async_runtime::spawn_blocking(move || {
|
||||||
|
manual_recieve_handshake(recieve_app, response.value);
|
||||||
|
});
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
_ => return Err(RemoteAccessError::HandshakeFailed(response.value)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Err(RemoteAccessError::HandshakeFailed(
|
||||||
|
"Failed to connect to websocket".to_string(),
|
||||||
|
))
|
||||||
|
};
|
||||||
|
|
||||||
|
let result = load().await;
|
||||||
|
if let Err(err) = result {
|
||||||
|
warn!("{err}");
|
||||||
|
app.emit("auth/failed", err.to_string()).unwrap();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(code)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
|
|||||||
Reference in New Issue
Block a user