Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.
 
 
 
 
 
 

207 рядки
5.9 KiB

  1. # -*- coding: utf-8 -*-
  2. # Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and contributors
  3. # For license information, please see license.txt
  4. from __future__ import unicode_literals
  5. import frappe
  6. import os
  7. import time
  8. import redis
  9. from io import FileIO
  10. from frappe.utils import get_site_path
  11. from frappe import conf
  12. END_LINE = '<!-- frappe: end-file -->'
  13. TASK_LOG_MAX_AGE = 86400 # 1 day in seconds
  14. redis_server = None
  15. @frappe.whitelist()
  16. def get_pending_tasks_for_doc(doctype, docname):
  17. return frappe.db.sql_list("select name from `tabAsync Task` where status in ('Queued', 'Running') and reference_doctype=%s and reference_name=%s", (doctype, docname))
  18. def set_task_status(task_id, status, response=None):
  19. if not response:
  20. response = {}
  21. response.update({
  22. "status": status,
  23. "task_id": task_id
  24. })
  25. emit_via_redis("task_status_change", response, room="task:" + task_id)
  26. def remove_old_task_logs():
  27. logs_path = get_site_path('task-logs')
  28. def full_path(_file):
  29. return os.path.join(logs_path, _file)
  30. files_to_remove = [full_path(_file) for _file in os.listdir(logs_path)]
  31. files_to_remove = [_file for _file in files_to_remove if is_file_old(_file) and os.path.isfile(_file)]
  32. for _file in files_to_remove:
  33. os.remove(_file)
  34. def is_file_old(file_path):
  35. return ((time.time() - os.stat(file_path).st_mtime) > TASK_LOG_MAX_AGE)
  36. def publish_progress(percent, title=None, doctype=None, docname=None, description=None):
  37. publish_realtime('progress', {'percent': percent, 'title': title, 'description': description},
  38. user=frappe.session.user, doctype=doctype, docname=docname)
  39. def publish_realtime(event=None, message=None, room=None,
  40. user=None, doctype=None, docname=None, task_id=None,
  41. after_commit=False):
  42. """Publish real-time updates
  43. :param event: Event name, like `task_progress` etc. that will be handled by the client (default is `task_progress` if within task or `global`)
  44. :param message: JSON message object. For async must contain `task_id`
  45. :param room: Room in which to publish update (default entire site)
  46. :param user: Transmit to user
  47. :param doctype: Transmit to doctype, docname
  48. :param docname: Transmit to doctype, docname
  49. :param after_commit: (default False) will emit after current transaction is committed"""
  50. if message is None:
  51. message = {}
  52. if event is None:
  53. if getattr(frappe.local, "task_id", None):
  54. event = "task_progress"
  55. else:
  56. event = "global"
  57. if event=='msgprint' and not user:
  58. user = frappe.session.user
  59. if not room:
  60. if not task_id and hasattr(frappe.local, "task_id"):
  61. task_id = frappe.local.task_id
  62. if task_id:
  63. room = get_task_progress_room(task_id)
  64. if not "task_id" in message:
  65. message["task_id"] = task_id
  66. after_commit = False
  67. elif user:
  68. room = get_user_room(user)
  69. elif doctype and docname:
  70. room = get_doc_room(doctype, docname)
  71. else:
  72. room = get_site_room()
  73. else:
  74. # frappe.chat
  75. room = get_chat_room(room)
  76. # end frappe.chat
  77. if after_commit:
  78. params = [event, message, room]
  79. if not params in frappe.local.realtime_log:
  80. frappe.local.realtime_log.append(params)
  81. else:
  82. emit_via_redis(event, message, room)
  83. def emit_via_redis(event, message, room):
  84. """Publish real-time updates via redis
  85. :param event: Event name, like `task_progress` etc.
  86. :param message: JSON message object. For async must contain `task_id`
  87. :param room: name of the room"""
  88. r = get_redis_server()
  89. try:
  90. r.publish('events', frappe.as_json({'event': event, 'message': message, 'room': room}))
  91. except redis.exceptions.ConnectionError:
  92. # print(frappe.get_traceback())
  93. pass
  94. def put_log(line_no, line, task_id=None):
  95. r = get_redis_server()
  96. if not task_id:
  97. task_id = frappe.local.task_id
  98. task_progress_room = get_task_progress_room(task_id)
  99. task_log_key = "task_log:" + task_id
  100. publish_realtime('task_progress', {
  101. "message": {
  102. "lines": {line_no: line}
  103. },
  104. "task_id": task_id
  105. }, room=task_progress_room)
  106. r.hset(task_log_key, line_no, line)
  107. r.expire(task_log_key, 3600)
  108. def get_redis_server():
  109. """returns redis_socketio connection."""
  110. global redis_server
  111. if not redis_server:
  112. from redis import Redis
  113. redis_server = Redis.from_url(conf.get("redis_socketio")
  114. or "redis://localhost:12311")
  115. return redis_server
  116. class FileAndRedisStream(FileIO):
  117. def __init__(self, *args, **kwargs):
  118. ret = super(FileAndRedisStream, self).__init__(*args, **kwargs)
  119. self.count = 0
  120. return ret
  121. def write(self, data):
  122. ret = super(FileAndRedisStream, self).write(data)
  123. if frappe.local.task_id:
  124. put_log(self.count, data, task_id=frappe.local.task_id)
  125. self.count += 1
  126. return ret
  127. def get_std_streams(task_id):
  128. stdout = FileAndRedisStream(get_task_log_file_path(task_id, 'stdout'), 'w')
  129. # stderr = FileAndRedisStream(get_task_log_file_path(task_id, 'stderr'), 'w')
  130. return stdout, stdout
  131. def get_task_log_file_path(task_id, stream_type):
  132. logs_dir = frappe.utils.get_site_path('task-logs')
  133. return os.path.join(logs_dir, task_id + '.' + stream_type)
  134. @frappe.whitelist(allow_guest=True)
  135. def can_subscribe_doc(doctype, docname, sid):
  136. if os.environ.get('CI'):
  137. return True
  138. from frappe.sessions import Session
  139. from frappe.exceptions import PermissionError
  140. session = Session(None, resume=True).get_session_data()
  141. if not frappe.has_permission(user=session.user, doctype=doctype, doc=docname, ptype='read'):
  142. raise PermissionError()
  143. return True
  144. @frappe.whitelist(allow_guest=True)
  145. def get_user_info(sid):
  146. from frappe.sessions import Session
  147. session = Session(None, resume=True).get_session_data()
  148. return {
  149. 'user': session.user,
  150. }
  151. def get_doc_room(doctype, docname):
  152. return ''.join([frappe.local.site, ':doc:', doctype, '/', docname])
  153. def get_user_room(user):
  154. return ''.join([frappe.local.site, ':user:', user])
  155. def get_site_room():
  156. return ''.join([frappe.local.site, ':all'])
  157. def get_task_progress_room(task_id):
  158. return "".join([frappe.local.site, ":task_progress:", task_id])
  159. # frappe.chat
  160. def get_chat_room(room):
  161. room = ''.join([frappe.local.site, ":room:", room])
  162. return room
  163. # end frappe.chat room