Small changes

This commit is contained in:
vlados31 2022-09-25 11:06:41 +03:00
parent 2c51925827
commit 9c27897485
5 changed files with 28 additions and 10 deletions

View File

@ -1,7 +1,8 @@
from django.db import models from django.db import models
from django.contrib.auth.models import PermissionsMixin, AbstractBaseUser, BaseUserManager from django.contrib.auth.models import PermissionsMixin, AbstractBaseUser, BaseUserManager
from django.core.validators import * from django.core.validators import *
from arka.settings import PHONE_VERIFICATION_ENABLE, PHONE_VERIFICATION_ATTEMPTS, PHONE_VERIFICATION_APP_ID from arka.settings import PHONE_VERIFICATION_ENABLE, PHONE_VERIFICATION_ATTEMPTS,\
PHONE_VERIFICATION_APP_ID, PHONE_VERIFICATION_RESEND_TIME_SECS
from threading import Thread, Lock, Event from threading import Thread, Lock, Event
import random import random
@ -29,9 +30,6 @@ class PhoneVerificationService:
def __service_run(): def __service_run():
while True: while True:
try: try:
# TODO запилить логику, которая предотвратит злоупотребление вызовами со стороны пользователей
PhoneVerificationService.__event.wait() PhoneVerificationService.__event.wait()
phones = None phones = None
with PhoneVerificationService.__lock: with PhoneVerificationService.__lock:
@ -95,6 +93,7 @@ class PhoneVerificationService:
CHECK_PHONE_FAILED = "failed" CHECK_PHONE_FAILED = "failed"
CHECK_PHONE_INVALID_CODE = "invalid" CHECK_PHONE_INVALID_CODE = "invalid"
CHECK_PHONE_MAX_ATTEMPTS = "max-attempts" CHECK_PHONE_MAX_ATTEMPTS = "max-attempts"
CHECK_PHONE_RESEND_LIMIT = "resend"
@staticmethod @staticmethod
def check_code(phone: str, code: int): def check_code(phone: str, code: int):
@ -127,9 +126,14 @@ class PhoneVerificationService:
@staticmethod @staticmethod
def send_verify(phone: str): def send_verify(phone: str):
with PhoneVerificationService.__lock: with PhoneVerificationService.__lock:
PhoneVerificationService.__to_verify.append(phone) if phone in PhoneVerificationService.__codes:
c = PhoneVerificationService.__codes[phone]
if (datetime.now() - c["time"]).total_seconds() <= PHONE_VERIFICATION_RESEND_TIME_SECS:
return False, PhoneVerificationService.CHECK_PHONE_RESEND_LIMIT
PhoneVerificationService.__event.set() PhoneVerificationService.__to_verify.append(phone)
PhoneVerificationService.__event.set()
return True, None
class SiteAccountManager(BaseUserManager): class SiteAccountManager(BaseUserManager):

View File

