25개 이상의 토픽을 선택하실 수 없습니다. Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

155 lines
4.7 KiB

  1. # Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
  2. # MIT License. See license.txt
  3. from __future__ import unicode_literals, print_function
  4. import frappe, os, json
  5. from frappe.modules import get_module_path, scrub_dt_dn
  6. from frappe.utils import get_datetime_str
  7. from frappe.model.base_document import get_controller
  8. ignore_values = {
  9. "Report": ["disabled", "prepared_report", "add_total_row"],
  10. "Print Format": ["disabled"],
  11. "Notification": ["enabled"],
  12. "Print Style": ["disabled"],
  13. "Module Onboarding": ['is_complete'],
  14. "Onboarding Step": ['is_complete', 'is_skipped']
  15. }
  16. ignore_doctypes = [""]
  17. def import_files(module, dt=None, dn=None, force=False, pre_process=None, reset_permissions=False):
  18. if type(module) is list:
  19. out = []
  20. for m in module:
  21. out.append(import_file(m[0], m[1], m[2], force=force, pre_process=pre_process,
  22. reset_permissions=reset_permissions))
  23. return out
  24. else:
  25. return import_file(module, dt, dn, force=force, pre_process=pre_process,
  26. reset_permissions=reset_permissions)
  27. def import_file(module, dt, dn, force=False, pre_process=None, reset_permissions=False):
  28. """Sync a file from txt if modifed, return false if not updated"""
  29. path = get_file_path(module, dt, dn)
  30. ret = import_file_by_path(path, force, pre_process=pre_process, reset_permissions=reset_permissions)
  31. return ret
  32. def get_file_path(module, dt, dn):
  33. dt, dn = scrub_dt_dn(dt, dn)
  34. path = os.path.join(get_module_path(module),
  35. os.path.join(dt, dn, dn + ".json"))
  36. return path
  37. def import_file_by_path(path, force=False, data_import=False, pre_process=None, ignore_version=None,
  38. reset_permissions=False, for_sync=False):
  39. try:
  40. docs = read_doc_from_file(path)
  41. except IOError:
  42. print (path + " missing")
  43. return
  44. if docs:
  45. if not isinstance(docs, list):
  46. docs = [docs]
  47. for doc in docs:
  48. if not force:
  49. # check if timestamps match
  50. db_modified = frappe.db.get_value(doc['doctype'], doc['name'], 'modified')
  51. if db_modified and doc.get('modified')==get_datetime_str(db_modified):
  52. return False
  53. original_modified = doc.get("modified")
  54. frappe.flags.in_import = True
  55. import_doc(doc, force=force, data_import=data_import, pre_process=pre_process,
  56. ignore_version=ignore_version, reset_permissions=reset_permissions)
  57. frappe.flags.in_import = False
  58. if original_modified:
  59. # since there is a new timestamp on the file, update timestamp in
  60. if doc["doctype"] == doc["name"] and doc["name"]!="DocType":
  61. frappe.db.sql("""update tabSingles set value=%s where field="modified" and doctype=%s""",
  62. (original_modified, doc["name"]))
  63. else:
  64. frappe.db.sql("update `tab%s` set modified=%s where name=%s" % \
  65. (doc['doctype'], '%s', '%s'),
  66. (original_modified, doc['name']))
  67. return True
  68. def read_doc_from_file(path):
  69. doc = None
  70. if os.path.exists(path):
  71. with open(path, 'r') as f:
  72. try:
  73. doc = json.loads(f.read())
  74. except ValueError:
  75. print("bad json: {0}".format(path))
  76. raise
  77. else:
  78. raise IOError('%s missing' % path)
  79. return doc
  80. def import_doc(docdict, force=False, data_import=False, pre_process=None,
  81. ignore_version=None, reset_permissions=False):
  82. frappe.flags.in_import = True
  83. docdict["__islocal"] = 1
  84. controller = get_controller(docdict['doctype'])
  85. if controller and hasattr(controller, 'prepare_for_import') and callable(getattr(controller, 'prepare_for_import')):
  86. controller.prepare_for_import(docdict)
  87. doc = frappe.get_doc(docdict)
  88. # Note on Tree DocTypes:
  89. # The tree structure is maintained in the database via the fields "lft" and
  90. # "rgt". They are automatically set and kept up-to-date. Importing them
  91. # would destroy any existing tree structure.
  92. if getattr(doc.meta, 'is_tree', None) and any([doc.lft, doc.rgt]):
  93. print('Ignoring values of `lft` and `rgt` for {} "{}"'.format(doc.doctype, doc.name))
  94. doc.lft = None
  95. doc.rgt = None
  96. doc.run_method("before_import")
  97. doc.flags.ignore_version = ignore_version
  98. if pre_process:
  99. pre_process(doc)
  100. ignore = []
  101. if frappe.db.exists(doc.doctype, doc.name):
  102. old_doc = frappe.get_doc(doc.doctype, doc.name)
  103. if doc.doctype in ignore_values:
  104. # update ignore values
  105. for key in ignore_values.get(doc.doctype) or []:
  106. doc.set(key, old_doc.get(key))
  107. # update ignored docs into new doc
  108. for df in doc.meta.get_table_fields():
  109. if df.options in ignore_doctypes and not reset_permissions:
  110. doc.set(df.fieldname, [])
  111. ignore.append(df.options)
  112. # delete old
  113. frappe.delete_doc(doc.doctype, doc.name, force=1, ignore_doctypes=ignore, for_reload=True)
  114. doc.flags.ignore_children_type = ignore
  115. doc.flags.ignore_links = True
  116. if not data_import:
  117. doc.flags.ignore_validate = True
  118. doc.flags.ignore_permissions = True
  119. doc.flags.ignore_mandatory = True
  120. doc.insert()
  121. frappe.flags.in_import = False