This commit is contained in:
parent
f54d752657
commit
e02bf1aa62
@ -2,10 +2,10 @@
|
||||
apiVersion: v1
|
||||
kind: ConfigMap
|
||||
metadata:
|
||||
name: input-service
|
||||
name: backend
|
||||
labels:
|
||||
app: input-service
|
||||
namespace: birbnetes
|
||||
app: backend
|
||||
namespace: videon
|
||||
data:
|
||||
SENTRY_DSN: https://58bd309272c642d884ff2332c336b977@sentry.kmlabz.com/21
|
||||
RELEASE_ID: kmlabz-k8s
|
||||
|
@ -1,7 +1,6 @@
|
||||
flask
|
||||
sentry-sdk[flask]
|
||||
flask-restful
|
||||
flask-security
|
||||
kubernetes
|
||||
gunicorn
|
||||
psycopg2-binary
|
||||
@ -10,4 +9,6 @@ flask_sqlalchemy
|
||||
marshmallow
|
||||
marshmallow-sqlalchemy
|
||||
flask-marshmallow
|
||||
py-healthcheck
|
||||
py-healthcheck
|
||||
flask-jwt-extended
|
||||
flask-bcrypt
|
17
src/app.py
17
src/app.py
@ -9,8 +9,11 @@ from healthcheck import HealthCheck
|
||||
|
||||
from config import *
|
||||
from db import db
|
||||
from jwtman import jwtman
|
||||
from fbcrypt import bcrypt
|
||||
from marshm import ma
|
||||
from healthchecks import health_database_status
|
||||
from resources import SignupApi, LoginApi
|
||||
|
||||
"""
|
||||
Main Flask RESTful API
|
||||
@ -35,15 +38,19 @@ if SENTRY_DSN:
|
||||
app = Flask(__name__)
|
||||
app.config['SQLALCHEMY_DATABASE_URI'] =\
|
||||
f"postgresql://{POSTGRES_USERNAME}:{POSTGRES_PASSWORD}@{POSTGRES_HOSTNAME}:5432/{POSTGRES_DB}"
|
||||
app.config['JWT_SECRET_KEY'] = JWT_SECRET_KEY
|
||||
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
|
||||
|
||||
api = Api(app)
|
||||
health = HealthCheck(app, "/healthz")
|
||||
db.init_app(app)
|
||||
ma.init_app(app)
|
||||
bcrypt.init_app(app)
|
||||
jwtman.init_app(app)
|
||||
|
||||
with app.app_context():
|
||||
db.create_all()
|
||||
""
|
||||
|
||||
formatter = logging.Formatter(
|
||||
fmt="%(asctime)s - %(levelname)s - %(module)s - %(message)s"
|
||||
)
|
||||
@ -56,9 +63,17 @@ logger.setLevel(logging.DEBUG)
|
||||
logger.addHandler(handler)
|
||||
|
||||
# api.add_resource(SampleResource, "/sample")
|
||||
api.add_resource(SignupApi, '/api/auth/signup')
|
||||
api.add_resource(LoginApi, '/api/auth/login')
|
||||
|
||||
health.add_check(health_database_status)
|
||||
|
||||
|
||||
@app.errorhandler(404)
|
||||
def page_not_found(e):
|
||||
return {'status': 'error', 'message': 'page not found'}, 404
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
app.run(
|
||||
debug=bool(DEBUG),
|
||||
|
@ -1,6 +1,6 @@
|
||||
#!/usr/bin/env python3
|
||||
import os
|
||||
|
||||
from uuid import uuid4
|
||||
|
||||
"""
|
||||
Configuration
|
||||
@ -25,3 +25,5 @@ POSTGRES_HOSTNAME = os.getenv("VIDEON_POSTGRES_HOSTNAME", "localhost")
|
||||
POSTGRES_USERNAME = os.getenv("VIDEON_POSTGRES_USERNAME", "videon")
|
||||
POSTGRES_PASSWORD = os.getenv("VIDEON_POSTGRES_PASSWORD", "videon")
|
||||
POSTGRES_DB = os.getenv("VIDEON_POSTGRES_DB", "videon")
|
||||
|
||||
JWT_SECRET_KEY = os.getenv("VIDEON_POSTGRES_DB", str(uuid4()))
|
||||
|
13
src/fbcrypt.py
Normal file
13
src/fbcrypt.py
Normal file
@ -0,0 +1,13 @@
|
||||
#!/usr/bin/env python3
|
||||
from flask_bcrypt import Bcrypt
|
||||
|
||||
"""
|
||||
Bcrypt object
|
||||
"""
|
||||
|
||||
__author__ = '@tormakris'
|
||||
__copyright__ = "Copyright 2020, videON Team"
|
||||
__module_name__ = "fbrypt"
|
||||
__version__text__ = "1"
|
||||
|
||||
bcrypt = Bcrypt()
|
14
src/jwtman.py
Normal file
14
src/jwtman.py
Normal file
@ -0,0 +1,14 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
from flask_jwt_extended import JWTManager
|
||||
|
||||
"""
|
||||
JWTManager
|
||||
"""
|
||||
|
||||
__author__ = '@tormakris'
|
||||
__copyright__ = "Copyright 2020, videON Team"
|
||||
__module_name__ = "jwtman"
|
||||
__version__text__ = "1"
|
||||
|
||||
jwtman = JWTManager()
|
31
src/models.py
Normal file
31
src/models.py
Normal file
@ -0,0 +1,31 @@
|
||||
#!/usr/bin/env python3
|
||||
from sqlalchemy.sql import func
|
||||
from flask_bcrypt import generate_password_hash, check_password_hash
|
||||
|
||||
from db import db
|
||||
|
||||
"""
|
||||
Database models
|
||||
"""
|
||||
|
||||
__author__ = '@tormakris'
|
||||
__copyright__ = "Copyright 2020, videON Team"
|
||||
__module_name__ = "models"
|
||||
__version__text__ = "1"
|
||||
|
||||
|
||||
class User(db.Model):
|
||||
id = db.Column(db.Integer, primary_key=True, autoincrement=True)
|
||||
|
||||
name = db.Column(db.String, nullable=False, unique=True)
|
||||
password = db.Column(db.String, nullable=False)
|
||||
|
||||
last_logon = db.Column(db.TIMESTAMP, nullable=False, server_default=func.now())
|
||||
|
||||
timestamp = db.Column(db.TIMESTAMP, nullable=False, server_default=func.now())
|
||||
|
||||
def hash_password(self):
|
||||
self.password = generate_password_hash(self.password).decode('utf8')
|
||||
|
||||
def check_password(self, password):
|
||||
return check_password_hash(self.password, password)
|
86
src/resources.py
Normal file
86
src/resources.py
Normal file
@ -0,0 +1,86 @@
|
||||
#!/usr/bin/env python3
|
||||
import datetime
|
||||
|
||||
from flask_jwt_extended import create_access_token, jwt_required
|
||||
from flask_restful import Resource
|
||||
from flask import request, current_app
|
||||
|
||||
from db import db
|
||||
from models import User
|
||||
from schemas import UserSchema, UserMetadataSchema
|
||||
|
||||
|
||||
"""
|
||||
Flask Restful endpoints
|
||||
"""
|
||||
|
||||
__author__ = '@tormakris'
|
||||
__copyright__ = "Copyright 2020, videON Team"
|
||||
__module_name__ = "resources"
|
||||
__version__text__ = "1"
|
||||
|
||||
|
||||
class SignupApi(Resource):
|
||||
"""
|
||||
See: https://swagger.kmlabz.com/?urls.primaryName=videON%20Backend#/backend/createuser
|
||||
"""
|
||||
|
||||
userschema = UserSchema(many=False)
|
||||
usermetadataschema = UserMetadataSchema(many=False)
|
||||
|
||||
def post(self):
|
||||
body = request.get_json()
|
||||
|
||||
try:
|
||||
userobj = self.userschema.load(body)
|
||||
except Exception as e:
|
||||
current_app.logger.exception(e)
|
||||
return {'status': 'error', 'message': 'Input JSON schema invalid'}, 417
|
||||
|
||||
user = User(name=userobj['name'], password=userobj['password'])
|
||||
try:
|
||||
user.hash_password()
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
except Exception as e:
|
||||
db.session.rollback()
|
||||
current_app.logger.exception(e)
|
||||
return {'status': 'error', 'message': 'db transaction error'}, 503
|
||||
|
||||
return self.usermetadataschema.dump(user), 200
|
||||
|
||||
|
||||
class LoginApi(Resource):
|
||||
"""
|
||||
See: https://swagger.kmlabz.com/?urls.primaryName=videON%20Backend#/backend/logon
|
||||
"""
|
||||
|
||||
userschema = UserSchema(many=False)
|
||||
usermetadataschema = UserMetadataSchema(many=False)
|
||||
|
||||
def post(self):
|
||||
body = request.get_json()
|
||||
|
||||
try:
|
||||
userobj = self.userschema.load(body)
|
||||
except Exception as e:
|
||||
current_app.logger.exception(e)
|
||||
return {'status': 'error', 'message': 'Input JSON schema invalid'}, 417
|
||||
|
||||
user = User.query.filter_by(name=userobj['name']).first()
|
||||
authorized = user.check_password(userobj['password'])
|
||||
if not authorized:
|
||||
return {'status': 'error', 'message': 'username or password invalid'}, 401
|
||||
|
||||
try:
|
||||
user.last_logon = datetime.datetime.now()
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
except Exception as e:
|
||||
db.session.rollback()
|
||||
current_app.logger.exception(e)
|
||||
return {'status': 'error', 'message': 'db transaction error'}, 503
|
||||
|
||||
expires = datetime.timedelta(days=7)
|
||||
access_token = create_access_token(identity=str(user.name), expires_delta=expires)
|
||||
return {'token': access_token}, 200
|
38
src/schemas.py
Normal file
38
src/schemas.py
Normal file
@ -0,0 +1,38 @@
|
||||
#!/usr/bin/env python3
|
||||
from flask_marshmallow.sqla import auto_field
|
||||
|
||||
from models import User
|
||||
from marshm import ma
|
||||
from marshmallow import fields
|
||||
|
||||
|
||||
"""
|
||||
Marshmallow schemas
|
||||
"""
|
||||
|
||||
|
||||
__author__ = "@tormakris"
|
||||
__copyright__ = "Copyright 2020, videON Team"
|
||||
__module_name__ = "schemas"
|
||||
__version__text__ = "1"
|
||||
|
||||
|
||||
class UserSchema(ma.Schema):
|
||||
"""
|
||||
Parameters:
|
||||
- name (string)
|
||||
- passowrd (string)
|
||||
"""
|
||||
|
||||
name = fields.String(required=True)
|
||||
password = fields.String(required=True)
|
||||
|
||||
|
||||
class UserMetadataSchema(ma.SQLAlchemyAutoSchema):
|
||||
"""
|
||||
Marshmallow schema generated
|
||||
"""
|
||||
class Meta:
|
||||
model = User
|
||||
exclude = ('timestamp', 'password',)
|
||||
creation_date = auto_field("timestamp", dump_only=False)
|
Loading…
Reference in New Issue
Block a user