@ -1,9 +1,11 @@
import traceback import traceback
from account.models import PhoneVerificationService from account.models import PhoneVerificationService
from arka.settings import PHONE_VERIFICATION_RESEND_TIME_SECS
# как создавать ошибку # как создавать ошибку
# raise Exception(API_ERROR_XXX, <related_obj>) # raise Exception(API_ERROR_XXX, <related_obj>)
API_OK_OBJ = {"status": "success"}
API_ERROR_INTERNAL_ERROR = (100, 'internal error') API_ERROR_INTERNAL_ERROR = (100, 'internal error')
@ -27,7 +29,9 @@ API_ERROR_VALIDATION_CURRENTLY_VERIFIED = (522, 'currently phone is verified')
API_ERROR_VALIDATION_FAILED = (523, 'cannot be verified') API_ERROR_VALIDATION_FAILED = (523, 'cannot be verified')
API_ERROR_VALIDATION_NOT_READY = (524, 'verification service not ready. call this method later') API_ERROR_VALIDATION_NOT_READY = (524, 'verification service not ready. call this method later')
API_ERROR_VALIDATION_NOT_FOUND = (525, 'verification service did not send code. call this method without \'code\'') API_ERROR_VALIDATION_NOT_FOUND = (525, 'verification service did not send code. call this method without \'code\'')
API_ERROR_VALIDATION_UNKNOWN = (526, 'unknown verification error') API_ERROR_VALIDATION_RESEND_LIMIT = (526, f'resend verification limit '
f'(one verify for {PHONE_VERIFICATION_RESEND_TIME_SECS} secs)')
API_ERROR_VALIDATION_UNKNOWN = (527, 'unknown verification error')
API_ERROR_VALIDATION = { API_ERROR_VALIDATION = {
PhoneVerificationService.CHECK_PHONE_INVALID_CODE: API_ERROR_VALIDATION_INVALID_CODE, PhoneVerificationService.CHECK_PHONE_INVALID_CODE: API_ERROR_VALIDATION_INVALID_CODE,
@ -35,6 +39,7 @@ API_ERROR_VALIDATION = {
PhoneVerificationService.CHECK_PHONE_FAILED: API_ERROR_VALIDATION_FAILED, PhoneVerificationService.CHECK_PHONE_FAILED: API_ERROR_VALIDATION_FAILED,
PhoneVerificationService.CHECK_PHONE_NOT_READY: API_ERROR_VALIDATION_NOT_READY, PhoneVerificationService.CHECK_PHONE_NOT_READY: API_ERROR_VALIDATION_NOT_READY,
PhoneVerificationService.CHECK_PHONE_NOT_FOUND: API_ERROR_VALIDATION_NOT_FOUND, PhoneVerificationService.CHECK_PHONE_NOT_FOUND: API_ERROR_VALIDATION_NOT_FOUND,
PhoneVerificationService.CHECK_PHONE_RESEND_LIMIT: API_ERROR_VALIDATION_RESEND_LIMIT,
} }

View File

@ -71,7 +71,14 @@ def account_verify_phone(params):
code = api_get_param_int(params, "code", False, None) code = api_get_param_int(params, "code", False, None)
if code is None: if code is None:
PhoneVerificationService.send_verify(user.phone) res, err_code = PhoneVerificationService.send_verify(user.phone)
if not res:
if err_code in API_ERROR_VALIDATION:
raise Exception(API_ERROR_VALIDATION[err_code])
else:
raise Exception(API_ERROR_VALIDATION_UNKNOWN)
return api_make_response({"action": "phone_call"}) return api_make_response({"action": "phone_call"})
else: else:
res, err_code = PhoneVerificationService.check_code(user.phone, code) res, err_code = PhoneVerificationService.check_code(user.phone, code)
@ -79,7 +86,7 @@ def account_verify_phone(params):
if res: if res:
user.is_phone_verified = True user.is_phone_verified = True
user.save() user.save()
return api_make_response({"status": "success"}) return api_make_response({})
else: else:
if err_code in API_ERROR_VALIDATION: if err_code in API_ERROR_VALIDATION:
raise Exception(API_ERROR_VALIDATION[err_code]) raise Exception(API_ERROR_VALIDATION[err_code])

View File

@ -7,7 +7,7 @@ def __make_invalid_argument_type_error(name, value, except_type):
def api_make_response(response): def api_make_response(response):
return {"response": response} return {"response": API_OK_OBJ | response}
def api_get_param_int(params: dict, name: str, required=True, default=0): def api_get_param_int(params: dict, name: str, required=True, default=0):

View File

@ -144,6 +144,8 @@ STATICFILES_DIRS = [os.path.join(BASE_DIR, "static")]
DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField' DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
# LOGIN_REDIRECT_URL = '/account/' # LOGIN_REDIRECT_URL = '/account/'
PHONE_VERIFICATION_ENABLE = False PHONE_VERIFICATION_ENABLE = False
PHONE_VERIFICATION_ATTEMPTS = 5 PHONE_VERIFICATION_ATTEMPTS = 5
PHONE_VERIFICATION_RESEND_TIME_SECS = 180
PHONE_VERIFICATION_APP_ID = "ACC90F4A-FE4A-5137-45C9-5D9E84DE9440" PHONE_VERIFICATION_APP_ID = "ACC90F4A-FE4A-5137-45C9-5D9E84DE9440"