|
- # Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
- # License: MIT. See LICENSE
- import json
- import os
- from typing import TYPE_CHECKING
-
- import frappe
- import frappe.model
- import frappe.utils
- from frappe import _
- from frappe.desk.reportview import validate_args
- from frappe.model.db_query import check_parent_permission
- from frappe.utils import get_safe_filters
-
- if TYPE_CHECKING:
- from frappe.model.document import Document
-
- """
- Handle RESTful requests that are mapped to the `/api/resource` route.
-
- Requests via FrappeClient are also handled here.
- """
-
-
- @frappe.whitelist()
- def get_list(
- doctype,
- fields=None,
- filters=None,
- order_by=None,
- limit_start=None,
- limit_page_length=20,
- parent=None,
- debug=False,
- as_dict=True,
- or_filters=None,
- ):
- """Returns a list of records by filters, fields, ordering and limit
-
- :param doctype: DocType of the data to be queried
- :param fields: fields to be returned. Default is `name`
- :param filters: filter list by this dict
- :param order_by: Order by this fieldname
- :param limit_start: Start at this index
- :param limit_page_length: Number of records to be returned (default 20)"""
- if frappe.is_table(doctype):
- check_parent_permission(parent, doctype)
-
- args = frappe._dict(
- doctype=doctype,
- parent_doctype=parent,
- fields=fields,
- filters=filters,
- or_filters=or_filters,
- order_by=order_by,
- limit_start=limit_start,
- limit_page_length=limit_page_length,
- debug=debug,
- as_list=not as_dict,
- )
-
- validate_args(args)
- return frappe.get_list(**args)
-
-
- @frappe.whitelist()
- def get_count(doctype, filters=None, debug=False, cache=False):
- return frappe.db.count(doctype, get_safe_filters(filters), debug, cache)
-
-
- @frappe.whitelist()
- def get(doctype, name=None, filters=None, parent=None):
- """Returns a document by name or filters
-
- :param doctype: DocType of the document to be returned
- :param name: return document of this `name`
- :param filters: If name is not set, filter by these values and return the first match"""
- if frappe.is_table(doctype):
- check_parent_permission(parent, doctype)
-
- if filters and not name:
- name = frappe.db.get_value(doctype, frappe.parse_json(filters))
- if not name:
- frappe.throw(_("No document found for given filters"))
-
- doc = frappe.get_doc(doctype, name)
- if not doc.has_permission("read"):
- raise frappe.PermissionError
-
- return frappe.get_doc(doctype, name).as_dict()
-
-
- @frappe.whitelist()
- def get_value(doctype, fieldname, filters=None, as_dict=True, debug=False, parent=None):
- """Returns a value form a document
-
- :param doctype: DocType to be queried
- :param fieldname: Field to be returned (default `name`)
- :param filters: dict or string for identifying the record"""
- if frappe.is_table(doctype):
- check_parent_permission(parent, doctype)
-
- if not frappe.has_permission(doctype, parent_doctype=parent):
- frappe.throw(_("No permission for {0}").format(_(doctype)), frappe.PermissionError)
-
- filters = get_safe_filters(filters)
- if isinstance(filters, str):
- filters = {"name": filters}
-
- try:
- fields = frappe.parse_json(fieldname)
- except (TypeError, ValueError):
- # name passed, not json
- fields = [fieldname]
-
- # check whether the used filters were really parseable and usable
- # and did not just result in an empty string or dict
- if not filters:
- filters = None
-
- if frappe.get_meta(doctype).issingle:
- value = frappe.db.get_values_from_single(fields, filters, doctype, as_dict=as_dict, debug=debug)
- else:
- value = get_list(
- doctype,
- filters=filters,
- fields=fields,
- debug=debug,
- limit_page_length=1,
- parent=parent,
- as_dict=as_dict,
- )
-
- if as_dict:
- return value[0] if value else {}
-
- if not value:
- return
-
- return value[0] if len(fields) > 1 else value[0][0]
-
-
- @frappe.whitelist()
- def get_single_value(doctype, field):
- if not frappe.has_permission(doctype):
- frappe.throw(_("No permission for {0}").format(_(doctype)), frappe.PermissionError)
- value = frappe.db.get_single_value(doctype, field)
- return value
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def set_value(doctype, name, fieldname, value=None):
- """Set a value using get_doc, group of values
-
- :param doctype: DocType of the document
- :param name: name of the document
- :param fieldname: fieldname string or JSON / dict with key value pair
- :param value: value if fieldname is JSON / dict"""
-
- if fieldname in (frappe.model.default_fields + frappe.model.child_table_fields):
- frappe.throw(_("Cannot edit standard fields"))
-
- if not value:
- values = fieldname
- if isinstance(fieldname, str):
- try:
- values = json.loads(fieldname)
- except ValueError:
- values = {fieldname: ""}
- else:
- values = {fieldname: value}
-
- # check for child table doctype
- if not frappe.get_meta(doctype).istable:
- doc = frappe.get_doc(doctype, name)
- doc.update(values)
- else:
- doc = frappe.db.get_value(doctype, name, ["parenttype", "parent"], as_dict=True)
- doc = frappe.get_doc(doc.parenttype, doc.parent)
- child = doc.getone({"doctype": doctype, "name": name})
- child.update(values)
-
- doc.save()
-
- return doc.as_dict()
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def insert(doc=None):
- """Insert a document
-
- :param doc: JSON or dict object to be inserted"""
- if isinstance(doc, str):
- doc = json.loads(doc)
-
- return insert_doc(doc).as_dict()
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def insert_many(docs=None):
- """Insert multiple documents
-
- :param docs: JSON or list of dict objects to be inserted in one request"""
- if isinstance(docs, str):
- docs = json.loads(docs)
-
- if len(docs) > 200:
- frappe.throw(_("Only 200 inserts allowed in one request"))
-
- out = set()
- for doc in docs:
- out.add(insert_doc(doc).name)
-
- return out
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def save(doc):
- """Update (save) an existing document
-
- :param doc: JSON or dict object with the properties of the document to be updated"""
- if isinstance(doc, str):
- doc = json.loads(doc)
-
- doc = frappe.get_doc(doc)
- doc.save()
-
- return doc.as_dict()
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def rename_doc(doctype, old_name, new_name, merge=False):
- """Rename document
-
- :param doctype: DocType of the document to be renamed
- :param old_name: Current `name` of the document to be renamed
- :param new_name: New `name` to be set"""
- new_name = frappe.rename_doc(doctype, old_name, new_name, merge=merge)
- return new_name
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def submit(doc):
- """Submit a document
-
- :param doc: JSON or dict object to be submitted remotely"""
- if isinstance(doc, str):
- doc = json.loads(doc)
-
- doc = frappe.get_doc(doc)
- doc.submit()
-
- return doc.as_dict()
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def cancel(doctype, name):
- """Cancel a document
-
- :param doctype: DocType of the document to be cancelled
- :param name: name of the document to be cancelled"""
- wrapper = frappe.get_doc(doctype, name)
- wrapper.cancel()
-
- return wrapper.as_dict()
-
-
- @frappe.whitelist(methods=["DELETE", "POST"])
- def delete(doctype, name):
- """Delete a remote document
-
- :param doctype: DocType of the document to be deleted
- :param name: name of the document to be deleted"""
- frappe.delete_doc(doctype, name, ignore_missing=False)
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def set_default(key, value, parent=None):
- """set a user default value"""
- frappe.db.set_default(key, value, parent or frappe.session.user)
- frappe.clear_cache(user=frappe.session.user)
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def make_width_property_setter(doc):
- """Set width Property Setter
-
- :param doc: Property Setter document with `width` property"""
- if isinstance(doc, str):
- doc = json.loads(doc)
- if doc["doctype"] == "Property Setter" and doc["property"] == "width":
- frappe.get_doc(doc).insert(ignore_permissions=True)
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def bulk_update(docs):
- """Bulk update documents
-
- :param docs: JSON list of documents to be updated remotely. Each document must have `docname` property"""
- docs = json.loads(docs)
- failed_docs = []
- for doc in docs:
- doc.pop("flags", None)
- try:
- existing_doc = frappe.get_doc(doc["doctype"], doc["docname"])
- existing_doc.update(doc)
- existing_doc.save()
- except Exception:
- failed_docs.append({"doc": doc, "exc": frappe.utils.get_traceback()})
-
- return {"failed_docs": failed_docs}
-
-
- @frappe.whitelist()
- def has_permission(doctype, docname, perm_type="read"):
- """Returns a JSON with data whether the document has the requested permission
-
- :param doctype: DocType of the document to be checked
- :param docname: `name` of the document to be checked
- :param perm_type: one of `read`, `write`, `create`, `submit`, `cancel`, `report`. Default is `read`"""
- # perm_type can be one of read, write, create, submit, cancel, report
- return {"has_permission": frappe.has_permission(doctype, perm_type.lower(), docname)}
-
-
- @frappe.whitelist()
- def get_password(doctype, name, fieldname):
- """Return a password type property. Only applicable for System Managers
-
- :param doctype: DocType of the document that holds the password
- :param name: `name` of the document that holds the password
- :param fieldname: `fieldname` of the password property
- """
- frappe.only_for("System Manager")
- return frappe.get_doc(doctype, name).get_password(fieldname)
-
-
- @frappe.whitelist()
- def get_js(items):
- """Load JS code files. Will also append translations
- and extend `frappe._messages`
-
- :param items: JSON list of paths of the js files to be loaded."""
- items = json.loads(items)
- out = []
- for src in items:
- src = src.strip("/").split("/")
-
- if ".." in src or src[0] != "assets":
- frappe.throw(_("Invalid file path: {0}").format("/".join(src)))
-
- contentpath = os.path.join(frappe.local.sites_path, *src)
- with open(contentpath) as srcfile:
- code = frappe.utils.cstr(srcfile.read())
-
- if frappe.local.lang != "en":
- messages = frappe.get_lang_dict("jsfile", contentpath)
- messages = json.dumps(messages)
- code += f"\n\n$.extend(frappe._messages, {messages})"
-
- out.append(code)
-
- return out
-
-
- @frappe.whitelist(allow_guest=True)
- def get_time_zone():
- """Returns default time zone"""
- return {"time_zone": frappe.defaults.get_defaults().get("time_zone")}
-
-
- @frappe.whitelist(methods=["POST", "PUT"])
- def attach_file(
- filename=None,
- filedata=None,
- doctype=None,
- docname=None,
- folder=None,
- decode_base64=False,
- is_private=None,
- docfield=None,
- ):
- """Attach a file to Document
-
- :param filename: filename e.g. test-file.txt
- :param filedata: base64 encode filedata which must be urlencoded
- :param doctype: Reference DocType to attach file to
- :param docname: Reference DocName to attach file to
- :param folder: Folder to add File into
- :param decode_base64: decode filedata from base64 encode, default is False
- :param is_private: Attach file as private file (1 or 0)
- :param docfield: file to attach to (optional)"""
-
- doc = frappe.get_doc(doctype, docname)
- doc.check_permission()
-
- file = frappe.get_doc(
- {
- "doctype": "File",
- "file_name": filename,
- "attached_to_doctype": doctype,
- "attached_to_name": docname,
- "attached_to_field": docfield,
- "folder": folder,
- "is_private": is_private,
- "content": filedata,
- "decode": decode_base64,
- }
- ).save()
-
- if docfield and doctype:
- doc.set(docfield, file.file_url)
- doc.save()
-
- return file
-
-
- @frappe.whitelist()
- def get_hooks(hook, app_name=None):
- return frappe.get_hooks(hook, app_name)
-
-
- @frappe.whitelist()
- def is_document_amended(doctype, docname):
- if frappe.permissions.has_permission(doctype):
- try:
- return frappe.db.exists(doctype, {"amended_from": docname})
- except frappe.db.InternalError:
- pass
-
- return False
-
-
- @frappe.whitelist()
- def validate_link(doctype: str, docname: str, fields=None):
- if not isinstance(doctype, str):
- frappe.throw(_("DocType must be a string"))
-
- if not isinstance(docname, str):
- frappe.throw(_("Document Name must be a string"))
-
- if doctype != "DocType" and not (
- frappe.has_permission(doctype, "select") or frappe.has_permission(doctype, "read")
- ):
- frappe.throw(
- _("You do not have Read or Select Permissions for {}").format(frappe.bold(doctype)),
- frappe.PermissionError,
- )
-
- values = frappe._dict()
- values.name = frappe.db.get_value(doctype, docname, cache=True)
-
- fields = frappe.parse_json(fields)
- if not values.name or not fields:
- return values
-
- try:
- values.update(get_value(doctype, fields, docname))
- except frappe.PermissionError:
- frappe.clear_last_message()
- frappe.msgprint(
- _("You need {0} permission to fetch values from {1} {2}").format(
- frappe.bold(_("Read")), frappe.bold(doctype), frappe.bold(docname)
- ),
- title=_("Cannot Fetch Values"),
- indicator="orange",
- )
-
- return values
-
-
- def insert_doc(doc) -> "Document":
- """Inserts document and returns parent document object with appended child document
- if `doc` is child document else returns the inserted document object
-
- :param doc: doc to insert (dict)"""
-
- doc = frappe._dict(doc)
- if frappe.is_table(doc.doctype):
- if not (doc.parenttype and doc.parent and doc.parentfield):
- frappe.throw(_("Parenttype, Parent and Parentfield are required to insert a child record"))
-
- # inserting a child record
- parent = frappe.get_doc(doc.parenttype, doc.parent)
- parent.append(doc.parentfield, doc)
- parent.save()
- return parent
-
- return frappe.get_doc(doc).insert()
|