NXPR-3 WIP

This commit is contained in:
Wojciech Janota 2022-11-17 22:45:43 +01:00
parent 637b4d88e2
commit 2ccc2e82a3
No known key found for this signature in database
GPG Key ID: E83FBD2850CC1F14
15 changed files with 162 additions and 14 deletions

View File

@ -5,3 +5,4 @@ database_file: "database.db"
server_host: "localhost" server_host: "localhost"
server_password: "sekret_password" server_password: "sekret_password"
server_access_username: "user" server_access_username: "user"
jwt_secret: "sekrit"

0
database.db Normal file
View File

View File

@ -5,7 +5,7 @@ from utils.config.config import ServerConfig
config = ServerConfig() config = ServerConfig()
server = Server(host=config.server_host, port=config.server_port, name=config.server_name, access_password=config.server_password, server = Server(host=config.server_host, port=config.server_port, name=config.server_name, access_password=config.server_password,
access_username=config.server_access_username, version="v0.0.1alpha", database_file_path=config.database_file, logging_level=config.server_loglevel) access_username=config.server_access_username, jwt_secret=config.jwt_secret, version="v0.0.1alpha", database_file_path=config.database_file, logging_level=config.server_loglevel)
server.run() server.run()

View File

@ -2,8 +2,11 @@ from http.client import NON_AUTHORITATIVE_INFORMATION
from flask import Flask, request, jsonify from flask import Flask, request, jsonify
from utils.database.database import Database from utils.database.database import Database
from utils.exceptions import DatabaseException from utils.exceptions import DatabaseException
from utils.models.models import Client from utils.models.models import Client, VMImage, User
from utils.middleware.auth import require_auth
import json import json
import bcrypt
import jwt
class FlaskAppWrapper(object): class FlaskAppWrapper(object):
@ -26,7 +29,7 @@ class FlaskAppWrapper(object):
class Server(): class Server():
def __init__(self, host: str, port: int, name: str, access_password: str, access_username: str, version: str, database_file_path: str, logging_level: str, ): def __init__(self, host: str, port: int, name: str, access_password: str, access_username: str, jwt_secret: str, version: str, database_file_path: str, logging_level: str, ):
self.host = host self.host = host
self.port = port self.port = port
self.name = name self.name = name
@ -36,12 +39,66 @@ class Server():
self.database = Database( self.database = Database(
database_file=database_file_path, logging_level=logging_level) database_file=database_file_path, logging_level=logging_level)
self.flask_app = Flask(name) self.flask_app = Flask(name)
self.flask_app.config['SECRET_KEY'] = jwt_secret
self.flask_app.config['DATABASE_FILE_PATH'] = database_file_path
self.flask_app.config['LOGGING_LEVEL'] = logging_level
self.app = FlaskAppWrapper(self.flask_app) self.app = FlaskAppWrapper(self.flask_app)
def basic_server_data(self): def basic_server_data(self):
return {"server_name": self.name, "server_version": self.version, "host": self.host} return {"server_name": self.name, "server_version": self.version, "host": self.host}
def register_new_client_to_database(self): def login(self):
try:
request_data = request.json
if not request_data:
return {
"message": "Please provide login info",
"data": None,
"error": "Bad request"
}, 400
current_user = self.database.get_user_by_name(
request_data["username"])
if current_user is None:
return {
"message": "Please provide correct login info",
"data": None,
"error": "Bad request"
}, 400
correct_password = False
temporary_salt = bcrypt.gensalt()
hashed_request_password = bcrypt.hashpw(
password=request_data["password"].encode("utf-8"), salt=temporary_salt)
if current_user.password_hash != hashed_request_password:
return {
"message": "Invalid login data",
"data": None,
"error": "Auth error"
}, 401
try:
current_user["token"] = jwt.encode({
"username": current_user["username"]
},
self.flask_app.config["SECRET_KEY"],
algorithm="HS256"
)
current_user.pop("password")
return current_user, 202
except Exception as ex:
return {
"message": "Error loging in",
"data": None,
"error": f"Internal server error: {str(ex)}",
}, 500
except Exception as ex:
return {
"message": "Error loging in",
"data": None,
"error": f"Internal server error: {str(ex)}",
}, 500
@require_auth
def register_new_client_to_database(self, request_user):
request_content_type = request.headers.get('Content-Type') request_content_type = request.headers.get('Content-Type')
json_string = "" json_string = ""
if request_content_type == 'application/json': if request_content_type == 'application/json':
@ -60,6 +117,15 @@ class Server():
return response return response
def run(self): def run(self):
# add admin user to dataabse (or update existing one)
salt = bcrypt.gensalt()
temp_password_hash = bcrypt.hashpw(
self.access_password.encode("utf-8"), salt)
admin_user = self.database.get_user_by_name(self.access_username)
if admin_user == None:
admin_user = User(username=self.access_username,
password_hash=temp_password_hash)
self.database.add_user(admin_user)
self.app.add_endpoint(endpoint="/", endpoint_name="server_data", self.app.add_endpoint(endpoint="/", endpoint_name="server_data",
handler=self.basic_server_data, methods=["GET"]) handler=self.basic_server_data, methods=["GET"])
self.app.add_endpoint(endpoint="/clients", endpoint_name="register_client", self.app.add_endpoint(endpoint="/clients", endpoint_name="register_client",

View File

@ -18,6 +18,7 @@ class ServerConfig:
server_port_override = os.environ.get("VALHALLA_SERVER_PORT") server_port_override = os.environ.get("VALHALLA_SERVER_PORT")
server_host_override = os.environ.get("VALHALLA_SERVER_HOST") server_host_override = os.environ.get("VALHALLA_SERVER_HOST")
server_password_override = os.environ.get("VALHALLA_SERVER_PASSWORD") server_password_override = os.environ.get("VALHALLA_SERVER_PASSWORD")
jwt_secret_override = os.environ.get("VALHALLA_JWT_SECRET")
server_access_username_override = os.environ.get( server_access_username_override = os.environ.get(
"VALHALLA_SERVER_ACCESS_USERNAME") "VALHALLA_SERVER_ACCESS_USERNAME")
database_file_override = os.environ.get("VALHALLA_DATABASE_FILE") database_file_override = os.environ.get("VALHALLA_DATABASE_FILE")
@ -38,6 +39,9 @@ class ServerConfig:
if server_password_override: if server_password_override:
config["server_password"] = server_password_override config["server_password"] = server_password_override
if jwt_secret_override:
config["jwt_secret"] = jwt_secret_override
if server_access_username_override: if server_access_username_override:
config["server_access_username"] = server_access_username_override config["server_access_username"] = server_access_username_override
@ -49,5 +53,6 @@ class ServerConfig:
self.database_file = config["database_file"] self.database_file = config["database_file"]
self.server_host = config["server_host"] self.server_host = config["server_host"]
self.server_password = config["server_password"] self.server_password = config["server_password"]
self.jwt_secret = config["jwt_secret"]
self.server_access_username = config["server_access_username"] self.server_access_username = config["server_access_username"]
self.server_loglevel = config["server_loglevel"] self.server_loglevel = config["server_loglevel"]

View File

@ -1,6 +1,7 @@
from sqlalchemy import create_engine from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker from sqlalchemy.orm import sessionmaker
from utils.models.models import Client, VMImage from sqlalchemy.ext.declarative import declarative_base
from utils.models.models import Client, VMImage, User
from utils.exceptions.DatabaseException import DatabaseException from utils.exceptions.DatabaseException import DatabaseException
import logging import logging
@ -163,3 +164,29 @@ class Database:
self.logger.error(f"Couldn't modify object in database: {ex}") self.logger.error(f"Couldn't modify object in database: {ex}")
raise DatabaseException( raise DatabaseException(
f"Couldn't modify object in database: {ex}") f"Couldn't modify object in database: {ex}")
def add_user(self, new_user: User):
try:
with self.session.begin():
self.session.add(new_user)
self.session.flush()
self.session.merge()
except Exception as ex:
self.logger.error(f"Couldn't add user to the database: {ex}")
raise DatabaseException(f"Couldn't add user to the database: {ex}")
def get_user_by_id(self, user_id: int) -> User:
try:
with self.session.begin():
return self.session.query(User).filter(User.user_id == user_id).first()
except Exception as ex:
self.logger.error(f"Error getting data from database: {ex}")
raise DatabaseException(f"Error getting data from database: {ex}")
def get_user_by_name(self, username: str) -> User:
try:
with self.session.begin():
return self.session.query(User).filter(User.username == username).first()
except Exception as ex:
self.logger.error(f"Error getting data from database: {ex}")
raise DatabaseException(f"Error getting data from database: {ex}")

View File

@ -0,0 +1 @@
from . import auth

Binary file not shown.

Binary file not shown.

45
utils/middleware/auth.py Normal file
View File

@ -0,0 +1,45 @@
from functools import wraps
import jwt
from flask import request, abort
from flask import current_app
from utils.models.models import User
from utils.database.database import Database
# Inspired by: https://blog.loginradius.com/engineering/guest-post/securing-flask-api-with-jwt/ [access: 16.11.2022, 18:33 CET]
def require_auth(f):
@wraps(f)
def decorated(*args, **kwargs):
token = None
if "Authorization" in request.headers:
token = request.headers["Authorization"].split(" ")[1]
if not token:
return {
"message": "Missing auth token",
"data": None,
"error": "Unauthorized"
}, 401
try:
database = Database(
database_file=current_app.config["DATABASE_FILE"], logging_level=current_app.config["LOGGING_LEVEL"])
user_data_from_request = jwt.decode(
token, current_app.config["SECRET_KEY"], algorithms=["HS256"])
request_user = database.get_user_by_name(
username=user_data_from_request["username"])
if request_user is None:
return {
"message": "Invalid auth token",
"data": None,
"error": "Unauthorized"
}, 403
except Exception as ex:
return {
"message": "Internal server error",
"data": None,
"error": str(ex)
}, 500
return f(request_user, *args, **kwargs)
return decorated

View File

@ -1,8 +1,13 @@
from sqlalchemy import Column, Integer, String, ForeignKey, Table from sqlalchemy import Column, Integer, String, ForeignKey, Table
from sqlalchemy.orm import relationship, backref from sqlalchemy.orm import relationship, backref
from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.ext.declarative import declarative_base
from utils.config.config import ServerConfig
Base = declarative_base() config = ServerConfig()
engine = create_engine(f"sqlite:///{config.database_file}")
Base = declarative_base(bind=engine)
Base.metadata.create_all()
client_image_table = Table( client_image_table = Table(
"client_image", "client_image",
@ -19,9 +24,8 @@ class Client(Base):
hostname = Column(String(100), nullable=False) hostname = Column(String(100), nullable=False)
client_version = Column(String(100), nullable=False) client_version = Column(String(100), nullable=False)
vm_list_on_machine = relationship( vm_list_on_machine = relationship(
"VMImages", "VMImage",
secondary=client_image_table, secondary=client_image_table,
back_populates="vm_images"
) )
def has_vm_installed(self, vm_object): def has_vm_installed(self, vm_object):
@ -37,16 +41,15 @@ class VMImage(Base):
image_name = Column(String(100), unique=True, nullable=False) image_name = Column(String(100), unique=True, nullable=False)
image_file = Column(String(500), unique=False, nullable=False) image_file = Column(String(500), unique=False, nullable=False)
image_version = Column(String(100), nullable=False) image_version = Column(String(100), nullable=False)
image_hash = Column(String(500), nullalbe=False) image_hash = Column(String(500), nullable=False)
clients = relationship( clients = relationship(
"Clients", "Client",
secondary=client_image_table, secondary=client_image_table
back_populates="clients"
) )
class User(Base): class User(Base):
__tablename__ = "users" __tablename__ = "users"
user_id = Column(Integer, primary_key=True) user_id = Column(Integer, primary_key=True, autoincrement=True)
username = Column(String) username = Column(String)
password_hash = Column(String) password_hash = Column(String)