refactor security to Flask-Security - Part 1
This commit is contained in:
27
config.py
27
config.py
@@ -7,9 +7,25 @@ class Config(object):
|
|||||||
DEBUG = False
|
DEBUG = False
|
||||||
DEVELOPMENT = False
|
DEVELOPMENT = False
|
||||||
SECRET_KEY = '97867c1491bea5ee6a8e8436eb11bf2ba6a69ff53ab1b17ecba450d0f2e572e1'
|
SECRET_KEY = '97867c1491bea5ee6a8e8436eb11bf2ba6a69ff53ab1b17ecba450d0f2e572e1'
|
||||||
JWT_SECRET_KEY = '60a4ba120437004cfc8fc1cf571150f16d950d31aa7c5a4a2fe7a262d4d24bec'
|
|
||||||
JWT_TOKEN_LOCATION = ['cookies']
|
# flask-security-too settings
|
||||||
JWT_COOKIE_SECURE = True
|
SECURITY_PASSWORD_SALT = '228614859439123264035565568761433607235'
|
||||||
|
REMEMBER_COOKIE_SAMESITE = 'strict'
|
||||||
|
SESSION_COOKIE_SAMESITE = 'strict'
|
||||||
|
SECURITY_CONFIRMABLE = True
|
||||||
|
SECURITY_TRACKABLE = True
|
||||||
|
SECURITY_PASSWORD_COMPLEXITY_CHECKER = 'zxcvbn'
|
||||||
|
SECURITY_POST_LOGIN_VIEW = '/user/tenant'
|
||||||
|
SECURITY_REGISTERABLE = False
|
||||||
|
SECURITY_LOGINABLE = False
|
||||||
|
SECURITY_LOGOUTABLE = False
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# flask-mailman settings
|
||||||
|
MAIL_SERVER = 'mail.flow-it.net'
|
||||||
|
MAIL_PORT = 465
|
||||||
|
MAIL_USE_TLS = True
|
||||||
|
|
||||||
|
|
||||||
class DevConfig(Config):
|
class DevConfig(Config):
|
||||||
@@ -18,7 +34,10 @@ class DevConfig(Config):
|
|||||||
SQLALCHEMY_DATABASE_URI = 'postgresql+pg8000://josako@localhost:5432/eveAI'
|
SQLALCHEMY_DATABASE_URI = 'postgresql+pg8000://josako@localhost:5432/eveAI'
|
||||||
SQLALCHEMY_BINDS = {'public': 'postgresql+pg8000://josako@localhost:5432/eveAI'}
|
SQLALCHEMY_BINDS = {'public': 'postgresql+pg8000://josako@localhost:5432/eveAI'}
|
||||||
EXPLAIN_TEMPLATE_LOADING = True
|
EXPLAIN_TEMPLATE_LOADING = True
|
||||||
JWT_COOKIE_SECURE = False
|
|
||||||
|
# flask-mailman settings
|
||||||
|
MAIL_USERNAME = 'eveai_admin@flow-it.net'
|
||||||
|
MAIL_PASSWORD = 'FgV650K3ow#5FeBcZc5'
|
||||||
|
|
||||||
|
|
||||||
class ProdConfig(Config):
|
class ProdConfig(Config):
|
||||||
|
|||||||
@@ -1,12 +1,16 @@
|
|||||||
import os
|
import os
|
||||||
from flask import Flask
|
from flask import Flask
|
||||||
from .extensions import db, migrate, bcrypt, bootstrap, jwt
|
from flask_security import SQLAlchemyUserDatastore
|
||||||
from .models.user import User, Tenant
|
from werkzeug.middleware.proxy_fix import ProxyFix
|
||||||
|
|
||||||
|
from .extensions import db, migrate, bootstrap, security, mail, login_manager
|
||||||
|
from .models.user import User, Tenant, Role
|
||||||
from .models.document import Document, DocumentLanguage, DocumentVersion
|
from .models.document import Document, DocumentLanguage, DocumentVersion
|
||||||
|
|
||||||
|
|
||||||
def create_app(config_file=None):
|
def create_app(config_file=None):
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
|
app.wsgi_app = ProxyFix(app.wsgi_app, x_for=1)
|
||||||
|
|
||||||
if config_file is None:
|
if config_file is None:
|
||||||
app.config.from_object('config.DevConfig')
|
app.config.from_object('config.DevConfig')
|
||||||
@@ -19,6 +23,10 @@ def create_app(config_file=None):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
register_extensions(app)
|
register_extensions(app)
|
||||||
|
# Setup Flask-Security-Too
|
||||||
|
user_datastore = SQLAlchemyUserDatastore(db, User, Role)
|
||||||
|
security.init_app(app, user_datastore)
|
||||||
|
|
||||||
register_blueprints(app)
|
register_blueprints(app)
|
||||||
|
|
||||||
print(app.config.get('SQLALCHEMY_DATABASE_URI'))
|
print(app.config.get('SQLALCHEMY_DATABASE_URI'))
|
||||||
@@ -28,11 +36,12 @@ def create_app(config_file=None):
|
|||||||
def register_extensions(app):
|
def register_extensions(app):
|
||||||
db.init_app(app)
|
db.init_app(app)
|
||||||
migrate.init_app(app, db)
|
migrate.init_app(app, db)
|
||||||
bcrypt.init_app(app)
|
|
||||||
bootstrap.init_app(app)
|
bootstrap.init_app(app)
|
||||||
jwt.init_app(app)
|
mail.init_app(app)
|
||||||
|
login_manager.init_app(app)
|
||||||
|
|
||||||
|
|
||||||
|
# Register Blueprints
|
||||||
def register_blueprints(app):
|
def register_blueprints(app):
|
||||||
from .views.user_views import user_bp
|
from .views.user_views import user_bp
|
||||||
app.register_blueprint(user_bp)
|
app.register_blueprint(user_bp)
|
||||||
|
|||||||
@@ -1,14 +1,17 @@
|
|||||||
from flask_sqlalchemy import SQLAlchemy
|
from flask_sqlalchemy import SQLAlchemy
|
||||||
from flask_migrate import Migrate
|
from flask_migrate import Migrate
|
||||||
from flask_bcrypt import Bcrypt
|
|
||||||
from flask_bootstrap import Bootstrap
|
from flask_bootstrap import Bootstrap
|
||||||
from flask_jwt_extended import JWTManager
|
from flask_security import Security
|
||||||
|
from flask_mailman import Mail
|
||||||
|
from flask_login import LoginManager
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Create extensions
|
# Create extensions
|
||||||
|
|
||||||
db = SQLAlchemy()
|
db = SQLAlchemy()
|
||||||
migrate = Migrate()
|
migrate = Migrate()
|
||||||
bcrypt = Bcrypt()
|
|
||||||
bootstrap = Bootstrap()
|
bootstrap = Bootstrap()
|
||||||
jwt = JWTManager()
|
security = Security()
|
||||||
|
mail = Mail()
|
||||||
|
login_manager = LoginManager()
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
from ..extensions import db
|
from ..extensions import db
|
||||||
|
from flask_security import UserMixin, RoleMixin
|
||||||
|
|
||||||
|
|
||||||
class Tenant(db.Model):
|
class Tenant(db.Model):
|
||||||
@@ -28,7 +29,24 @@ class Tenant(db.Model):
|
|||||||
return '<Tenant %r>' % self.name
|
return '<Tenant %r>' % self.name
|
||||||
|
|
||||||
|
|
||||||
class User(db.Model):
|
class Role(db.Model, RoleMixin):
|
||||||
|
__bind_key__ = 'public'
|
||||||
|
__table_args__ = {'schema': 'public'}
|
||||||
|
|
||||||
|
id = db.Column(db.Integer(), primary_key=True)
|
||||||
|
name = db.Column(db.String(80), unique=True)
|
||||||
|
description = db.Column(db.String(255))
|
||||||
|
|
||||||
|
|
||||||
|
class RolesUsers(db.Model):
|
||||||
|
__bind_key__ = 'public'
|
||||||
|
__table_args__ = {'schema': 'public'}
|
||||||
|
|
||||||
|
user_id = db.Column(db.Integer(), db.ForeignKey('public.user.id', ondelete='CASCADE'), primary_key=True)
|
||||||
|
role_id = db.Column(db.Integer(), db.ForeignKey('public.role.id', ondelete='CASCADE'), primary_key=True)
|
||||||
|
|
||||||
|
|
||||||
|
class User(db.Model, UserMixin):
|
||||||
"""User model"""
|
"""User model"""
|
||||||
|
|
||||||
__bind_key__ = 'public'
|
__bind_key__ = 'public'
|
||||||
@@ -46,17 +64,19 @@ class User(db.Model):
|
|||||||
first_name = db.Column(db.String(80), nullable=False)
|
first_name = db.Column(db.String(80), nullable=False)
|
||||||
last_name = db.Column(db.String(80), nullable=False)
|
last_name = db.Column(db.String(80), nullable=False)
|
||||||
is_active = db.Column(db.Boolean, default=True)
|
is_active = db.Column(db.Boolean, default=True)
|
||||||
is_tester = db.Column(db.Boolean, default=False)
|
fs_uniquifier = db.Column(db.String(255), unique=True, nullable=False)
|
||||||
is_admin = db.Column(db.Boolean, default=False)
|
|
||||||
is_super = db.Column(db.Boolean, default=False)
|
|
||||||
confirmed_at = db.Column(db.DateTime, nullable=True)
|
confirmed_at = db.Column(db.DateTime, nullable=True)
|
||||||
valid_to = db.Column(db.Date, nullable=True)
|
valid_to = db.Column(db.Date, nullable=True)
|
||||||
|
|
||||||
# Login Information
|
# Security Trackable Information
|
||||||
last_login = db.Column(db.DateTime, nullable=True)
|
last_login_at = db.Column(db.DateTime, nullable=True)
|
||||||
authenticated = db.Column(db.Boolean, default=False)
|
current_login_at = db.Column(db.DateTime, nullable=True)
|
||||||
|
last_login_ip = db.Column(db.String(255), nullable=True)
|
||||||
|
current_login_ip = db.Column(db.String(255), nullable=True)
|
||||||
|
login_count = db.Column(db.Integer, nullable=False, default=0)
|
||||||
|
|
||||||
# Relations
|
# Relations
|
||||||
|
roles = db.relationship('Role', secondary='public.roles_users', backref=db.backref('users', lazy='dynamic'))
|
||||||
tenant_id = db.Column(db.Integer, db.ForeignKey('public.tenant.id'), nullable=False)
|
tenant_id = db.Column(db.Integer, db.ForeignKey('public.tenant.id'), nullable=False)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
|||||||
17
eveai_app/templates/login_user.html
Normal file
17
eveai_app/templates/login_user.html
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
{% block title %}Login{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<form action="" method="post" novalidate>
|
||||||
|
<p>
|
||||||
|
{{ form.email.label }}<br>
|
||||||
|
{{ form.email(size=80) }}
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
{{ form.password.label }}<br>
|
||||||
|
{{ form.password(size=80) }}
|
||||||
|
</p>
|
||||||
|
<p>{{ form.submit() }}</p>
|
||||||
|
</form>
|
||||||
|
{% endblock %}
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
from functools import wraps
|
|
||||||
from flask_jwt_extended import get_jwt, verify_jwt_in_request
|
|
||||||
|
|
||||||
|
|
||||||
def super_required():
|
|
||||||
def wrapper(fn):
|
|
||||||
@wraps(fn)
|
|
||||||
def decorator(*args, **kwargs):
|
|
||||||
verify_jwt_in_request()
|
|
||||||
claims = get_jwt()
|
|
||||||
if not claims['is_super']:
|
|
||||||
return {'message': 'Authentication Error: Super users only!'}, 403
|
|
||||||
else:
|
|
||||||
return fn(*args, **kwargs)
|
|
||||||
|
|
||||||
return decorator
|
|
||||||
return wrapper
|
|
||||||
|
|
||||||
|
|
||||||
# Decorators
|
|
||||||
|
|
||||||
|
|
||||||
def admin_required():
|
|
||||||
def wrapper(fn):
|
|
||||||
@wraps(fn)
|
|
||||||
def decorator(*args, **kwargs):
|
|
||||||
verify_jwt_in_request()
|
|
||||||
claims = get_jwt()
|
|
||||||
if not claims['is_admin']:
|
|
||||||
return {'message': 'Authentication Error: Admins only!'}, 403
|
|
||||||
else:
|
|
||||||
return fn(*args, **kwargs)
|
|
||||||
|
|
||||||
return decorator
|
|
||||||
return wrapper
|
|
||||||
|
|
||||||
|
|
||||||
def tester_required():
|
|
||||||
def wrapper(fn):
|
|
||||||
@wraps(fn)
|
|
||||||
def decorator(*args, **kwargs):
|
|
||||||
verify_jwt_in_request()
|
|
||||||
claims = get_jwt()
|
|
||||||
if not claims['is_tester']:
|
|
||||||
return {'message': 'Authentication Error: Testers only!'}, 403
|
|
||||||
else:
|
|
||||||
return fn(*args, **kwargs)
|
|
||||||
|
|
||||||
return decorator
|
|
||||||
return wrapper
|
|
||||||
@@ -6,5 +6,5 @@ from wtforms.validators import DataRequired, Length, Email
|
|||||||
class LoginForm(FlaskForm):
|
class LoginForm(FlaskForm):
|
||||||
email = EmailField('Email', validators=[DataRequired(), Email()])
|
email = EmailField('Email', validators=[DataRequired(), Email()])
|
||||||
password = PasswordField('Password', validators=[DataRequired(), Length(min=8)])
|
password = PasswordField('Password', validators=[DataRequired(), Length(min=8)])
|
||||||
# remember_me = BooleanField('Remember me')
|
remember_me = BooleanField('Remember me')
|
||||||
submit = SubmitField('Login')
|
submit = SubmitField('Login')
|
||||||
|
|||||||
@@ -1,65 +1,46 @@
|
|||||||
from datetime import datetime as dt, timezone as tz
|
from datetime import datetime as dt, timezone as tz
|
||||||
from flask import request, redirect, url_for, flash, render_template, Blueprint, jsonify, session
|
from flask import request, redirect, url_for, flash, render_template, Blueprint, jsonify, session
|
||||||
from ..models.user import User, Tenant
|
from flask_security import login_user, logout_user
|
||||||
from ..extensions import db, bcrypt
|
|
||||||
from .auth_forms import LoginForm
|
|
||||||
from flask_jwt_extended import (create_access_token, create_refresh_token, set_access_cookies, set_refresh_cookies,
|
|
||||||
unset_jwt_cookies)
|
|
||||||
|
|
||||||
auth_bp = Blueprint('auth_bp', __name__)
|
from ..models.user import User, Tenant
|
||||||
|
from .auth_forms import LoginForm
|
||||||
|
|
||||||
|
auth_bp = Blueprint('auth_bp', __name__, template_folder='templates')
|
||||||
|
|
||||||
|
|
||||||
@auth_bp.route('/login', methods=['GET', 'POST'])
|
@auth_bp.route('/login', methods=['GET', 'POST'])
|
||||||
def login():
|
def login():
|
||||||
if request.method == 'POST':
|
form = LoginForm()
|
||||||
email = request.form.get('email')
|
if form.validate_on_submit():
|
||||||
password = request.form.get('password')
|
email = form.email.data
|
||||||
# remember_me = True if request.form.get('remember_me') else False
|
password = form.password.data
|
||||||
|
remember_me = True if form.remember_me.data else False
|
||||||
|
|
||||||
user = User.query.filter_by(email=email).first()
|
user = User.query.filter_by(email=email).first()
|
||||||
tenant = Tenant.query.filter_by(id=user.tenant_id).first()
|
tenant = Tenant.query.filter_by(id=user.tenant_id).first()
|
||||||
if user:
|
if user and user.verify_and_update_password(password):
|
||||||
if user.is_active:
|
if user.is_active:
|
||||||
if bcrypt.check_password_hash(user.password, password):
|
login_user(user, remember=remember_me)
|
||||||
response = jsonify({'msg': 'Login Successful'})
|
next_page = request.args.get('next')
|
||||||
flash('Logged in successfully!', category='success')
|
|
||||||
|
|
||||||
# set session information
|
session['tenant_id'] = user.tenant_id
|
||||||
# session['user_id'] = user.id
|
session['tenant_name'] = tenant.name
|
||||||
# session['user_name'] = user.user_name
|
|
||||||
# session['email'] = user.email
|
|
||||||
# session['tenant_id'] = user.tenant_id
|
|
||||||
# session['tenant_name'] = tenant.name
|
|
||||||
|
|
||||||
# set JWT header information
|
return redirect(next_page)
|
||||||
additional_claims = {'tenant': user.tenant_id,
|
|
||||||
'is_super': user.is_super,
|
|
||||||
'is_admin': user.is_admin,
|
|
||||||
'is_tester': user.is_tester}
|
|
||||||
access_token = create_access_token(
|
|
||||||
identity=user.id,
|
|
||||||
additional_claims=additional_claims)
|
|
||||||
refresh_token = create_refresh_token(
|
|
||||||
identity=user.id,
|
|
||||||
additional_claims=additional_claims)
|
|
||||||
set_access_cookies(response, access_token)
|
|
||||||
set_refresh_cookies(response, refresh_token)
|
|
||||||
response.headers['Location'] = url_for('user_bp.user')
|
|
||||||
|
|
||||||
return response, 302
|
|
||||||
else:
|
|
||||||
flash('Incorrect email/password combination, try again.', category='error')
|
|
||||||
else:
|
else:
|
||||||
flash('Account disabled. Please contact your administrator.', category='error')
|
flash('Account disabled. Please contact your administrator.', category='error')
|
||||||
else:
|
else:
|
||||||
flash('Incorrect email/password combination, try again.', category='error')
|
flash('Invalid email or password.', category='error')
|
||||||
|
|
||||||
form = LoginForm()
|
|
||||||
return render_template('login.html', form=form)
|
return render_template('login.html', form=form)
|
||||||
|
|
||||||
|
|
||||||
@auth_bp.route('/logout', methods=['POST'])
|
@auth_bp.route('/logout', methods=['POST'])
|
||||||
def logout():
|
def logout():
|
||||||
response = jsonify({'msg': 'Logout Successful'})
|
logout_user()
|
||||||
unset_jwt_cookies(response)
|
|
||||||
|
# Clear session data
|
||||||
|
session.pop('tenant_id', None)
|
||||||
|
session.pop('tenant_name', None)
|
||||||
|
|
||||||
return redirect(url_for('/'))
|
return redirect(url_for('/'))
|
||||||
|
|||||||
@@ -16,12 +16,10 @@ class UserForm(FlaskForm):
|
|||||||
user_name = StringField('Name', validators=[DataRequired(), Length(max=80)])
|
user_name = StringField('Name', validators=[DataRequired(), Length(max=80)])
|
||||||
email = EmailField('Email', validators=[DataRequired(), Email()])
|
email = EmailField('Email', validators=[DataRequired(), Email()])
|
||||||
password = PasswordField('Password', validators=[DataRequired(), Length(min=8)])
|
password = PasswordField('Password', validators=[DataRequired(), Length(min=8)])
|
||||||
|
confirm_password = PasswordField('Confirm Password', validators=[DataRequired(), Length(min=8)])
|
||||||
first_name = StringField('First Name', validators=[DataRequired(), Length(max=80)])
|
first_name = StringField('First Name', validators=[DataRequired(), Length(max=80)])
|
||||||
last_name = StringField('Last Name', validators=[DataRequired(), Length(max=80)])
|
last_name = StringField('Last Name', validators=[DataRequired(), Length(max=80)])
|
||||||
is_active = BooleanField('Is Active')
|
is_active = BooleanField('Is Active')
|
||||||
is_tester = BooleanField('Is Tester')
|
|
||||||
is_admin = BooleanField('Is Administrator')
|
|
||||||
is_super = BooleanField('Is Super User')
|
|
||||||
valid_to = DateField('Valid to', id='datepicker')
|
valid_to = DateField('Valid to', id='datepicker')
|
||||||
tenant_id = IntegerField('Tenant ID', validators=[NumberRange(min=0)])
|
tenant_id = IntegerField('Tenant ID', validators=[NumberRange(min=0)])
|
||||||
submit = SubmitField('Submit')
|
submit = SubmitField('Submit')
|
||||||
|
|||||||
@@ -1,18 +1,17 @@
|
|||||||
# from . import user_bp
|
# from . import user_bp
|
||||||
from datetime import datetime as dt, timezone as tz
|
from datetime import datetime as dt, timezone as tz
|
||||||
from flask import request, redirect, url_for, flash, render_template, Blueprint, session
|
from flask import request, redirect, url_for, flash, render_template, Blueprint, session
|
||||||
from flask_jwt_extended import verify_jwt_in_request, get_jwt, get_jwt_identity, jwt_required
|
from flask_security import hash_password
|
||||||
|
|
||||||
from ..models.user import User, Tenant
|
from ..models.user import User, Tenant
|
||||||
from ..extensions import db, bcrypt
|
from ..extensions import db
|
||||||
from .user_forms import TenantForm, UserForm
|
from .user_forms import TenantForm, UserForm
|
||||||
from ..utils.database import Database
|
from ..utils.database import Database
|
||||||
from ..utils.security import admin_required, super_required, tester_required
|
|
||||||
|
|
||||||
user_bp = Blueprint('user_bp', __name__, url_prefix='/user')
|
user_bp = Blueprint('user_bp', __name__, url_prefix='/user')
|
||||||
|
|
||||||
|
|
||||||
@user_bp.route('/tenant', methods=['GET', 'POST'])
|
@user_bp.route('/tenant', methods=['GET', 'POST'])
|
||||||
@super_required
|
|
||||||
def tenant():
|
def tenant():
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
# Handle the required attributes
|
# Handle the required attributes
|
||||||
@@ -65,61 +64,39 @@ def tenant():
|
|||||||
|
|
||||||
|
|
||||||
@user_bp.route('/user', methods=['GET', 'POST'])
|
@user_bp.route('/user', methods=['GET', 'POST'])
|
||||||
@admin_required
|
|
||||||
@jwt_required()
|
|
||||||
def user():
|
def user():
|
||||||
if request.method == 'POST':
|
|
||||||
# Handle the required attributes
|
|
||||||
username = request.form.get('user_name')
|
|
||||||
email = request.form.get('email')
|
|
||||||
password = request.form.get('password')
|
|
||||||
first_name = request.form.get('first_name')
|
|
||||||
last_name = request.form.get('last_name')
|
|
||||||
error = None
|
|
||||||
|
|
||||||
if not username:
|
|
||||||
error = 'Username is required.'
|
|
||||||
elif not email:
|
|
||||||
error = 'Email is required.'
|
|
||||||
elif not password:
|
|
||||||
error = 'Password is required.'
|
|
||||||
elif not first_name:
|
|
||||||
error = 'First name is required.'
|
|
||||||
elif not last_name:
|
|
||||||
error = 'Last name is required.'
|
|
||||||
if error is None:
|
|
||||||
password_hash = bcrypt.generate_password_hash(password).decode('utf-8')
|
|
||||||
|
|
||||||
# Create new user if there is no error
|
|
||||||
new_user = User(user_name=username, email=email, password=password_hash, first_name=first_name,
|
|
||||||
last_name=last_name)
|
|
||||||
|
|
||||||
# Handle optional attributes
|
|
||||||
new_user.is_active = bool(request.form.get('is_active'))
|
|
||||||
new_user.is_tester = bool(request.form.get('is_tester'))
|
|
||||||
new_user.is_admin = bool(request.form.get('is_admin'))
|
|
||||||
new_user.is_super = bool(request.form.get('is_super'))
|
|
||||||
new_user.valid_to = request.form.get('valid_to')
|
|
||||||
|
|
||||||
# Handle Timestamps
|
|
||||||
timestamp = dt.now(tz.utc)
|
|
||||||
new_user.created_at = timestamp
|
|
||||||
new_user.updated_at = timestamp
|
|
||||||
|
|
||||||
# Handle the relations
|
|
||||||
tenant_id = request.form.get('tenant_id')
|
|
||||||
the_tenant = Tenant.query.get(tenant_id)
|
|
||||||
new_user.tenant = the_tenant
|
|
||||||
|
|
||||||
# Add the new user to the database and commit the changes
|
|
||||||
|
|
||||||
try:
|
|
||||||
db.session.add(new_user)
|
|
||||||
db.session.commit()
|
|
||||||
except Exception as e:
|
|
||||||
error = e.args
|
|
||||||
|
|
||||||
flash(error) if error else flash('User added successfully.')
|
|
||||||
|
|
||||||
form = UserForm()
|
form = UserForm()
|
||||||
|
if form.validate_on_submit():
|
||||||
|
hashed_password = hash_password(form.password.data)
|
||||||
|
new_user = User(
|
||||||
|
user_name=form.user_name.data,
|
||||||
|
email=form.email.data,
|
||||||
|
password=hashed_password,
|
||||||
|
first_name=form.first_name.data,
|
||||||
|
last_name=form.last_name.data,
|
||||||
|
is_active=form.is_active.data,
|
||||||
|
valid_to=form.valid_to.data,
|
||||||
|
tenant_id=form.tenant_id.data
|
||||||
|
)
|
||||||
|
|
||||||
|
timestamp = dt.now(tz.utc)
|
||||||
|
new_user.created_at = timestamp
|
||||||
|
new_user.updated_at = timestamp
|
||||||
|
|
||||||
|
# Handle the relations
|
||||||
|
tenant_id = request.form.get('tenant_id')
|
||||||
|
the_tenant = Tenant.query.get(tenant_id)
|
||||||
|
new_user.tenant = the_tenant
|
||||||
|
|
||||||
|
# Add the new user to the database and commit the changes
|
||||||
|
|
||||||
|
try:
|
||||||
|
db.session.add(new_user)
|
||||||
|
db.session.commit()
|
||||||
|
flash('User added successfully.')
|
||||||
|
# return redirect(url_for('user/user'))
|
||||||
|
except Exception as e:
|
||||||
|
db.session.rollback()
|
||||||
|
flash(f'Failed to add user. Error: {str(e)}')
|
||||||
|
|
||||||
return render_template('user/user.html', form=form)
|
return render_template('user/user.html', form=form)
|
||||||
|
|||||||
61
templates/base.html
Normal file
61
templates/base.html
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en" itemscope itemtype="http://schema.org/WebPage">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
|
||||||
|
<link rel="apple-touch-icon" sizes="76x76" href="{{url_for('static', filename='assets/img/apple-icon.png')}}">
|
||||||
|
<link rel="icon" type="image/png" href="{{url_for('static', filename='./assets/img/favicon.png')}}">
|
||||||
|
<title>
|
||||||
|
{% block title %}{% endblock %}
|
||||||
|
</title>
|
||||||
|
<!-- Fonts and icons -->
|
||||||
|
<link rel="stylesheet" type="text/css" href="https://fonts.googleapis.com/css?family=Roboto:300,400,500,700,900|Roboto+Slab:400,700" />
|
||||||
|
<!-- Nucleo Icons -->
|
||||||
|
<link href="{{url_for('static', filename='/assets/css/nucleo-icons.css" rel="stylesheet')}}" />
|
||||||
|
<link href="{{url_for('static', filename='/assets/css/nucleo-svg.css" rel="stylesheet')}}" />
|
||||||
|
<!-- Font Awesome Icons -->
|
||||||
|
<script src="https://kit.fontawesome.com/42d5adcbca.js" crossorigin="anonymous"></script>
|
||||||
|
<!-- Material Icons -->
|
||||||
|
<link href="https://fonts.googleapis.com/icon?family=Material+Icons+Round" rel="stylesheet">
|
||||||
|
<!-- CSS Files -->
|
||||||
|
<link id="pagestyle" href="{{url_for('static', filename='/assets/css/material-kit-pro.css')}}" rel="stylesheet" />
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body class="presentation-page bg-gray-200">
|
||||||
|
{% include 'navbar.html' %}
|
||||||
|
{% include 'header.html' %}
|
||||||
|
{% with messages = get_flashed_messages() %}
|
||||||
|
{% if messages%}
|
||||||
|
{% for message in messages%}
|
||||||
|
<p>{{message}}</p>
|
||||||
|
{%endfor%}
|
||||||
|
{%endif%}
|
||||||
|
{%endwith%}
|
||||||
|
<hr>
|
||||||
|
<div class="container">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-md-12">
|
||||||
|
<span></span>
|
||||||
|
</div>
|
||||||
|
<div class="col-md-12">
|
||||||
|
<span>
|
||||||
|
{% block content %}{% endblock %}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="col-md-12">
|
||||||
|
<span></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
{# {% include 'footer.html' %}#}
|
||||||
|
|
||||||
|
<!-- Optional JavaScript -->
|
||||||
|
<!-- jQuery first, then Popper.js, then Bootstrap JS -->
|
||||||
|
<script src="https://code.jquery.com/jquery-3.3.1.slim.min.js" integrity="sha384-q8i/X+965DzO0rT7abK41JStQIAqVgRVzpbzo5smXKp4YfRvH+8abtTE1Pi6jizo" crossorigin="anonymous"></script>
|
||||||
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.14.7/umd/popper.min.js" integrity="sha384-UO2eT0CpHqdSJQ6hJty5KVphtPhzWj9WO1clHTMGa3JDZwrnQq4sF86dIHNDz0W1" crossorigin="anonymous"></script>
|
||||||
|
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js" integrity="sha384-JjSmVgyd0p3pXB1rRibZUAYoIIy6OrQ6VrjIEaFf/nJGzIxFDsf4x0xIM+B07jRM" crossorigin="anonymous"></script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
13
templates/header.html
Normal file
13
templates/header.html
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<header class="header-2">
|
||||||
|
<div class="page-header min-vh-75" style="background-image: url({{url_for('static', filename='/assets/img/EveAI_bg1.jpg')}})" loading="lazy">
|
||||||
|
<span class="mask bg-gradient-primary opacity-4"></span>
|
||||||
|
<div class="container">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-7 text-center mx-auto">
|
||||||
|
<h1 class="text-white pt-3 mt-n5">EveAI Virtual Assistant</h1>
|
||||||
|
<p class="lead text-white mt-3 px-5">Enhance Customer Interaction with AI</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
17
templates/login.html
Normal file
17
templates/login.html
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
{% block title %}Login{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<form action="" method="post" novalidate>
|
||||||
|
<p>
|
||||||
|
{{ form.email.label }}<br>
|
||||||
|
{{ form.email(size=80) }}
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
{{ form.password.label }}<br>
|
||||||
|
{{ form.password(size=80) }}
|
||||||
|
</p>
|
||||||
|
<p>{{ form.submit() }}</p>
|
||||||
|
</form>
|
||||||
|
{% endblock %}
|
||||||
17
templates/login_user.html
Normal file
17
templates/login_user.html
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
{% block title %}Login{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<form action="" method="post" novalidate>
|
||||||
|
<p>
|
||||||
|
{{ form.email.label }}<br>
|
||||||
|
{{ form.email(size=80) }}
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
{{ form.password.label }}<br>
|
||||||
|
{{ form.password(size=80) }}
|
||||||
|
</p>
|
||||||
|
<p>{{ form.submit() }}</p>
|
||||||
|
</form>
|
||||||
|
{% endblock %}
|
||||||
68
templates/navbar.html
Normal file
68
templates/navbar.html
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
<!-- Navbar Light -->
|
||||||
|
<div class="navbar navbar-expand-lg navbar-light bg-white z-index-3 py-3">
|
||||||
|
<div class="container position-sticky z-index-sticky top-0">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-12">
|
||||||
|
<nav class="navbar navbar-expand-lg blur border-radius-xl top-0 z-index-fixed shadow position-absolute my-3 py-2 start-0 end-0 mx-4">
|
||||||
|
<div class="container-fluid px-0">
|
||||||
|
<a class="navbar-brand font-weight-bolder ms-sm-3 d-none d-md-block" href="https://www.flow-it.net" rel="tooltip" title="Idee Generated & Implemented by Flow IT" data-placement="bottom">
|
||||||
|
EveAI
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="collapse navbar-collapse w-100 pt-3 pb-2 py-lg-0" id="navigation">
|
||||||
|
<ul class="navbar-nav navbar-nav-hover mx-auto">
|
||||||
|
<li class="nav-item dropdown dropdown-hover mx-2">
|
||||||
|
<a role="button" class="nav-link ps-2 d-flex cursor-pointer align-items-center" id="dropdownMenuUser" data-bs-toggle="dropdown" aria-expanded="false">
|
||||||
|
<i class="material-icons opacity-6 me-2 text-md">contacts</i>
|
||||||
|
User Mgmt
|
||||||
|
<img src="{{url_for('static', filename='/assets/img/down-arrow-dark.svg')}}" alt="down-arrow" class="arrow ms-2 d-lg-block d-none">
|
||||||
|
<img src="{{url_for('static', filename='/assets/img/down-arrow-dark.svg')}}" alt="down-arrow" class="arrow ms-1 d-lg-none d-block ms-auto">
|
||||||
|
</a>
|
||||||
|
<div class="dropdown-menu dropdown-menu-animation ms-n3 dropdown-md p-3 border-radius-xl mt-0 mt-lg-3" aria-labelledby="dropdownMenuPages">
|
||||||
|
<div class="d-none d-lg-flex">
|
||||||
|
<ul class="list-group w-100">
|
||||||
|
<li class="nav-item dropdown dropdown-hover dropdown-subitem list-group-item border-0 p-0">
|
||||||
|
<a class="dropdown-item ps-3 border-radius-md mb-1" href="/user/tenant">
|
||||||
|
<span>Tenant Registration</span>
|
||||||
|
</a>
|
||||||
|
<a class="dropdown-item ps-3 border-radius-md mb-1" href="/user/user">
|
||||||
|
<span>User Registration</span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
<li class="nav-item dropdown dropdown-hover mx-2">
|
||||||
|
<a role="button" class="nav-link ps-2 d-flex cursor-pointer align-items-center" id="dropdownMenuUser" data-bs-toggle="dropdown" aria-expanded="false">
|
||||||
|
<i class="material-icons opacity-6 me-2 text-md">contacts</i>
|
||||||
|
Account
|
||||||
|
<img src="{{url_for('static', filename='/assets/img/down-arrow-dark.svg')}}" alt="down-arrow" class="arrow ms-2 d-lg-block d-none">
|
||||||
|
<img src="{{url_for('static', filename='/assets/img/down-arrow-dark.svg')}}" alt="down-arrow" class="arrow ms-1 d-lg-none d-block ms-auto">
|
||||||
|
</a>
|
||||||
|
<div class="dropdown-menu dropdown-menu-animation ms-n3 dropdown-md p-3 border-radius-xl mt-0 mt-lg-3" aria-labelledby="dropdownMenuPages">
|
||||||
|
<div class="d-none d-lg-flex">
|
||||||
|
<ul class="list-group w-100">
|
||||||
|
<li class="nav-item dropdown dropdown-hover dropdown-subitem list-group-item border-0 p-0">
|
||||||
|
<a class="dropdown-item ps-3 border-radius-md mb-1" href="/login">
|
||||||
|
<span>Login</span>
|
||||||
|
</a>
|
||||||
|
<a class="dropdown-item ps-3 border-radius-md mb-1" href="/logout">
|
||||||
|
<span>Logout</span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- End Navbar -->
|
||||||
|
|
||||||
|
|
||||||
Reference in New Issue
Block a user