from bson import ObjectId from flask import request from mongoengine import Document, ObjectIdField, ListField, \ EmbeddedDocumentField, ReferenceField, ImageField, FileField, DecimalField, QuerySet from base64 import b64encode from mongoengine.base import BaseField class Resource: model: Document = None qs: dict = {} eqs: dict = {} build_reference: bool = False base64_image: bool = True base64_image_as_full: bool = False limit: int = 10 offset: int = 0 fields: list = [] ignore_fields: list = [] mask_fields: dict = {} meta: dict = {} as_pk: str = 'id' query_document: Document = None mongo_qs = None def __init__(self, model: Document): self.model = model self.limit = 10 if 'no_image' in request.args: self.base64_image = False if 'with_sub_docs' in request.args: self.build_reference = request.args.get('with_sub_docs') if 'thumb' in request.args: self.base64_image = True self.base64_image_as_full = False if 'full_image' in request.args: self.base64_image = True self.base64_image_as_full = True if 'limit' in request.args: self.limit = int(request.args.get('limit')) if 'fields' in request.args: self.fields = request.args.get('fields') if 'offset' in request.args: self.offset = int(request.args.get('offset')) self.meta = getattr(self.model, '_meta') if 'ignore_fields' in self.meta: self.ignore_fields = self.meta.get('ignore_fields') if 'with_sub_docs' in self.meta: self.build_reference = self.meta.get('with_sub_docs') if 'mask_fields' in self.meta: self.mask_fields = self.meta.get('mask_fields') if 'document' in self.meta: self.model = self.meta.get('document') if 'query' in self.meta: self.qs = self.meta.get('query') if 'as_pk' in self.meta: self.as_pk = self.meta.get('as_pk') if 'query_document' in self.meta: self.query_document = self.meta.get('query_document') def external_query(self, qs: dict): self.eqs = qs def query(self, qs: dict) -> None: self.qs = qs def to_qs(self, pk: str = None) -> tuple: query = {} if self.qs: for key, val in self.qs.items(): if callable(val): query.update({key: val()}) else: query.update({key: val}) if self.query_document: query = {} if self.eqs: for key, val in self.eqs.items(): if callable(val): query.update({key: val()}) else: query.update({key: val}) data = self.query_document.objects.filter(**query) field_name = "" for i in self.model._fields_ordered: try: if isinstance(getattr(self.model, i).document_type, self.query_document.__class__): field_name = i + "__in" except: pass query.clear() query = {field_name: [i.id for i in data]} for key, val in self.qs.items(): if callable(val): query.update({key: val()}) else: query.update({key: val}) print(query) data = self.model.objects.filter(**query) if pk: data.get(**{self.as_pk: pk}) return data def to_json(self, pk: str = None) -> tuple: if self.mongo_qs is None: self.mongo_qs = self.to_qs(pk) data = self.mongo_qs count = data.count() if pk: json_data = self.parse_fields(self.model) else: data = data[self.offset:self.limit + self.offset] json_data = [] for item in data: item_data = self.parse_fields(item, self.model) json_data.append(item_data) return count, json_data def parse_field(self, field: BaseField, value: any) -> any: if isinstance(field, ObjectIdField): return str(value) elif isinstance(field, ReferenceField): if self.build_reference: return self.parse_fields(value, field.document_type) if value: return str(value.id) else: return value elif isinstance(field, ImageField): if self.base64_image and value: if self.base64_image_as_full: file = b64encode(value.read()).decode("utf-8") return { "size": value.size, "upload_date": value.gridout.upload_date, "format": value.format, "base64": file } else: file = b64encode(value.thumbnail.read()).decode("utf-8") return { "upload_date": value.gridout.upload_date, "format": value.format, "base64": file } else: if value: return { "size": value.size } else: return None elif isinstance(field, DecimalField): return float(value) elif isinstance(field, FileField): return "file" elif isinstance(field, ListField): if isinstance(field.field, EmbeddedDocumentField): return [self.parse_fields(item, field.field.document_type) for item in value] elif isinstance(field.field, ReferenceField) and self.build_reference: return [self.parse_fields(item, field.field.document_type) for item in value] else: return [self.parse_field(field.field, item) for item in value] elif isinstance(value, QuerySet): if value.count() > 0: return [self.parse_fields(i, value._document) for i in value] else: return value def parse_fields(self, item: Document, model: Document) -> dict: parsed_item = {} fields = list(getattr(model, "_fields_ordered")) if self.model != model: in_meta = getattr(model,'_meta') if in_meta.get('ignore_fields'): self.ignore_fields += in_meta.get('ignore_fields') for key, val in model.__dict__.items(): if isinstance(getattr(model, key), property): fields.append(key) for i in fields: if i not in self.ignore_fields: field = getattr(model, i) if i not in self.mask_fields: if not item is None: value = getattr(item, i) else: value = None else: value = getattr(item, i) if self.mask_fields.get(i) == 'all': value = '*************' elif self.mask_fields.get(i) == 'email': value = '{}****@{}***.{}'.format(value[0], value.split("@")[1][0], '.'.join(value.split("@")[1].split(".")[1:])) else: value = '*************' if not value is None: parsed_item.update({i: self.parse_field(field, value)}) else: parsed_item.update({i: None}) return parsed_item