2024-12-17 00:55:08 +00:00
|
|
|
from flask import jsonify
|
|
|
|
|
2024-12-14 17:54:01 +00:00
|
|
|
from database import db
|
2024-12-15 02:46:37 +00:00
|
|
|
from models import Organization, User, File
|
2024-12-14 17:54:01 +00:00
|
|
|
from sqlalchemy.orm.attributes import flag_modified
|
2024-12-17 01:18:45 +00:00
|
|
|
|
2024-12-16 18:38:24 +00:00
|
|
|
from utils import Perm, PermOperation
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RoleService:
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def create_role(org: Organization, role: str, perms: list[Perm]) -> dict | tuple:
|
2024-12-14 17:54:01 +00:00
|
|
|
if role in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} already exists in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
roles = org.roles.copy()
|
|
|
|
roles[role] = {
|
|
|
|
"permissions": Perm.get_int(perms),
|
|
|
|
"users": [],
|
|
|
|
"status": "active"
|
|
|
|
}
|
|
|
|
org.roles = roles
|
|
|
|
flag_modified(org, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(org)
|
|
|
|
return org.roles[role]
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def delete_role(org: Organization, role: str) -> dict:
|
|
|
|
roles = org.roles.copy()
|
|
|
|
del roles[role]
|
|
|
|
org.roles = roles
|
|
|
|
flag_modified(org, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(org)
|
|
|
|
return roles
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def change_role_status(org: Organization, role: str, status: str) -> Organization | tuple:
|
2024-12-14 17:54:01 +00:00
|
|
|
if role not in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} does not exist in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
2024-12-16 18:38:24 +00:00
|
|
|
if org.roles[role]["status"] == status:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} is already {status} in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
roles = org.roles.copy()
|
2024-12-16 18:38:24 +00:00
|
|
|
roles[role]["status"] = status
|
2024-12-14 17:54:01 +00:00
|
|
|
org.roles = roles
|
|
|
|
flag_modified(org, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(org)
|
|
|
|
return org
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-16 18:38:24 +00:00
|
|
|
def get_role(org: Organization, role: str) -> dict | None:
|
2024-12-14 17:54:01 +00:00
|
|
|
if role not in org.roles:
|
2024-12-16 18:38:24 +00:00
|
|
|
return None
|
2024-12-14 17:54:01 +00:00
|
|
|
return org.roles[role]
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def check_role_permission(org: Organization, role: str, perm: Perm, doc_handle=None) -> bool | tuple:
|
2024-12-15 02:46:37 +00:00
|
|
|
from services import FileService
|
|
|
|
|
|
|
|
if doc_handle:
|
|
|
|
file = FileService.get_file_by_document_handle(doc_handle)
|
|
|
|
if not file:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": "File not found"}), 404
|
2024-12-15 02:46:37 +00:00
|
|
|
|
|
|
|
if not Perm.check_perm(file.acl[role], perm.value):
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
if role not in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} does not exist in organization {org.name}"}), 400
|
2024-12-15 02:46:37 +00:00
|
|
|
|
2024-12-14 17:54:01 +00:00
|
|
|
role_perms = org.roles[role]["permissions"]
|
|
|
|
return Perm.check_perm(role_perms, perm.value)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def check_user_permission(user: User, org: Organization, perm: Perm) -> bool:
|
|
|
|
for role in user.roles[org.id]:
|
|
|
|
if RoleService.check_role_permission(org, role, perm):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-16 18:38:24 +00:00
|
|
|
def get_roles(org: Organization) -> dict:
|
2024-12-14 17:54:01 +00:00
|
|
|
return org.roles
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def get_users_in_role(org: Organization, role: str) -> list | tuple:
|
2024-12-16 18:38:24 +00:00
|
|
|
if role not in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} does not exist in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
return org.roles[role]["users"]
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-16 18:38:24 +00:00
|
|
|
def get_roles_for_user(user: User, org: Organization) -> list:
|
2024-12-14 17:54:01 +00:00
|
|
|
return [role for role in org.roles if user.id in org.roles[role]["users"]]
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-16 18:38:24 +00:00
|
|
|
def get_perms_for_role(org: Organization, role: str, return_str=False) -> list[Perm | str]:
|
2024-12-17 15:32:39 +00:00
|
|
|
from services import FileService
|
2024-12-17 01:18:45 +00:00
|
|
|
perms_list = Perm.get_perms(org.roles[role]["permissions"], return_str)
|
|
|
|
for f in FileService.list_files_in_org(org):
|
|
|
|
perms_list.append({f.document_handle: Perm.get_perms(f.acl[role], return_str)})
|
|
|
|
return perms_list
|
|
|
|
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2024-12-16 18:38:24 +00:00
|
|
|
def get_roles_for_perm(org: Organization, perm: Perm) -> list:
|
2024-12-14 17:54:01 +00:00
|
|
|
roles = []
|
|
|
|
for role in org.roles:
|
|
|
|
if RoleService.check_role_permission(org, role, perm):
|
|
|
|
roles.append(role)
|
|
|
|
return roles
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def add_user_to_role(role: str, org: Organization, user: User) -> User | tuple:
|
2024-12-14 17:54:01 +00:00
|
|
|
if role not in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} does not exist in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
if user.id in org.roles[role]["users"]:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"User {user.username} already has role {role} in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
if org.roles[role]["status"] != "active":
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} is not active in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
roles = user.roles.copy()
|
|
|
|
roles[org.id] = role
|
|
|
|
user.roles = roles
|
|
|
|
flag_modified(user, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(user)
|
|
|
|
|
|
|
|
roles = org.roles.copy()
|
|
|
|
roles[role]["users"].append(user.id)
|
|
|
|
org.roles = roles
|
|
|
|
flag_modified(org, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(org)
|
|
|
|
return user
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def remove_user_from_role(role: str, org: Organization, user: User) -> User | tuple:
|
2024-12-14 17:54:01 +00:00
|
|
|
if role not in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} does not exist in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
if user.id not in org.roles[role]["users"]:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"User {user.username} does not have role {role} in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
if org.roles[role]["status"] != "active":
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} is not active in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
roles = user.roles.copy()
|
|
|
|
roles.pop(org.id)
|
|
|
|
user.roles = roles
|
|
|
|
flag_modified(user, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(user)
|
|
|
|
|
|
|
|
roles = org.roles.copy()
|
|
|
|
roles[role]["users"].remove(user.id)
|
|
|
|
org.roles = roles
|
|
|
|
flag_modified(org, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(org)
|
|
|
|
return user
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def change_perm_on_role(org: Organization, role: str, perm: Perm, operation: PermOperation) -> dict | tuple:
|
2024-12-16 18:38:24 +00:00
|
|
|
if Perm.get_int([perm]) <= 0b111:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Permission {perm} is not allowed for organization's roles"}), 400
|
2024-12-15 02:46:37 +00:00
|
|
|
|
2024-12-14 17:54:01 +00:00
|
|
|
if role not in org.roles:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} does not exist in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
if org.roles[role]["status"] != "active":
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} is not active in organization {org.name}"}), 400
|
2024-12-14 17:54:01 +00:00
|
|
|
|
|
|
|
roles = org.roles.copy()
|
2024-12-16 18:38:24 +00:00
|
|
|
roles[role]["permissions"] = PermOperation.calc(roles[role]["permissions"], perm, operation)
|
2024-12-14 17:54:01 +00:00
|
|
|
org.roles = roles
|
|
|
|
flag_modified(org, "roles")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(org)
|
|
|
|
return org.roles[role]
|
|
|
|
|
|
|
|
@staticmethod
|
2024-12-17 00:55:08 +00:00
|
|
|
def change_perm_on_role_in_file(file: File, role: str, perm: Perm, operation: PermOperation) -> dict | tuple:
|
2024-12-16 18:38:24 +00:00
|
|
|
if Perm.get_int([perm]) > 0b111:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Permission {perm} is not allowed for files' roles"}), 400
|
2024-12-15 02:46:37 +00:00
|
|
|
|
|
|
|
if role not in file.acl:
|
|
|
|
file.acl[role] = 0
|
|
|
|
|
|
|
|
if file.acl[role] & perm.value != 0:
|
2024-12-17 00:55:08 +00:00
|
|
|
return jsonify({"error": f"Role {role} already has permission {perm} in file {file.document_handle}"}), 400
|
2024-12-15 02:46:37 +00:00
|
|
|
|
2024-12-16 18:38:24 +00:00
|
|
|
file.acl[role] = PermOperation.calc(file.acl[role], perm, operation)
|
2024-12-15 02:46:37 +00:00
|
|
|
flag_modified(file, "acl")
|
|
|
|
db.commit()
|
|
|
|
db.refresh(file)
|
|
|
|
return file.acl
|