actually add the script
This commit is contained in:
parent
c0ab400af6
commit
b808a72b35
77
web3_proxy/src/bin/web3_proxy_cli/create_key.rs
Normal file
77
web3_proxy/src/bin/web3_proxy_cli/create_key.rs
Normal file
@ -0,0 +1,77 @@
|
||||
use anyhow::Context;
|
||||
use argh::FromArgs;
|
||||
use entities::{rpc_key, user};
|
||||
use ethers::prelude::Address;
|
||||
use log::info;
|
||||
use migration::sea_orm::{self, ActiveModelTrait, ColumnTrait, EntityTrait, QueryFilter};
|
||||
use ulid::Ulid;
|
||||
use uuid::Uuid;
|
||||
use web3_proxy::frontend::authorization::RpcSecretKey;
|
||||
|
||||
#[derive(FromArgs, PartialEq, Debug, Eq)]
|
||||
/// Create a new user and api key
|
||||
#[argh(subcommand, name = "create_key")]
|
||||
pub struct CreateKeySubCommand {
|
||||
/// the user's ethereum address or descriptive string.
|
||||
/// If a string is given, it will be converted to hex and potentially truncated.
|
||||
/// Users from strings are only for testing since they won't be able to log in.
|
||||
#[argh(positional)]
|
||||
address: String,
|
||||
|
||||
/// the user's api ULID or UUID key.
|
||||
/// If none given, one will be created.
|
||||
#[argh(option)]
|
||||
rpc_secret_key: Option<RpcSecretKey>,
|
||||
|
||||
/// an optional short description of the key's purpose.
|
||||
#[argh(option)]
|
||||
description: Option<String>,
|
||||
}
|
||||
|
||||
impl CreateKeySubCommand {
|
||||
pub async fn main(self, db: &sea_orm::DatabaseConnection) -> anyhow::Result<()> {
|
||||
// TODO: would be nice to use the fixed array instead of a Vec in the entities
|
||||
// take a simple String. If it starts with 0x, parse as address. otherwise convert ascii to hex
|
||||
let address: Vec<u8> = if self.address.starts_with("0x") {
|
||||
let address = self.address.parse::<Address>()?;
|
||||
|
||||
address.to_fixed_bytes().into()
|
||||
} else {
|
||||
// TODO: allow ENS
|
||||
// left pad and truncate the string
|
||||
let address = &format!("{:\x00>20}", self.address)[0..20];
|
||||
|
||||
// convert the string to bytes
|
||||
let bytes = address.as_bytes();
|
||||
|
||||
// convert the slice to a Vec
|
||||
bytes.try_into().expect("Bytes can always be a Vec<u8>")
|
||||
};
|
||||
|
||||
// TODO: get existing or create a new one
|
||||
let u = user::Entity::find()
|
||||
.filter(user::Column::Address.eq(address))
|
||||
.one(db)
|
||||
.await?
|
||||
.context("No user found with that address")?;
|
||||
|
||||
info!("user #{}", u.id);
|
||||
|
||||
let rpc_secret_key = self.rpc_secret_key.unwrap_or_else(RpcSecretKey::new);
|
||||
|
||||
// create a key for the new user
|
||||
let uk = rpc_key::ActiveModel {
|
||||
user_id: sea_orm::Set(u.id),
|
||||
secret_key: sea_orm::Set(rpc_secret_key.into()),
|
||||
description: sea_orm::Set(self.description),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
let _uk = uk.save(db).await.context("Failed saving new user key")?;
|
||||
|
||||
info!("user key as ULID: {}", Ulid::from(rpc_secret_key));
|
||||
info!("user key as UUID: {}", Uuid::from(rpc_secret_key));
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user