all is change; in change is all again made new
This commit is contained in:
parent
cee43342b9
commit
c19b573b26
@ -23,7 +23,7 @@ pub async fn serve(addr: SocketAddrV4, app_handle: AppHandle) -> io::Result<()>
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
println!("Error accepting connection: {e}");
|
eprintln!("Error accepting connection: {e}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -73,27 +73,33 @@ use tokio::io::{stdin, stdout, BufReader, AsyncBufReadExt};
|
|||||||
use crate::storage;
|
use crate::storage;
|
||||||
|
|
||||||
use tokio::sync::oneshot;
|
use tokio::sync::oneshot;
|
||||||
|
use std::sync::Mutex;
|
||||||
|
|
||||||
async fn get_creds(app_handle: &AppHandle) -> io::Result<String> {
|
async fn get_creds(app_handle: &AppHandle) -> io::Result<String> {
|
||||||
app_handle.emit_all("credentials-request", ()).unwrap();
|
{
|
||||||
|
let state_guard = app_handle.state::<Mutex<crate::AppState>>();
|
||||||
// let mut out = stdout();
|
let mut state = state_guard.lock().unwrap();
|
||||||
// out.write_all(b"Enter passphrase: ").await?;
|
state.num_requests += 1;
|
||||||
// out.flush().await?;
|
let req = crate::CredentialsRequest {
|
||||||
|
request_id: state.num_requests
|
||||||
|
};
|
||||||
|
app_handle.emit_all("credentials-request", req).unwrap();
|
||||||
|
// lock gets released here in case somebody else needs app state while we're waiting
|
||||||
|
}
|
||||||
|
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
app_handle.once_global("passphrase-entered", |event| {
|
app_handle.once_global("request-response", |event| {
|
||||||
match event.payload() {
|
let response = event.payload().unwrap_or("").to_string();
|
||||||
Some(p) => {tx.send(p.to_string());}
|
tx.send(response).unwrap();
|
||||||
None => {tx.send("".to_string());} // will fail decryption, we just need to unblock the outer function
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
// Error is only returned if the rx is closed/dropped before receiving, which should never happen
|
// Error is only returned if the rx is closed/dropped before receiving, which should never happen
|
||||||
let passphrase = rx.await.unwrap();
|
// LOL who am I kidding this happens all the time
|
||||||
|
// fix it later
|
||||||
|
|
||||||
|
// todo: handle "denied" response
|
||||||
|
let _response = rx.await.unwrap();
|
||||||
|
let state_guard = app_handle.state::<Mutex<crate::AppState>>();
|
||||||
|
let state = state_guard.lock().unwrap();
|
||||||
|
|
||||||
// let mut passphrase = String::new();
|
Ok(state.current_session.clone().unwrap())
|
||||||
// let mut reader = BufReader::new(stdin());
|
|
||||||
// reader.read_line(&mut passphrase).await?;
|
|
||||||
|
|
||||||
Ok(storage::load(&passphrase.trim()))
|
|
||||||
}
|
}
|
||||||
|
10
src-tauri/src/ipc.rs
Normal file
10
src-tauri/src/ipc.rs
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
pub enum RequestResponse {
|
||||||
|
Approved,
|
||||||
|
Denied,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
pub struct Request {
|
||||||
|
pub id: u64,
|
||||||
|
pub response: RequestResponse,
|
||||||
|
}
|
@ -3,15 +3,25 @@
|
|||||||
windows_subsystem = "windows"
|
windows_subsystem = "windows"
|
||||||
)]
|
)]
|
||||||
|
|
||||||
|
use std::collections::HashMap;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
use std::sync::Mutex;
|
||||||
// use tokio::runtime::Runtime;
|
// use tokio::runtime::Runtime;
|
||||||
|
|
||||||
|
use serde::{Serialize, Deserialize};
|
||||||
|
use tauri::{Manager, State};
|
||||||
|
use tokio::sync::oneshot;
|
||||||
|
|
||||||
mod storage;
|
mod storage;
|
||||||
mod http;
|
mod http;
|
||||||
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
tauri::Builder::default()
|
tauri::Builder::default()
|
||||||
|
.manage(CurrentSession)
|
||||||
|
.manage(RequestCount)
|
||||||
|
.manage(OpenRequests)
|
||||||
|
.invoke_handler(tauri::generate_handler![unlock])
|
||||||
.setup(|app| {
|
.setup(|app| {
|
||||||
let addr = std::net::SocketAddrV4::from_str("127.0.0.1:12345").unwrap();
|
let addr = std::net::SocketAddrV4::from_str("127.0.0.1:12345").unwrap();
|
||||||
tauri::async_runtime::spawn(http::serve(addr, app.handle()));
|
tauri::async_runtime::spawn(http::serve(addr, app.handle()));
|
||||||
@ -28,3 +38,35 @@ fn main() {
|
|||||||
// let creds = std::fs::read_to_string("credentials.json").unwrap();
|
// let creds = std::fs::read_to_string("credentials.json").unwrap();
|
||||||
// storage::save(&creds, "correct horse battery staple");
|
// storage::save(&creds, "correct horse battery staple");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
pub enum Session {
|
||||||
|
Unlocked(String),
|
||||||
|
Locked,
|
||||||
|
Empty,
|
||||||
|
}
|
||||||
|
|
||||||
|
type CurrentSession = Mutex<Session>;
|
||||||
|
type RequestCount = Mutex<u64>;
|
||||||
|
type OpenRequests = Mutex<HashMap<u64, oneshot::Sender>>;
|
||||||
|
|
||||||
|
#[derive(Clone, Serialize, Deserialize)]
|
||||||
|
pub struct CredentialsRequest {
|
||||||
|
pub request_id: u64,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// struct Session {
|
||||||
|
// key_id: String,
|
||||||
|
// secret_key: String,
|
||||||
|
// token: String,
|
||||||
|
// expires: u64,
|
||||||
|
// }
|
||||||
|
|
||||||
|
|
||||||
|
#[tauri::command]
|
||||||
|
fn unlock(passphrase: String, current_session: State<'_, CurrentSession>) -> bool {
|
||||||
|
let credentials = storage::load(&passphrase);
|
||||||
|
*current_session.lock().unwrap() = CredentialStatus::Unlocked(credentials);
|
||||||
|
true
|
||||||
|
}
|
||||||
|
74
src-tauri/src/state.rs
Normal file
74
src-tauri/src/state.rs
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
use std::sync::RwLock;
|
||||||
|
|
||||||
|
use serde::{Serialize, Deserialize};
|
||||||
|
use tokio::sync::oneshot::Sender;
|
||||||
|
|
||||||
|
use crate::ipc;
|
||||||
|
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
pub enum Credentials {
|
||||||
|
LongLived {
|
||||||
|
key_id: String,
|
||||||
|
secret_key: String,
|
||||||
|
},
|
||||||
|
ShortLived {
|
||||||
|
key_id: String,
|
||||||
|
secret_key: String,
|
||||||
|
session_token: String,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
pub enum CurrentSession {
|
||||||
|
Unlocked(String),
|
||||||
|
Locked,
|
||||||
|
Empty,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
pub struct AppState {
|
||||||
|
current_session: RwLock<CurrentSession>,
|
||||||
|
request_count: RwLock<u64>,
|
||||||
|
open_requests: RwLock<HashMap<u64, Sender>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AppState {
|
||||||
|
pub fn new(current_session: CurrentSession) -> Self {
|
||||||
|
AppState {
|
||||||
|
current_session,
|
||||||
|
request_count: 0,
|
||||||
|
open_requests: HashMap::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn register_request(&mut self, chan: Sender) -> u64 {
|
||||||
|
let count = {
|
||||||
|
let c = self.request_count.write().unwrap();
|
||||||
|
*c += 1;
|
||||||
|
c
|
||||||
|
};
|
||||||
|
|
||||||
|
let open_requests = self.open_requests.write().unwrap();
|
||||||
|
self.open_requests.insert(count, chan);
|
||||||
|
count
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn send_response(&mut self, req_id: u64, response: ipc::RequestResponse) -> Result<(), SendResponseError> {
|
||||||
|
let mut open_requests = self.open_requests.write().unwrap();
|
||||||
|
let chan = self.open_requests
|
||||||
|
.remove(&req_id)
|
||||||
|
.ok_or(SendResponseError::NotFound)
|
||||||
|
?;
|
||||||
|
|
||||||
|
chan.send(response)
|
||||||
|
.map_err(|_e| SendResponseError::Abandoned)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
pub enum SendResponseError {
|
||||||
|
NotFound, // no request with the given id
|
||||||
|
Abandoned, // request has already been closed by client
|
||||||
|
}
|
@ -1,33 +1,36 @@
|
|||||||
<script>
|
<script>
|
||||||
import { emit, listen } from '@tauri-apps/api/event';
|
import { emit, listen } from '@tauri-apps/api/event';
|
||||||
import queue from './lib/queue.js';
|
import queue from './lib/queue.js';
|
||||||
import Home from './views/Home.svelte';
|
|
||||||
import Approve from './views/Approve.svelte';
|
|
||||||
import ShowApproved from './views/ShowApproved.svelte';
|
|
||||||
import ShowDenied from './views/ShowDenied.svelte';
|
|
||||||
|
|
||||||
const VIEWS = {
|
const VIEWS = import.meta.glob('./views/*.svelte', {eager: true});
|
||||||
Home: Home,
|
|
||||||
Approve: Approve,
|
window.emit = emit;
|
||||||
ShowApproved: ShowApproved,
|
window.queue = queue;
|
||||||
ShowDenied: ShowDenied,
|
|
||||||
};
|
var appState = {
|
||||||
|
currentRequest: null,
|
||||||
|
pendingRequests: queue(),
|
||||||
|
credentialStatus: 'locked',
|
||||||
|
}
|
||||||
|
window.appState = appState;
|
||||||
|
|
||||||
|
|
||||||
let currentView = Home;
|
import { invoke } from '@tauri-apps/api/tauri';
|
||||||
function navigate(event) {
|
window.invoke = invoke;
|
||||||
currentView = VIEWS[event.detail.target];
|
|
||||||
|
|
||||||
|
var currentView = VIEWS['./views/Home.svelte'].default;
|
||||||
|
window.currentView = currentView;
|
||||||
|
window.VIEWS = VIEWS;
|
||||||
|
function navigate(svelteEvent) {
|
||||||
|
const moduleName = `./views/${svelteEvent.detail.target}.svelte`;
|
||||||
|
currentView = VIEWS[moduleName].default;
|
||||||
}
|
}
|
||||||
|
|
||||||
listen('credentials-request', (event) => {
|
listen('credentials-request', (tauriEvent) => {
|
||||||
queue.put(1)
|
appState.pendingRequests.put(tauriEvent.payload);
|
||||||
|
console.log('Received request.');
|
||||||
});
|
});
|
||||||
|
|
||||||
let requests = queue();
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if currentView === Home}
|
<svelte:component this={currentView} on:navigate={navigate} bind:appState={appState} />
|
||||||
<svelte:component this={currentView} on:navigate={navigate} {requests} />
|
|
||||||
{:else}
|
|
||||||
<svelte:component this={currentView} on:navigate={navigate} />
|
|
||||||
{/if}
|
|
||||||
|
@ -1,10 +1,18 @@
|
|||||||
<script>
|
<script>
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
|
import { invoke } from '@tauri-apps/api/tauri';
|
||||||
|
|
||||||
|
export let appState;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
function approve() {
|
async function approve() {
|
||||||
dispatch('navigate', {target: 'ShowApproved'});
|
if (appState.credentialStatus === 'unlocked') {
|
||||||
|
dispatch('navigate', {target: 'ShowApproved'});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
dispatch('navigate', {target: 'Unlock'});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function deny() {
|
function deny() {
|
||||||
|
@ -1,12 +1,19 @@
|
|||||||
<script>
|
<script>
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { onMount, createEventDispatcher } from 'svelte';
|
||||||
|
|
||||||
|
export let appState;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
let requests;
|
onMount(async () => {
|
||||||
let r = await requests.get();
|
// will block until a request comes in
|
||||||
|
let req = await appState.pendingRequests.get();
|
||||||
dispatch('navigate', {target: 'Approve.svelte'});
|
console.log(req);
|
||||||
|
appState.currentRequest = req;
|
||||||
|
console.log('Got credentials request from queue.');
|
||||||
|
console.log(appState);
|
||||||
|
dispatch('navigate', {target: 'Approve'});
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<h1 class="text-4xl text-gray-300">Creddy</h1>
|
<h1 class="text-4xl text-gray-300">Creddy</h1>
|
@ -1,5 +1,14 @@
|
|||||||
<script>
|
<script>
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
|
import { emit } from '@tauri-apps/api/event';
|
||||||
|
|
||||||
|
export let appState;
|
||||||
|
|
||||||
|
var p = emit('request-response', {response: 'approved', requestId: 1});
|
||||||
|
console.log('event emitted');
|
||||||
|
console.log(p);
|
||||||
|
appState.currentRequest = null;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
window.setTimeout(() => dispatch('navigate', {target: 'Home'}), 3000);
|
window.setTimeout(() => dispatch('navigate', {target: 'Home'}), 3000);
|
||||||
</script>
|
</script>
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
<script>
|
<script>
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
|
import { emit } from '@tauri-apps/api/event';
|
||||||
|
|
||||||
|
export let appState;
|
||||||
|
|
||||||
|
emit('request-response', {response: 'denied', requestId: 1});
|
||||||
|
appState.currentRequest = null;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
window.setTimeout(() => dispatch('navigate', {target: 'Home'}), 3000);
|
window.setTimeout(() => dispatch('navigate', {target: 'Home'}), 3000);
|
||||||
</script>
|
</script>
|
||||||
|
29
src/views/Unlock.svelte
Normal file
29
src/views/Unlock.svelte
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
<script>
|
||||||
|
import { invoke } from '@tauri-apps/api/tauri';
|
||||||
|
import { createEventDispatcher } from 'svelte';
|
||||||
|
|
||||||
|
export let appState;
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
let passphrase = '';
|
||||||
|
async function unlock() {
|
||||||
|
console.log('invoking unlock command.')
|
||||||
|
let res = await invoke('unlock', {passphrase});
|
||||||
|
if (res) {
|
||||||
|
appState.credentialStatus = 'unlocked';
|
||||||
|
console.log('Unlock successful!');
|
||||||
|
if (appState.currentRequest) {
|
||||||
|
dispatch('navigate', {target: 'ShowApproved'});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// indicate decryption failed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<form action="#" on:submit|preventDefault="{unlock}">
|
||||||
|
<div class="text-gray-200">Enter your passphrase:</div>
|
||||||
|
<input class="text-gray-200 bg-zinc-800" type="password" placeholder="correct horse battery staple" bind:value="{passphrase}" />
|
||||||
|
</form>
|
Loading…
x
Reference in New Issue
Block a user