|
@@ -1,5 +1,6 @@
|
|
|
use crate::{schema, user::UserExt, UIDCError};
|
|
|
use microrm::{cli::CLIError, prelude::*, schema::entity::EntityID};
|
|
|
+use ring::rand::SecureRandom;
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
pub struct UserInterface;
|
|
@@ -20,12 +21,19 @@ pub enum UserCommands {
|
|
|
username: String,
|
|
|
provider: schema::ExternalAuthProvider,
|
|
|
},
|
|
|
+ OnetimeLink {
|
|
|
+ username: String,
|
|
|
+ #[clap(long)]
|
|
|
+ endpoint: Option<String>,
|
|
|
+ #[clap(long)]
|
|
|
+ expiry: Option<String>,
|
|
|
+ },
|
|
|
}
|
|
|
|
|
|
impl microrm::cli::EntityInterface for UserInterface {
|
|
|
type Error = UIDCError;
|
|
|
type Entity = schema::User;
|
|
|
- type Context = microrm::schema::Stored<schema::Realm>;
|
|
|
+ type Context = super::RunArgs; // microrm::schema::Stored<schema::Realm>;
|
|
|
type CustomCommand = UserCommands;
|
|
|
|
|
|
fn run_custom(
|
|
@@ -36,7 +44,7 @@ impl microrm::cli::EntityInterface for UserInterface {
|
|
|
match cmd {
|
|
|
UserCommands::Create { username } => {
|
|
|
query_ctx.insert(schema::User {
|
|
|
- realm: ctx.id(),
|
|
|
+ realm: ctx.realm.id(),
|
|
|
username,
|
|
|
pending_external_auths: Default::default(),
|
|
|
auth: Default::default(),
|
|
@@ -96,6 +104,53 @@ impl microrm::cli::EntityInterface for UserInterface {
|
|
|
user.pending_external_auths.as_mut().dedup();
|
|
|
user.sync().expect("couldn't sync user model");
|
|
|
}
|
|
|
+ UserCommands::OnetimeLink {
|
|
|
+ username,
|
|
|
+ endpoint,
|
|
|
+ expiry,
|
|
|
+ } => {
|
|
|
+ let user = query_ctx
|
|
|
+ .with(schema::User::Username, &username)
|
|
|
+ .first()
|
|
|
+ .get_ids()?
|
|
|
+ .ok_or(Self::Error::no_such_entity("user", username))?;
|
|
|
+
|
|
|
+ let rng = ring::rand::SystemRandom::new();
|
|
|
+ let mut code = [0u8; 16];
|
|
|
+ rng.fill(&mut code)
|
|
|
+ .map_err(|_| UIDCError::Abort("couldn't generate random values"))?;
|
|
|
+
|
|
|
+ let code = base64::encode_config(code, base64::URL_SAFE_NO_PAD);
|
|
|
+
|
|
|
+ let expiry = time::OffsetDateTime::now_utc()
|
|
|
+ .checked_add(time::Duration::days(7))
|
|
|
+ .unwrap();
|
|
|
+
|
|
|
+ ctx.realm
|
|
|
+ .single_use_auth
|
|
|
+ .insert(schema::SingleUseAuth {
|
|
|
+ code: code.clone(),
|
|
|
+ user,
|
|
|
+ expiry,
|
|
|
+ })
|
|
|
+ .expect("couldn't insert single-use authentication code?");
|
|
|
+
|
|
|
+ let target = tide::http::Url::parse_with_params(
|
|
|
+ format!(
|
|
|
+ "{base}/{realm}/v1/session/onetime",
|
|
|
+ base = ctx.config.base_url,
|
|
|
+ realm = ctx.realm.shortname
|
|
|
+ )
|
|
|
+ .as_str(),
|
|
|
+ &[
|
|
|
+ ("code", code.as_str()),
|
|
|
+ ("redirect", endpoint.as_deref().unwrap_or("")),
|
|
|
+ ],
|
|
|
+ )
|
|
|
+ .expect("couldn't generate URL");
|
|
|
+
|
|
|
+ println!("target URL: {target}");
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
Ok(())
|
|
@@ -116,7 +171,7 @@ impl microrm::cli::EntityInterface for UserInterface {
|
|
|
_role: microrm::cli::ValueRole,
|
|
|
) -> String {
|
|
|
if field == "realm" {
|
|
|
- format!("{}", ctx.id().into_raw())
|
|
|
+ format!("{}", ctx.realm.id().into_raw())
|
|
|
} else {
|
|
|
unreachable!()
|
|
|
}
|