303 lines
7.3 KiB
Rust
303 lines
7.3 KiB
Rust
use crate::{forms::LinkForm, ServerConfig, ServerError};
|
|
|
|
use argonautica::Hasher;
|
|
use dotenv::dotenv;
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
#[derive(PartialEq, Serialize, Clone, Debug)]
|
|
pub struct User {
|
|
pub id: i64,
|
|
pub username: String,
|
|
pub email: String,
|
|
pub password: String,
|
|
pub role: i64,
|
|
pub language: String,
|
|
}
|
|
|
|
impl User {
|
|
pub(crate) async fn get_user(
|
|
id: i64,
|
|
server_config: &ServerConfig,
|
|
) -> Result<Self, ServerError> {
|
|
let user = sqlx::query_as!(Self, "Select * from users where id = ? ", id)
|
|
.fetch_one(&server_config.db_pool)
|
|
.await;
|
|
user.map_err(ServerError::Database)
|
|
}
|
|
pub(crate) async fn get_user_by_name(
|
|
name: &str,
|
|
server_config: &ServerConfig,
|
|
) -> Result<Self, ServerError> {
|
|
let user = sqlx::query_as!(Self, "Select * from users where username = ? ", name)
|
|
.fetch_one(&server_config.db_pool)
|
|
.await;
|
|
user.map_err(ServerError::Database)
|
|
}
|
|
|
|
pub(crate) async fn get_all_users(
|
|
server_config: &ServerConfig,
|
|
) -> Result<Vec<Self>, ServerError> {
|
|
let user = sqlx::query_as!(Self, "Select * from users")
|
|
.fetch_all(&server_config.db_pool)
|
|
.await;
|
|
user.map_err(ServerError::Database)
|
|
}
|
|
|
|
pub(crate) async fn update_user(
|
|
&self,
|
|
server_config: &ServerConfig,
|
|
) -> Result<(), ServerError> {
|
|
sqlx::query!(
|
|
"UPDATE users SET
|
|
username = ?,
|
|
email = ?,
|
|
password = ?,
|
|
role = ? where id = ?",
|
|
self.username,
|
|
self.email,
|
|
self.password,
|
|
self.role,
|
|
self.id
|
|
)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
|
|
pub(crate) async fn toggle_admin(
|
|
self,
|
|
server_config: &ServerConfig,
|
|
) -> Result<(), ServerError> {
|
|
let new_role = 2 - (self.role + 1) % 2;
|
|
sqlx::query!("UPDATE users SET role = ? where id = ?", new_role, self.id)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
|
|
pub(crate) async fn set_language(
|
|
self,
|
|
server_config: &ServerConfig,
|
|
new_language: &str,
|
|
) -> Result<(), ServerError> {
|
|
sqlx::query!(
|
|
"UPDATE users SET language = ? where id = ?",
|
|
new_language,
|
|
self.id
|
|
)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
|
|
pub(crate) async fn count_admins(server_config: &ServerConfig) -> Result<Count, ServerError> {
|
|
let num = sqlx::query_as!(Count, "select count(*) as number from users where role = 2")
|
|
.fetch_one(&server_config.db_pool)
|
|
.await?;
|
|
Ok(num)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct NewUser {
|
|
pub username: String,
|
|
pub email: String,
|
|
pub password: String,
|
|
}
|
|
|
|
impl NewUser {
|
|
pub(crate) fn new(
|
|
username: String,
|
|
email: String,
|
|
password: &str,
|
|
config: &ServerConfig,
|
|
) -> Result<Self, ServerError> {
|
|
let hash = Self::hash_password(password, config)?;
|
|
|
|
Ok(Self {
|
|
username,
|
|
email,
|
|
password: hash,
|
|
})
|
|
}
|
|
|
|
pub(crate) fn hash_password(
|
|
password: &str,
|
|
config: &ServerConfig,
|
|
) -> Result<String, ServerError> {
|
|
dotenv().ok();
|
|
|
|
let secret = &config.secret;
|
|
|
|
let hash = Hasher::default()
|
|
.with_password(password)
|
|
.with_secret_key(secret)
|
|
.hash()?;
|
|
|
|
Ok(hash)
|
|
}
|
|
pub(crate) async fn insert_user(
|
|
&self,
|
|
server_config: &ServerConfig,
|
|
) -> Result<(), ServerError> {
|
|
sqlx::query!(
|
|
"Insert into users (
|
|
username,
|
|
email,
|
|
password,
|
|
role) VALUES (?,?,?,1)",
|
|
self.username,
|
|
self.email,
|
|
self.password,
|
|
)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserialize)]
|
|
pub struct LoginUser {
|
|
pub username: String,
|
|
pub password: String,
|
|
}
|
|
|
|
#[derive(Serialize, Debug)]
|
|
pub struct Link {
|
|
pub id: i64,
|
|
pub title: String,
|
|
pub target: String,
|
|
pub code: String,
|
|
pub author: i64,
|
|
pub created_at: chrono::NaiveDateTime,
|
|
}
|
|
|
|
impl Link {
|
|
pub(crate) async fn get_link_by_code(
|
|
code: &str,
|
|
server_config: &ServerConfig,
|
|
) -> Result<Self, ServerError> {
|
|
let link = sqlx::query_as!(Self, "Select * from links where code = ? ", code)
|
|
.fetch_one(&server_config.db_pool)
|
|
.await;
|
|
slog_info!(server_config.log, "Found link: {:?}", &link);
|
|
link.map_err(ServerError::Database)
|
|
}
|
|
|
|
pub(crate) async fn delete_link_by_code(
|
|
code: &str,
|
|
server_config: &ServerConfig,
|
|
) -> Result<(), ServerError> {
|
|
sqlx::query!("DELETE from links where code = ? ", code)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
pub(crate) async fn update_link(
|
|
&self,
|
|
server_config: &ServerConfig,
|
|
) -> Result<(), ServerError> {
|
|
sqlx::query!(
|
|
"UPDATE links SET
|
|
title = ?,
|
|
target = ?,
|
|
code = ?,
|
|
author = ?,
|
|
created_at = ? where id = ?",
|
|
self.title,
|
|
self.target,
|
|
self.code,
|
|
self.author,
|
|
self.created_at,
|
|
self.id
|
|
)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Serialize, Debug)]
|
|
pub struct NewLink {
|
|
pub title: String,
|
|
pub target: String,
|
|
pub code: String,
|
|
pub author: i64,
|
|
pub created_at: chrono::NaiveDateTime,
|
|
}
|
|
|
|
impl NewLink {
|
|
pub(crate) fn from_link_form(form: LinkForm, uid: i64) -> Self {
|
|
Self {
|
|
title: form.title,
|
|
target: form.target,
|
|
code: form.code,
|
|
author: uid,
|
|
created_at: chrono::Local::now().naive_utc(),
|
|
}
|
|
}
|
|
|
|
pub(crate) async fn insert(self, server_config: &ServerConfig) -> Result<(), ServerError> {
|
|
sqlx::query!(
|
|
"Insert into links (
|
|
title,
|
|
target,
|
|
code,
|
|
author,
|
|
created_at) VALUES (?,?,?,?,?)",
|
|
self.title,
|
|
self.target,
|
|
self.code,
|
|
self.author,
|
|
self.created_at,
|
|
)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Serialize, Debug)]
|
|
pub struct Click {
|
|
pub id: i64,
|
|
pub link: i64,
|
|
pub created_at: chrono::NaiveDateTime,
|
|
}
|
|
|
|
#[derive(Serialize)]
|
|
pub struct NewClick {
|
|
pub link: i64,
|
|
pub created_at: chrono::NaiveDateTime,
|
|
}
|
|
|
|
impl NewClick {
|
|
#[must_use]
|
|
pub fn new(link_id: i64) -> Self {
|
|
Self {
|
|
link: link_id,
|
|
created_at: chrono::Local::now().naive_utc(),
|
|
}
|
|
}
|
|
|
|
pub(crate) async fn insert_click(
|
|
self,
|
|
server_config: &ServerConfig,
|
|
) -> Result<(), ServerError> {
|
|
sqlx::query!(
|
|
"Insert into clicks (
|
|
link,
|
|
created_at) VALUES (?,?)",
|
|
self.link,
|
|
self.created_at,
|
|
)
|
|
.execute(&server_config.db_pool)
|
|
.await?;
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Serialize, Debug)]
|
|
pub struct Count {
|
|
pub number: i32,
|
|
}
|