Skip to content

Commit

Permalink
fix: revert addition of apikey to auth (shuttle-hq#886)
Browse files Browse the repository at this point in the history
* fix: revert addition of apikey to auth

* fix: display impl is needed for key.to_string()
  • Loading branch information
oddgrd authored and morlinbrot committed May 16, 2023
1 parent ab12fdd commit f5f407b
Show file tree
Hide file tree
Showing 10 changed files with 133 additions and 37 deletions.
2 changes: 1 addition & 1 deletion auth/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ tracing-subscriber = { workspace = true }

[dependencies.shuttle-common]
workspace = true
features = ["backend", "models", "persist"]
features = ["backend", "models"]

[dev-dependencies]
axum-extra = { version = "0.7.1", features = ["cookie"] }
Expand Down
2 changes: 1 addition & 1 deletion auth/src/api/handlers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,7 @@ pub(crate) async fn convert_key(
let User {
name, account_tier, ..
} = user_manager
.get_user_by_key(key.as_ref().clone())
.get_user_by_key(key)
.await
.map_err(|_| StatusCode::UNAUTHORIZED)?;

Expand Down
17 changes: 8 additions & 9 deletions auth/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@ mod user;
use std::{io, str::FromStr, time::Duration};

use args::StartArgs;
use shuttle_common::ApiKey;
use sqlx::{
migrate::Migrator,
query,
Expand All @@ -16,7 +15,10 @@ use sqlx::{
};
use tracing::info;

use crate::{api::serve, user::AccountTier};
use crate::{
api::serve,
user::{AccountTier, Key},
};
pub use api::ApiBuilder;
pub use args::{Args, Commands, InitArgs};

Expand All @@ -39,8 +41,8 @@ pub async fn start(pool: SqlitePool, args: StartArgs) -> io::Result<()> {

pub async fn init(pool: SqlitePool, args: InitArgs) -> io::Result<()> {
let key = match args.key {
Some(ref key) => ApiKey::parse(key).unwrap(),
None => ApiKey::generate(),
Some(ref key) => Key::from_str(key).unwrap(),
None => Key::new_random(),
};

query("INSERT INTO users (account_name, key, account_tier) VALUES (?1, ?2, ?3)")
Expand All @@ -51,11 +53,8 @@ pub async fn init(pool: SqlitePool, args: InitArgs) -> io::Result<()> {
.await
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;

println!(
"`{}` created as super user with key: {}",
args.name,
key.as_ref()
);
println!("`{}` created as super user with key: {key}", args.name,);

Ok(())
}

Expand Down
62 changes: 36 additions & 26 deletions auth/src/user.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,9 @@ use axum::{
http::request::Parts,
TypedHeader,
};
use rand::distributions::{Alphanumeric, DistString};
use serde::{Deserialize, Deserializer, Serialize};
use shuttle_common::{
claims::{Scope, ScopeBuilder},
ApiKey,
};
use shuttle_common::claims::{Scope, ScopeBuilder};
use sqlx::{query, Row, SqlitePool};
use tracing::{trace, Span};

Expand All @@ -21,7 +19,7 @@ use crate::{api::UserManagerState, error::Error};
pub trait UserManagement: Send + Sync {
async fn create_user(&self, name: AccountName, tier: AccountTier) -> Result<User, Error>;
async fn get_user(&self, name: AccountName) -> Result<User, Error>;
async fn get_user_by_key(&self, key: ApiKey) -> Result<User, Error>;
async fn get_user_by_key(&self, key: Key) -> Result<User, Error>;
}

#[derive(Clone)]
Expand All @@ -32,7 +30,7 @@ pub struct UserManager {
#[async_trait]
impl UserManagement for UserManager {
async fn create_user(&self, name: AccountName, tier: AccountTier) -> Result<User, Error> {
let key = ApiKey::generate();
let key = Key::new_random();

query("INSERT INTO users (account_name, key, account_tier) VALUES (?1, ?2, ?3)")
.bind(&name)
Expand All @@ -57,7 +55,7 @@ impl UserManagement for UserManager {
.ok_or(Error::UserNotFound)
}

async fn get_user_by_key(&self, key: ApiKey) -> Result<User, Error> {
async fn get_user_by_key(&self, key: Key) -> Result<User, Error> {
query("SELECT account_name, key, account_tier FROM users WHERE key = ?1")
.bind(&key)
.fetch_optional(&self.pool)
Expand All @@ -71,10 +69,10 @@ impl UserManagement for UserManager {
}
}

#[derive(Clone, Deserialize, PartialEq, Eq, Serialize, Debug)]
#[derive(Clone, Debug, Deserialize, PartialEq, Eq, Serialize)]
pub struct User {
pub name: AccountName,
pub key: ApiKey,
pub key: Key,
pub account_tier: AccountTier,
}

Expand All @@ -83,7 +81,7 @@ impl User {
self.account_tier == AccountTier::Admin
}

pub fn new(name: AccountName, key: ApiKey, account_tier: AccountTier) -> Self {
pub fn new(name: AccountName, key: Key, account_tier: AccountTier) -> Self {
Self {
name,
key,
Expand All @@ -106,7 +104,7 @@ where
let user_manager: UserManagerState = UserManagerState::from_ref(state);

let user = user_manager
.get_user_by_key(key.as_ref().clone())
.get_user_by_key(key)
.await
// Absorb any error into `Unauthorized`
.map_err(|_| Error::Unauthorized)?;
Expand All @@ -122,21 +120,16 @@ impl From<User> for shuttle_common::models::user::Response {
fn from(user: User) -> Self {
Self {
name: user.name.to_string(),
key: user.key.as_ref().to_string(),
key: user.key.to_string(),
account_tier: user.account_tier.to_string(),
}
}
}

/// A wrapper around [ApiKey] so we can implement [FromRequestParts]
/// for it.
pub struct Key(ApiKey);

impl AsRef<ApiKey> for Key {
fn as_ref(&self) -> &ApiKey {
&self.0
}
}
#[derive(Clone, sqlx::Type, PartialEq, Hash, Eq, Serialize, Deserialize, Debug)]
#[serde(transparent)]
#[sqlx(transparent)]
pub struct Key(String);

#[async_trait]
impl<S> FromRequestParts<S> for Key
Expand All @@ -149,14 +142,31 @@ where
let key = TypedHeader::<Authorization<Bearer>>::from_request_parts(parts, state)
.await
.map_err(|_| Error::KeyMissing)
.and_then(|TypedHeader(Authorization(bearer))| {
let bearer = bearer.token().trim();
ApiKey::parse(bearer).map_err(|_| Self::Rejection::Unauthorized)
})?;
.and_then(|TypedHeader(Authorization(bearer))| bearer.token().trim().parse())?;

trace!("got bearer key");

Ok(Key(key))
Ok(key)
}
}

impl std::fmt::Display for Key {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.0)
}
}

impl FromStr for Key {
type Err = Error;

fn from_str(s: &str) -> Result<Self, Self::Err> {
Ok(Self(s.to_string()))
}
}

impl Key {
pub fn new_random() -> Self {
Self(Alphanumeric.sample_string(&mut rand::thread_rng(), 16))
}
}

Expand Down
17 changes: 17 additions & 0 deletions cargo-shuttle/src/provisioner_server.rs
Original file line number Diff line number Diff line change
Expand Up @@ -410,5 +410,22 @@ fn db_type_to_config(db_type: Type) -> EngineConfig {
"show databases;".to_string(),
],
},
Type::Local => EngineConfig {
// TODO: Update with real values
r#type: "local_sqlite".to_string(),
image: "docker.io/library/postgres:11".to_string(),
engine: "sqlite".to_string(),
username: "sqlite".to_string(),
password: "sqlite".to_string(),
database_name: "sqlite".to_string(),
port: "5432/tcp".to_string(),
env: Some(vec!["SQLITE_PASSWORD=sqlite".to_string()]),
is_ready_cmd: vec![],
// is_ready_cmd: vec![
// "/bin/sh".to_string(),
// "-c".to_string(),
// "pg_isready | grep 'accepting connections'".to_string(),
// ],
},
}
}
2 changes: 2 additions & 0 deletions common/src/database.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ use utoipa::ToSchema;
pub enum Type {
AwsRds(AwsRdsEngine),
Shared(SharedEngine),
Local,
}

#[derive(Clone, Debug, Deserialize, Display, Serialize, Eq, PartialEq)]
Expand All @@ -38,6 +39,7 @@ impl Display for Type {
match self {
Type::AwsRds(rds_type) => write!(f, "aws_rds::{rds_type}"),
Type::Shared(shared_type) => write!(f, "shared::{shared_type}"),
Type::Local => write!(f, "local"),
}
}
}
4 changes: 4 additions & 0 deletions deployer/src/persistence/resource/database.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ use strum::{Display, EnumString};
pub enum Type {
AwsRds(AwsRdsType),
Shared(SharedType),
Local,
}

#[derive(Clone, Copy, Debug, Display, EnumString, Eq, PartialEq)]
Expand All @@ -28,6 +29,7 @@ impl From<Type> for shuttle_common::database::Type {
match r#type {
Type::AwsRds(rds_type) => Self::AwsRds(rds_type.into()),
Type::Shared(shared_type) => Self::Shared(shared_type.into()),
Type::Local => Self::Local,
}
}
}
Expand Down Expand Up @@ -56,6 +58,7 @@ impl From<shuttle_common::database::Type> for Type {
match r#type {
shuttle_common::database::Type::AwsRds(rds_type) => Self::AwsRds(rds_type.into()),
shuttle_common::database::Type::Shared(shared_type) => Self::Shared(shared_type.into()),
shuttle_common::database::Type::Local => Self::Local,
}
}
}
Expand Down Expand Up @@ -84,6 +87,7 @@ impl Display for Type {
match self {
Type::AwsRds(rds_type) => write!(f, "aws_rds::{rds_type}"),
Type::Shared(shared_type) => write!(f, "shared::{shared_type}"),
Type::Local => write!(f, "local"),
}
}
}
Expand Down
3 changes: 3 additions & 0 deletions provisioner/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ use mongodb::{bson::doc, options::ClientOptions};
use rand::Rng;
use shuttle_common::claims::{Claim, Scope};
pub use shuttle_proto::provisioner::provisioner_server::ProvisionerServer;
use shuttle_proto::provisioner::Local;
use shuttle_proto::provisioner::{
aws_rds, database_request::DbType, shared, AwsRds, DatabaseRequest, DatabaseResponse, Shared,
};
Expand Down Expand Up @@ -425,6 +426,7 @@ impl Provisioner for MyProvisioner {
self.request_aws_rds(&request.project_name, engine.expect("oneof to be set"))
.await?
}
DbType::Local(Local {}) => todo!(),
};

