Add editable user profiles

fixes: #5
This commit is contained in:
Dietrich 2021-02-08 12:39:28 +01:00
parent c9c29748b2
commit a71ba86e45
Signed by: dietrich
GPG Key ID: 9F3C20C0F85DF67C
7 changed files with 216 additions and 28 deletions

View File

@ -10,7 +10,6 @@ mod views;
use actix_identity::{CookieIdentityPolicy, IdentityService};
use actix_web::middleware::Logger;
use actix_web::{web, App, HttpResponse, HttpServer};
use actix_web_static_files;
use dotenv::dotenv;
use tera::Tera;
@ -124,7 +123,29 @@ async fn main() -> std::io::Result<()> {
// view an existing url
.service(
web::scope("/view")
.route("/{redirect_id}", web::get().to(views::view_link)),
.service(
web::scope("/link")
.route("/{redirect_id}", web::get().to(views::view_link)),
)
.service(
web::scope("/profile")
.route("/{user_id}", web::get().to(views::view_profile)),
),
)
.service(
web::scope("/edit")
.service(
web::scope("/link")
.route("/{redirect_id}", web::get().to(views::view_link)),
)
.service(
web::scope("/profile")
.route("/{user_id}", web::get().to(views::edit_profile))
.route(
"/{user_id}",
web::post().to(views::process_edit_profile),
),
),
)
.service(
web::scope("/download")

View File

@ -1,7 +1,7 @@
use crate::{forms::LinkForm, ServerError};
use super::schema::{links, users};
use argonautica::Hasher;
use argonautica::{Hasher, Verifier};
use diesel::{Insertable, Queryable};
use dotenv::dotenv;
use serde::{Deserialize, Serialize};
@ -28,22 +28,28 @@ impl NewUser {
email: String,
password: String,
) -> Result<Self, ServerError> {
let hash = Self::hash_password(password)?;
dotenv().ok();
let secret = std::env::var("SECRET_KEY")?;
let hash = Hasher::default()
.with_password(password)
.with_secret_key(secret)
.hash()
.unwrap();
Ok(NewUser {
username,
email,
password: hash,
})
}
pub(crate) fn hash_password(password: String) -> Result<String, ServerError> {
dotenv().ok();
let secret = std::env::var("SECRET_KEY")?;
let hash = Hasher::default()
.with_password(&password)
.with_secret_key(&secret)
.hash()?;
Ok(hash)
}
}
#[derive(Debug, Deserialize)]

View File

@ -1,19 +1,16 @@
use std::time::SystemTime;
use actix_identity::Identity;
use crate::ServerError;
use actix_identity::Identity;
use actix_web::{
http::header::{CacheControl, CacheDirective, ContentType, Expires},
web, HttpResponse,
};
use argonautica::Verifier;
use diesel::sqlite::SqliteConnection;
use diesel::{prelude::*, result::Error::NotFound};
use diesel::{prelude::*, result::Error::NotFound, sqlite::SqliteConnection};
use dotenv::dotenv;
use image::{DynamicImage, ImageOutputFormat, Luma};
use qrcode::render::svg;
use qrcode::QrCode;
use qrcode::{render::svg, QrCode};
use tera::{Context, Tera};
use super::forms::LinkForm;
@ -75,7 +72,6 @@ pub(crate) async fn view_link(
id: Identity,
link_id: web::Path<String>,
) -> Result<HttpResponse, ServerError> {
println!("Viewing link!");
use super::schema::links::dsl::{code, links};
if let Some(id) = id.identity() {
let connection = establish_connection()?;
@ -111,6 +107,104 @@ pub(crate) async fn view_link(
}
}
pub(crate) async fn view_profile(
tera: web::Data<Tera>,
identity: Identity,
user_id: web::Path<String>,
) -> Result<HttpResponse, ServerError> {
use super::schema::users::dsl::{id, users};
println!("Viewing Profile!");
if let Some(identity) = identity.identity() {
let connection = establish_connection()?;
if let Ok(uid) = user_id.parse::<i32>() {
let user = users.filter(id.eq(&uid)).first::<User>(&connection)?;
let mut data = Context::new();
data.insert("name", &identity);
data.insert(
"title",
&format!(
"Benutzer {} der Freien Hochschule Stuttgart",
&user.username
),
);
data.insert("user", &user);
let rendered = tera.render("view_profile.html", &data)?;
Ok(HttpResponse::Ok().body(rendered))
} else {
Ok(redirect_builder("/admin/index/"))
}
} else {
Ok(redirect_builder("/admin/login/"))
}
}
pub(crate) async fn edit_profile(
tera: web::Data<Tera>,
identity: Identity,
user_id: web::Path<String>,
) -> Result<HttpResponse, ServerError> {
use super::schema::users::dsl::{id, users};
println!("Viewing Profile!");
if let Some(identity) = identity.identity() {
let connection = establish_connection()?;
if let Ok(uid) = user_id.parse::<i32>() {
let user = users.filter(id.eq(&uid)).first::<User>(&connection)?;
let mut data = Context::new();
data.insert("name", &identity);
data.insert(
"title",
&format!(
"Benutzer {} der Freien Hochschule Stuttgart",
&user.username
),
);
data.insert("user", &user);
let rendered = tera.render("edit_profile.html", &data)?;
Ok(HttpResponse::Ok().body(rendered))
} else {
Ok(redirect_builder("/admin/index/"))
}
} else {
Ok(redirect_builder("/admin/login/"))
}
}
pub(crate) async fn process_edit_profile(
data: web::Form<NewUser>,
id: Identity,
user_id: web::Path<String>,
) -> Result<HttpResponse, ServerError> {
if let Some(_id) = id.identity() {
use super::schema::users::dsl::{email, id, password, username, users};
if let Ok(uid) = user_id.parse::<i32>() {
println!("Updating userinfo: ");
let connection = establish_connection()?;
diesel::update(users.filter(id.eq(uid)))
.set((
username.eq(data.username.clone()),
email.eq(data.email.clone()),
))
.execute(&connection)?;
if data.password.len() > 3 {
let hash = NewUser::hash_password(data.password.clone())?;
diesel::update(users.filter(id.eq(uid)))
.set((password.eq(hash),))
.execute(&connection)?;
}
Ok(HttpResponse::Ok().body(format!("Successfully saved user: {}", data.username)))
} else {
Ok(redirect_builder("/admin/index/"))
}
} else {
Ok(redirect_builder("/admin/login/"))
}
}
pub(crate) async fn download_png(
id: Identity,
link_id: web::Path<String>,
@ -210,23 +304,25 @@ pub(crate) async fn process_login(
Ok(u) => {
dotenv().ok();
let secret = std::env::var("SECRET_KEY")?;
let valid = Verifier::default()
.with_hash(u.password)
.with_password(data.password.clone())
.with_secret_key(secret)
.with_hash(&u.password)
.with_password(&data.password)
.with_secret_key(&secret)
.verify()?;
if valid {
println!("Login of user: {}", &u.username);
let session_token = u.username;
id.remember(session_token);
Ok(redirect_builder("/admin/index/"))
} else {
Ok(redirect_builder("/admin/login/"))
}
}
Err(_e) => Ok(redirect_builder("/admin/login/")),
Err(e) => {
println!("Failed to login: {}", e);
Ok(redirect_builder("/admin/login/"))
}
}
}

View File

@ -49,5 +49,26 @@ svg {
div.welcome {
text-align: right;
font-size: 24pt;
font-size: 16pt;
}
div.actions {
width: 400px;
height: 400px;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
padding: 30px;
color: #333;
}
a.button {
display:block;
padding: 15px;
text-align: center;
border-radius: 1px;
border: 1px solid rgb(90, 90, 90);
font-family: inherit;
background-color: #eae9ea;
}

View File

@ -0,0 +1,21 @@
{% extends "admin.html" %}
{% block admin %}
<h1>Profil von {{user.username}}</h1>
<form action="" method="POST">
<div>
<label for="username">Benutzername:</label>
<input type="text" name="username" value="{{ user.username }}">
</div>
<div>
<label for="email">E-mail:</label>
<input type="email" name="email" value="{{ user.email }}">
</div>
<div>
<label for="password">Passwort:</label>
<input type="password" name="password" placeholder="Leer lassen um nichts zu ändern">
</div>
<input type="submit" value="Editieren">
</form>
<h2>&nbsp;</h2>
{% endblock %}

View File

@ -7,16 +7,16 @@
{% set u = links_user[1] %}
<tr>
<td>
<a href="/admin/view/{{l.code}}"><span>{{l.code}}:</span>
<a href="/admin/view/link/{{l.code}}"><span>{{l.code}}:</span>
</a>
</td>
<td>
<a href="/admin/view/{{l.code}}">{{ l.target }}
<a href="/admin/view/link/{{l.code}}">{{ l.target }}
</a>
</td>
<td>
<a href="/admin/view/{{l.code}}"><small>{{ u.username }}</small>
<a href="/admin/view/profile/{{u.id}}"><small>{{ u.username }}</small>
</a>
</td>
</tr>

View File

@ -0,0 +1,23 @@
{% extends "admin.html" %}
{% block admin %}
<h1>Profil von {{user.username}}</h1>
<form action="" method="POST">
<div>
<label for="username">Benutzername:</label>
<input type="text" name="username" value="{{ user.username }}" readonly>
</div>
<div>
<label for="email">E-mail:</label>
<input type="email" name="email" value="{{ user.email }}" readonly>
</div>
<div>
<label for="password">Passwort:</label>
<input type="password" name="password" value="verschlüsselt" readonly>
</div>
</form>
<div class="actions">
<a class="button" href="/admin/edit/profile/{{ user.id }}">Editieren</a>
</div>
<h2>&nbsp;</h2>
{% endblock %}