Implement login API, closes #1 #2

Open
BlackCorbeau wants to merge 25 commits from dev into main
12 changed files with 148 additions and 0 deletions
+2
View File
@@ -0,0 +1,2 @@
KEY= # Key for JWT token
POSTGRES_URL=postgresql://
+28
View File
@@ -0,0 +1,28 @@
from flask import Blueprint, request, jsonify
from model.user import User
auth = Blueprint("auth", __name__)
@auth.route('/login', methods = ['POST'])
def login():
if request.is_json:
req = request.json
email = req.get('email')
password = req.get('password')
if not email or not password:
return "Request must have email and password", 400
if len(email.strip()) < 4 or '@' not in email or '.' not in email:
return "Email is incorrect", 400
if len(password.strip()) < 8:
return "Password is too short", 400
user = User(email, password)
return jsonify(user.toJson())
else:
return "Request is not a json", 400
+24
View File
@@ -0,0 +1,24 @@
from flask import Blueprint, request, jsonify
from model.user import User
#from module.db.repositories.robot_reposytory import update_robot get_by_id
from loguru import logger as log
robots = Blueprint("robots", __name__)
robots.route('/data', methods = ['POST'])
def data():
if request.headers.get("Authorization"):
if request.is_json:
req = request.json
id = req.get('robot_id')
tms = req.get('timestamp')
loc = req.get('location')
scanRes = req.get('scan_results')
battery = req.get('battery_level')
if update_robot(id, "received", battery, loc["zone"], loc["row"], loc["self"]):
res = {"status": "received", "message_id": "123"} #for that moment i don`t now what is message
return jsonify(res)
else:
log.error('failed to update robot data')
return("Server error")
+26
View File
@@ -0,0 +1,26 @@
from app import app
from flask_socketio import SocketIO, emit
from loguru import logger as log
#from module.db.repositories.robot_reposytory import update_robot get_by_id get_all
#from module.db.repositories.inventory_repository import get_all as getRecords
ws = socketIO(app)
@ws.on('connect')
def is_connect():
log.info('client is connected')
@ws.on('disconnect')
def is_disconnect():
log.info('client is disconnected')
@ws.on('robot_update')
def robotUpdate():
robots = get_all()
emit('response', jsonify({"type": "robot_update", "data": {robots.toJSON()}})
@ws.on('inventory_alert')
def robotUpdate():
records = getRecords()
emit('response', jsonify({"type": "inventory_alert", "data": {records.toJSON()}})
+11
View File
@@ -1,4 +1,15 @@
from sys import exit
from flask import Flask
from api.auth import auth
from utils.loadDotEnv import initializeENV
from utils.PostgressConnect import PSQLConnect, PSQLCursor
if not initializeENV():
exit(-1)
#conn = PSQLConnect()
#cur = PSQLCursor(conn)
app = Flask(__name__)
app.register_blueprint(auth, url_prefix='/api/auth')
View File
+5
View File
@@ -1 +1,6 @@
flask==3.1.2
flask-socketio
python-dotenv
psycopg2-binary
pyjwt
loguru
View File
+20
View File
@@ -0,0 +1,20 @@
from dataclasses import dataclass
import json
from utils.token import generateKey
@dataclass
class User:
id: int
name: str
role: str
token: str
def __init__(self, email: str, passwd: str):
#us = getUsModel() #возвращает словарь
Sweetbread marked this conversation as resolved
Review
  1. Отдели имя и аннотацию пробелами (email: str, ...)
  2. Зачем initialize (user.initialize(...)), когда уже есть __init__ (user(...))?
1. Отдели имя и аннотацию пробелами (`email: str`, ...) 2. Зачем `initialize` (`user.initialize(...)`), когда уже есть `__init__` (`user(...)`)?
self.id = 1#us['id']
self.name = 'Bob'#us['name']
self.role = 'Backend'#us['role']
self.token = generateKey(email, passwd)
Outdated
Review

Выглядит, как что-то недоработанное

Выглядит, как что-то недоработанное
Outdated
Review

Может потому что нет бд?)

Может потому что нет бд?)
Outdated
Review

Так а зачем PR отправил?

Так а зачем PR отправил?
def toJson(self):
Outdated
Review

Попробуй просто по порядку перечислять аргументы, а то x=x так себе выглядит

Попробуй просто по порядку перечислять аргументы, а то `x=x` так себе выглядит
return {"user": {"id": self.id, "name": self.name, "role": self.role}, "token": self.token}
+10
View File
@@ -0,0 +1,10 @@
import psycopg2
import os
def PSQLConnect():
Review

Где используется?

Где используется?
Review

Вспомогательный модуль для работы с БД

Вспомогательный модуль для работы с БД
Review

Я вижу что это. Где используется?

Я вижу что это. Где используется?
Review

./db/Initalizedb

./db/Initalizedb
Review

Такого файла нет

Такого файла нет
Review

Он пока не готов я его дописываю

Он пока не готов я его дописываю
conn = psycopg2.connect(os.getenv('POSTGRES_URL'))
return conn
def PSQLCursor(conn):
Review

Где используется?

Где используется?
Review

./db/Initalizedb

./db/Initalizedb
cur = conn.cursor()
Outdated
Review

Пробел лишний

Пробел лишний
return cur
+14
View File
@@ -0,0 +1,14 @@
import os
from dotenv import load_dotenv
from loguru import logger as log
Outdated
Review

Где используется?

Где используется?
Outdated
Review

Функция загружающая переменные окружения

Функция загружающая переменные окружения
Outdated
Review

Я вижу что это. Где используется?

Я вижу что это. Где используется?
Outdated
Review

./db/Initalizedb, щас поставлю в другое место

./db/Initalizedb, щас поставлю в другое место
DOTENV_PATH = '.env'
def initializeENV() -> bool:
if os.path.exists(DOTENV_PATH):
load_dotenv(DOTENV_PATH)
log.info('.env is loaded')
return True
Outdated
Review

Пока ладно, но надо бы либы для логирования использовать

Пока ладно, но надо бы либы для логирования использовать
else:
log.error('.env isn`t loaded')
return False
+8
View File
@@ -0,0 +1,8 @@
import jwt
import os
from time import time
def generateKey(email, passwd):
key = os.getenv('KEY')
Outdated
Review

1. encoded = jwt.encode({email: passwd}, key, algorithm="HS256")
2. Добавь iat, на всякий случай

~~1. `encoded = jwt.encode({email: passwd}, key, algorithm="HS256")`~~ 2. Добавь [`iat`](https://pyjwt.readthedocs.io/en/latest/usage.html#issued-at-claim-iat), на всякий случай
encoded = jwt.encode({email: passwd, 'iat': time()}, key, algorithm="HS256")
return encoded