1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
use sqlx::sqlite::SqlitePool;
use super::repo::Provider as _;
use crate::{
clock::DateTime,
event::Broadcaster,
name::Name,
password::Password,
token::{Secret, Token, repo::Provider as _},
user::create::{self, Create},
};
#[derive(Clone)]
pub struct Setup {
db: SqlitePool,
events: Broadcaster,
}
impl Setup {
pub const fn new(db: SqlitePool, events: Broadcaster) -> Self {
Self { db, events }
}
pub async fn initial(
&self,
name: &Name,
password: &Password,
created_at: &DateTime,
) -> Result<Secret, Error> {
let create = Create::begin(name, password, created_at);
let validated = create.validate()?;
let mut tx = self.db.begin().await?;
if tx.setup().completed().await? {
Err(Error::SetupCompleted)
} else {
let stored = validated.store(&mut tx).await?;
let login = stored.login();
let (token, secret) = Token::generate(login, created_at);
tx.tokens().create(&token, &secret).await?;
tx.commit().await?;
stored.publish(&self.events);
Ok(secret)
}
}
pub async fn completed(&self) -> Result<bool, sqlx::Error> {
let mut tx = self.db.begin().await?;
let completed = tx.setup().completed().await?;
tx.commit().await?;
Ok(completed)
}
}
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("initial setup previously completed")]
SetupCompleted,
#[error("invalid user name: {0}")]
InvalidName(Name),
#[error(transparent)]
Database(#[from] sqlx::Error),
#[error(transparent)]
PasswordHash(#[from] password_hash::Error),
}
impl From<create::Error> for Error {
fn from(error: create::Error) -> Self {
match error {
create::Error::InvalidName(name) => Self::InvalidName(name),
create::Error::PasswordHash(error) => Self::PasswordHash(error),
}
}
}
|