extern crate base64; extern crate openidconnect; extern crate url; use std::sync::{Arc, Mutex}; use std::thread::{sleep, spawn, JoinHandle}; use std::time::Duration; use openidconnect::core::{CoreClient, CoreProviderMetadata}; use openidconnect::reqwest::http_client; use openidconnect::{ClientId, IssuerUrl, RedirectUrl}; use url::Url; use std::ffi::CStr; use std::os::raw::c_char; pub struct ZeroIDC { inner: Arc>, } struct Inner { running: bool, oidc_thread: Option>, oidc_client: Option, } impl ZeroIDC { fn new(issuer: &str, client_id: &str, port: u16) -> Result { let idc = ZeroIDC { inner: Arc::new(Mutex::new(Inner { running: false, oidc_thread: None, oidc_client: None, })), }; let iss = match IssuerUrl::new(issuer.to_string()) { Ok(i) => i, Err(e) => return Err(e.to_string()), }; let provider_meta = match CoreProviderMetadata::discover(&iss, http_client) { Ok(m) => m, Err(e) => return Err(e.to_string()), }; let r = format!("http://localhost:{}", port); let redir_url = match Url::parse(&r) { Ok(s) => s, Err(e) => return Err(e.to_string()), }; let redirect = match RedirectUrl::new(redir_url.to_string()) { Ok(s) => s, Err(e) => return Err(e.to_string()), }; (*idc.inner.lock().unwrap()).oidc_client = Some( CoreClient::from_provider_metadata( provider_meta, ClientId::new(client_id.to_string()), None, ) .set_redirect_uri(redirect), ); Ok(idc) } fn start(&mut self) { let local = Arc::clone(&self.inner); if !(*local.lock().unwrap()).running { let inner_local = Arc::clone(&self.inner); (*local.lock().unwrap()).oidc_thread = Some(spawn(move || { (*inner_local.lock().unwrap()).running = true; while (*inner_local.lock().unwrap()).running { println!("tick"); sleep(Duration::from_secs(1)); } println!("thread done!") })); } } fn stop(&mut self) { let local = self.inner.clone(); if (*local.lock().unwrap()).running { if let Some(u) = (*local.lock().unwrap()).oidc_thread.take() { u.join().expect("join failed"); } } } } #[no_mangle] pub extern "C" fn zeroidc_new( issuer: *const c_char, client_id: *const c_char, web_port: u16, ) -> *mut ZeroIDC { if issuer.is_null() { println!("issuer is null"); return std::ptr::null_mut(); } if client_id.is_null() { println!("client_id is null"); return std::ptr::null_mut(); } let iss = unsafe { CStr::from_ptr(issuer) }; let c_id = unsafe { CStr::from_ptr(client_id) }; match ZeroIDC::new(iss.to_str().unwrap(), c_id.to_str().unwrap(), web_port) { Ok(idc) => { return Box::into_raw(Box::new(idc)); } Err(s) => { println!("Error creating ZeroIDC instance: {}", s); return std::ptr::null_mut(); } } } #[no_mangle] pub extern "C" fn zeroidc_delete(ptr: *mut ZeroIDC) { if ptr.is_null() { return; } unsafe { Box::from_raw(ptr); } } #[no_mangle] pub extern "C" fn zeroidc_start(ptr: *mut ZeroIDC) { let idc = unsafe { assert!(!ptr.is_null()); &mut *ptr }; idc.start(); } #[no_mangle] pub extern "C" fn zeroidc_stop(ptr: *mut ZeroIDC) { let idc = unsafe { assert!(!ptr.is_null()); &mut *ptr }; idc.stop(); }