Ok(Response::new(reply))
Expand All @@ -449,6 +451,7 @@ impl Provisioner for MyProvisioner {
self.delete_aws_rds(&request.project_name, engine.expect("oneof to be set"))
.await?
}
DbType::Local(Local {}) => todo!(),
};

Ok(Response::new(reply))
Expand Down
12 changes: 12 additions & 0 deletions resources/sqlite/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
[package]
name = "sqlite"
version = "0.1.0"
edition = "2021"

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[dependencies]
async-trait = "0.1.68"
serde = "1.0.163"
shuttle-service = "0.16.0"
shuttle-common = { version = "0.16.0", path = "../../common" }
49 changes: 49 additions & 0 deletions resources/sqlite/src/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
use async_trait::async_trait;
use serde::{Deserialize, Serialize};
use shuttle_service::{database, Factory, ResourceBuilder, Type};

// Builder struct
#[derive(Serialize)]
pub struct SQLite {}

// Resource struct
#[derive(Deserialize, Serialize, Clone)]
pub struct SQLiteInstance {}

#[async_trait]
impl ResourceBuilder<SQLiteInstance> for SQLite {
const TYPE: Type = Type::Database(database::Type::Local);

type Config = Self;

type Output = SQLiteInstance;

fn new() -> Self {
Self {}
}

fn config(&self) -> &Self::Config {
&self
}

async fn output(
self,
factory: &mut dyn Factory,
) -> Result<Self::Output, shuttle_service::Error> {
Ok(SQLiteInstance {})
}

async fn build(build_data: &Self::Output) -> Result<SQLiteInstance, shuttle_service::Error> {
Ok(build_data.clone())
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn it_works() {
//
}
}

0 comments on commit f5f407b

Please sign in to comment.