2
0
mirror of https://github.com/frappe/frappe.git synced 2024-06-12 21:22:27 +00:00
frappe/frappe/realtime.py
Ankush Menat 2a56869c99 refactor: better permission checking on sockets
- no more "can subscribe X"
- add a new util function on socket `has_permission`

Basic usage:

```js
socket.has_permission(doctype, [name]).then(() => { // do something });
```
2024-04-06 21:14:48 +05:30

155 lines
4.0 KiB
Python

# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and contributors
# License: MIT. See LICENSE
from contextlib import suppress
import redis
import frappe
from frappe.utils.data import cstr
def publish_progress(percent, title=None, doctype=None, docname=None, description=None, task_id=None):
publish_realtime(
"progress",
{"percent": percent, "title": title, "description": description},
user=None if doctype and docname else frappe.session.user,
doctype=doctype,
docname=docname,
task_id=task_id,
)
def publish_realtime(
event: str | None = None,
message: dict | None = None,
room: str | None = None,
user: str | None = None,
doctype: str | None = None,
docname: str | None = None,
task_id: str | None = None,
after_commit: bool = False,
):
"""Publish real-time updates
:param event: Event name, like `task_progress` etc. that will be handled by the client (default is `task_progress` if within task or `global`)
:param message: JSON message object. For async must contain `task_id`
:param room: Room in which to publish update (default entire site)
:param user: Transmit to user
:param doctype: Transmit to doctype, docname
:param docname: Transmit to doctype, docname
:param after_commit: (default False) will emit after current transaction is committed"""
if message is None:
message = {}
if not task_id and hasattr(frappe.local, "task_id"):
task_id = frappe.local.task_id
if event is None:
event = "task_progress" if task_id else "global"
elif event == "msgprint" and not user:
user = frappe.session.user
elif event == "list_update":
doctype = doctype or message.get("doctype")
room = get_doctype_room(doctype)
elif event == "docinfo_update":
room = get_doc_room(doctype, docname)
if not room:
if task_id:
after_commit = False
if "task_id" not in message:
message["task_id"] = task_id
room = get_task_progress_room(task_id)
elif user:
# transmit to specific user: System, Website or Guest
room = get_user_room(user)
elif doctype and docname:
room = get_doc_room(doctype, docname)
else:
# This will be broadcasted to all Desk users
room = get_site_room()
if after_commit:
if not hasattr(frappe.local, "_realtime_log"):
frappe.local._realtime_log = []
frappe.db.after_commit.add(flush_realtime_log)
frappe.db.after_rollback.add(clear_realtime_log)
params = [event, message, room]
if params not in frappe.local._realtime_log:
frappe.local._realtime_log.append(params)
else:
emit_via_redis(event, message, room)
def flush_realtime_log():
for args in frappe.local._realtime_log:
frappe.realtime.emit_via_redis(*args)
clear_realtime_log()
def clear_realtime_log():
if hasattr(frappe.local, "_realtime_log"):
del frappe.local._realtime_log
def emit_via_redis(event, message, room):
"""Publish real-time updates via redis
:param event: Event name, like `task_progress` etc.
:param message: JSON message object. For async must contain `task_id`
:param room: name of the room"""
from frappe.utils.background_jobs import get_redis_connection_without_auth
with suppress(redis.exceptions.ConnectionError):
r = get_redis_connection_without_auth()
r.publish(
"events",
frappe.as_json(
{"event": event, "message": message, "room": room, "namespace": frappe.local.site}
),
)
@frappe.whitelist(allow_guest=True)
def has_permission(doctype: str, name: str) -> bool:
if not frappe.has_permission(doctype=doctype, doc=name, ptype="read"):
raise frappe.PermissionError
return True
@frappe.whitelist(allow_guest=True)
def get_user_info():
return {
"user": frappe.session.user,
"user_type": frappe.session.data.user_type,
"installed_apps": frappe.get_installed_apps(),
}
def get_doctype_room(doctype):
return f"doctype:{doctype}"
def get_doc_room(doctype, docname):
return f"doc:{doctype}/{cstr(docname)}"
def get_user_room(user):
return f"user:{user}"
def get_site_room():
return "all"
def get_task_progress_room(task_id):
return f"task_progress:{task_id}"
def get_website_room():
return "website"