Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.
 
 
 
 
 
 

864 wiersze
26 KiB

  1. #!/usr/bin/env python2.7
  2. # Copyright (c) 2013, Web Notes Technologies Pvt. Ltd. and Contributors
  3. # MIT License. See license.txt
  4. from __future__ import unicode_literals
  5. import sys, os
  6. import frappe
  7. site_arg_optional = ['serve', 'build', 'watch', 'celery']
  8. def get_site(parsed_args):
  9. if not parsed_args.get("site") and os.path.exists(os.path.join(parsed_args["sites_path"], "currentsite.txt")):
  10. with open(os.path.join(parsed_args["sites_path"], "currentsite.txt"), "r") as sitefile:
  11. parsed_args["site"] = sitefile.read().strip()
  12. return parsed_args["site"]
  13. return parsed_args.get("site")
  14. def main():
  15. parsed_args = frappe._dict(vars(setup_parser()))
  16. fn = get_function(parsed_args)
  17. if parsed_args.get("sites_path"):
  18. parsed_args["sites_path"] = parsed_args["sites_path"][0]
  19. else:
  20. parsed_args["sites_path"] = os.environ.get("SITES_PATH", ".")
  21. sites_path = parsed_args.get("sites_path")
  22. if not parsed_args.get("make_app"):
  23. if parsed_args.get("site")=="all":
  24. for site in get_sites(parsed_args["sites_path"]):
  25. print "\nRunning", fn, "for", site
  26. print "-"*50
  27. args = parsed_args.copy()
  28. args["site"] = site
  29. frappe.init(site, sites_path=sites_path)
  30. run(fn, args)
  31. else:
  32. site = get_site(parsed_args)
  33. if fn not in site_arg_optional and not site:
  34. print 'site argument required'
  35. exit(1)
  36. elif site:
  37. frappe.init(site, sites_path=sites_path)
  38. else:
  39. # site argument optional
  40. frappe.init("", sites_path=sites_path)
  41. run(fn, parsed_args)
  42. else:
  43. run(fn, parsed_args)
  44. def cmd(fn):
  45. def new_fn(*args, **kwargs):
  46. import inspect
  47. fnargs, varargs, varkw, defaults = inspect.getargspec(fn)
  48. new_kwargs = {}
  49. for i, a in enumerate(fnargs):
  50. # should not pass an argument more than once
  51. if i >= len(args) and a in kwargs:
  52. new_kwargs[a] = kwargs.get(a)
  53. return fn(*args, **new_kwargs)
  54. return new_fn
  55. def run(fn, args):
  56. if isinstance(args.get(fn), (list, tuple)):
  57. out = globals().get(fn)(*args.get(fn), **args)
  58. else:
  59. out = globals().get(fn)(**args)
  60. return out
  61. def get_function(args):
  62. for fn, val in args.items():
  63. if (val or isinstance(val, list)) and globals().get(fn):
  64. return fn
  65. def get_sites(sites_path=None):
  66. import os
  67. if not sites_path:
  68. sites_path = '.'
  69. return [site for site in os.listdir(sites_path)
  70. if not os.path.islink(os.path.join(sites_path, site))
  71. and os.path.isdir(os.path.join(sites_path, site))
  72. and not site in ('assets',)]
  73. def setup_parser():
  74. import argparse
  75. parser = argparse.ArgumentParser(description="Run frappe utility functions")
  76. setup_install(parser)
  77. setup_utilities(parser)
  78. setup_translation(parser)
  79. setup_test(parser)
  80. parser.add_argument("site", nargs="?")
  81. # common
  82. parser.add_argument("-f", "--force", default=False, action="store_true",
  83. help="Force execution where applicable (look for [-f] in help)")
  84. parser.add_argument("--quiet", default=True, action="store_false", dest="verbose",
  85. help="Do not show verbose output (where applicable)")
  86. return parser.parse_args()
  87. def setup_install(parser):
  88. parser.add_argument("--make_app", default=False, action="store_true",
  89. help="Make a new application with boilerplate")
  90. parser.add_argument("--install", metavar="DB-NAME", nargs=1,
  91. help="Install a new db")
  92. parser.add_argument("--sites_path", metavar="SITES_PATH", nargs=1,
  93. help="path to directory with sites")
  94. parser.add_argument("--install_app", metavar="APP-NAME", nargs=1,
  95. help="Install a new app")
  96. parser.add_argument("--add_to_installed_apps", metavar="APP-NAME", nargs="*",
  97. help="Add these app(s) to Installed Apps")
  98. parser.add_argument("--root-password", nargs=1,
  99. help="Root password for new app")
  100. parser.add_argument("--reinstall", default=False, action="store_true",
  101. help="Install a fresh app in db_name specified in conf.py")
  102. parser.add_argument("--restore", metavar=("DB-NAME", "SQL-FILE"), nargs=2,
  103. help="Restore from an sql file")
  104. parser.add_argument("--add_system_manager", nargs="+",
  105. metavar=("EMAIL", "[FIRST-NAME] [LAST-NAME]"), help="Add a user with all roles")
  106. def setup_test(parser):
  107. parser.add_argument("--run_tests", default=False, action="store_true",
  108. help="Run tests options [-d doctype], [-m module]")
  109. parser.add_argument("--app", metavar="APP-NAME", nargs=1,
  110. help="Run command for specified app")
  111. parser.add_argument("-d", "--doctype", metavar="DOCTYPE", nargs=1,
  112. help="Run command for specified doctype")
  113. parser.add_argument("-m", "--module", metavar="MODULE", nargs=1,
  114. help="Run command for specified module")
  115. def setup_utilities(parser):
  116. # update
  117. parser.add_argument("-u", "--update", nargs="*", metavar=("REMOTE", "BRANCH"),
  118. help="Perform git pull, run patches, sync schema and rebuild files/translations")
  119. parser.add_argument("--reload_gunicorn", default=False, action="store_true", help="reload gunicorn on update")
  120. parser.add_argument("--patch", nargs=1, metavar="PATCH-MODULE",
  121. help="Run a particular patch [-f]")
  122. parser.add_argument("-l", "--latest", default=False, action="store_true",
  123. help="Run patches, sync schema and rebuild files/translations")
  124. parser.add_argument("--sync_all", default=False, action="store_true",
  125. help="Reload all doctypes, pages, etc. using txt files [-f]")
  126. parser.add_argument("--update_all_sites", nargs="*", metavar=("REMOTE", "BRANCH"),
  127. help="Perform git pull, run patches, sync schema and rebuild files/translations")
  128. parser.add_argument("--reload_doc", nargs=3,
  129. metavar=('"MODULE"', '"DOCTYPE"', '"DOCNAME"'))
  130. # build
  131. parser.add_argument("-b", "--build", default=False, action="store_true",
  132. help="Minify + concatenate JS and CSS files, build translations")
  133. parser.add_argument("-w", "--watch", default=False, action="store_true",
  134. help="Watch and concatenate JS and CSS files as and when they change")
  135. # misc
  136. parser.add_argument("--backup", default=False, action="store_true",
  137. help="Take backup of database in backup folder [--with_files]")
  138. parser.add_argument("--move", default=False, action="store_true",
  139. help="Move site to different directory defined by --dest_dir")
  140. parser.add_argument("--dest_dir", nargs=1, metavar="DEST-DIR",
  141. help="Move site to different directory")
  142. parser.add_argument("--with_files", default=False, action="store_true",
  143. help="Also take backup of files")
  144. parser.add_argument("--domain", nargs="*",
  145. help="Get or set domain in Website Settings")
  146. parser.add_argument("--make_conf", nargs="*", metavar=("DB-NAME", "DB-PASSWORD"),
  147. help="Create new conf.py file")
  148. parser.add_argument("--make_custom_server_script", nargs=1, metavar="DOCTYPE",
  149. help="Create new conf.py file")
  150. parser.add_argument("--set_admin_password", metavar='ADMIN-PASSWORD', nargs=1,
  151. help="Set administrator password")
  152. parser.add_argument("--request", metavar='URL-ARGS', nargs=1, help="Run request as admin")
  153. parser.add_argument("--mysql", action="store_true", help="get mysql shell for a site")
  154. parser.add_argument("--serve", action="store_true", help="Run development server")
  155. parser.add_argument("--profile", action="store_true", help="enable profiling in development server")
  156. parser.add_argument("--smtp", action="store_true", help="Run smtp debug server",
  157. dest="smtp_debug_server")
  158. parser.add_argument("--python", action="store_true", help="get python shell for a site")
  159. parser.add_argument("--flush_memcache", action="store_true", help="flush memcached")
  160. parser.add_argument("--ipython", action="store_true", help="get ipython shell for a site")
  161. parser.add_argument("--execute", help="execute a function", nargs=1, metavar="FUNCTION")
  162. parser.add_argument("--get_site_status", action="store_true", help="Get site details")
  163. parser.add_argument("--update_site_config", nargs=1,
  164. metavar="site-CONFIG-JSON",
  165. help="Update site_config.json for a given site")
  166. parser.add_argument("--port", default=8000, type=int, help="port for development server")
  167. parser.add_argument("--use", action="store_true", help="Set current site for development.")
  168. # clear
  169. parser.add_argument("--clear_web", default=False, action="store_true",
  170. help="Clear website cache")
  171. parser.add_argument("--build_sitemap", default=False, action="store_true",
  172. help="Build Website Route")
  173. parser.add_argument("--sync_statics", default=False, action="store_true",
  174. help="Sync files from templates/statics to Web Pages")
  175. parser.add_argument("--clear_cache", default=False, action="store_true",
  176. help="Clear cache, doctype cache and defaults")
  177. parser.add_argument("--reset_perms", default=False, action="store_true",
  178. help="Reset permissions for all doctypes")
  179. # scheduler
  180. parser.add_argument("--run_scheduler", default=False, action="store_true",
  181. help="Trigger scheduler")
  182. parser.add_argument("--celery", nargs="*", help="Run Celery Commands")
  183. parser.add_argument("--run_scheduler_event", nargs=1,
  184. metavar="all | daily | weekly | monthly",
  185. help="Run a scheduler event")
  186. # replace
  187. parser.add_argument("--replace", nargs=3,
  188. metavar=("SEARCH-REGEX", "REPLACE-BY", "FILE-EXTN"),
  189. help="Multi-file search-replace [-f]")
  190. # import/export
  191. parser.add_argument("--export_doc", nargs=2, metavar=('"DOCTYPE"', '"DOCNAME"'))
  192. parser.add_argument("--export_doclist", nargs=3, metavar=("DOCTYPE", "NAME", "PATH"),
  193. help="""Export doclist as json to the given path, use '-' as name for Singles.""")
  194. parser.add_argument("--export_csv", nargs=2, metavar=("DOCTYPE", "PATH"),
  195. help="""Dump DocType as csv""")
  196. parser.add_argument("--export_fixtures", default=False, action="store_true",
  197. help="""Export fixtures""")
  198. parser.add_argument("--import_doclist", nargs=1, metavar="PATH",
  199. help="""Import (insert/update) doclist. If the argument is a directory, all files ending with .json are imported""")
  200. def setup_translation(parser):
  201. parser.add_argument("--build_message_files", default=False, action="store_true",
  202. help="Build message files for translation.")
  203. parser.add_argument("--get_untranslated", nargs=2, metavar=("LANG-CODE", "TARGET-FILE-PATH"),
  204. help="""Get untranslated strings for lang.""")
  205. parser.add_argument("--update_translations", nargs=3,
  206. metavar=("LANG-CODE", "UNTRANSLATED-FILE-PATH", "TRANSLATED-FILE-PATH"),
  207. help="""Update translated strings.""")
  208. # methods
  209. @cmd
  210. def make_app():
  211. from frappe.utils.boilerplate import make_boilerplate
  212. make_boilerplate()
  213. @cmd
  214. def use(sites_path):
  215. with open(os.path.join(sites_path, "currentsite.txt"), "w") as sitefile:
  216. sitefile.write(frappe.local.site)
  217. # install
  218. @cmd
  219. def install(db_name, root_login="root", root_password=None, source_sql=None,
  220. admin_password = 'admin', verbose=True, force=False, site_config=None, reinstall=False):
  221. from frappe.installer import install_db, install_app, make_site_dirs
  222. install_db(root_login=root_login, root_password=root_password, db_name=db_name, source_sql=source_sql,
  223. admin_password = admin_password, verbose=verbose, force=force, site_config=site_config, reinstall=reinstall)
  224. make_site_dirs()
  225. install_app("frappe", verbose=verbose, set_as_patched=not source_sql)
  226. frappe.destroy()
  227. @cmd
  228. def install_app(app_name, verbose=False):
  229. from frappe.installer import install_app
  230. frappe.connect()
  231. install_app(app_name, verbose=verbose)
  232. frappe.destroy()
  233. @cmd
  234. def add_to_installed_apps(*apps):
  235. from frappe.installer import add_to_installed_apps
  236. frappe.connect()
  237. all_apps = frappe.get_all_apps(with_frappe=True)
  238. for each in apps:
  239. if each in all_apps:
  240. add_to_installed_apps(each)
  241. frappe.destroy()
  242. @cmd
  243. def reinstall(verbose=True):
  244. install(db_name=frappe.conf.db_name, verbose=verbose, force=True, reinstall=True)
  245. @cmd
  246. def restore(db_name, source_sql, verbose=True, force=False):
  247. install(db_name, source_sql=source_sql, verbose=verbose, force=force)
  248. @cmd
  249. def add_system_manager(email, first_name=None, last_name=None):
  250. import frappe.utils.user
  251. frappe.connect()
  252. frappe.utils.user.add_system_manager(email, first_name, last_name)
  253. frappe.db.commit()
  254. frappe.destroy()
  255. # utilities
  256. @cmd
  257. def update(remote=None, branch=None, reload_gunicorn=False):
  258. pull(remote=remote, branch=branch)
  259. # maybe there are new framework changes, any consequences?
  260. reload(frappe)
  261. build()
  262. latest()
  263. if reload_gunicorn:
  264. import subprocess
  265. subprocess.check_output("killall -HUP gunicorn".split())
  266. @cmd
  267. def latest(verbose=True, rebuild_website_config=True):
  268. import frappe.modules.patch_handler
  269. import frappe.model.sync
  270. from frappe.website import rebuild_config
  271. from frappe.utils.fixtures import sync_fixtures
  272. import frappe.translate
  273. from frappe.website import statics
  274. frappe.connect()
  275. try:
  276. # run patches
  277. frappe.local.patch_log_list = []
  278. frappe.modules.patch_handler.run_all()
  279. if verbose:
  280. print "\n".join(frappe.local.patch_log_list)
  281. # sync
  282. frappe.model.sync.sync_all()
  283. # build website config if any changes in templates etc.
  284. if rebuild_website_config:
  285. rebuild_config()
  286. statics.sync().start()
  287. sync_fixtures()
  288. frappe.translate.clear_cache()
  289. except frappe.modules.patch_handler.PatchError, e:
  290. print "\n".join(frappe.local.patch_log_list)
  291. raise
  292. finally:
  293. frappe.destroy()
  294. @cmd
  295. def sync_all(force=False):
  296. import frappe.model.sync
  297. frappe.connect()
  298. frappe.model.sync.sync_all(force=force)
  299. frappe.destroy()
  300. @cmd
  301. def patch(patch_module, force=False):
  302. import frappe.modules.patch_handler
  303. frappe.connect()
  304. frappe.local.patch_log_list = []
  305. frappe.modules.patch_handler.run_single(patch_module, force=force)
  306. print "\n".join(frappe.local.patch_log_list)
  307. frappe.destroy()
  308. @cmd
  309. def update_all_sites(remote=None, branch=None, verbose=True):
  310. pull(remote, branch)
  311. # maybe there are new framework changes, any consequences?
  312. reload(frappe)
  313. build()
  314. for site in get_sites():
  315. frappe.init(site)
  316. latest(verbose=verbose)
  317. @cmd
  318. def reload_doc(module, doctype, docname, force=False):
  319. frappe.connect()
  320. frappe.reload_doc(module, doctype, docname, force=force)
  321. frappe.db.commit()
  322. frappe.destroy()
  323. @cmd
  324. def build():
  325. import frappe.build
  326. import frappe
  327. frappe.build.bundle(False)
  328. @cmd
  329. def watch():
  330. import frappe.build
  331. frappe.build.watch(True)
  332. @cmd
  333. def backup(with_files=False, verbose=True, backup_path_db=None, backup_path_files=None):
  334. from frappe.utils.backups import scheduled_backup
  335. frappe.connect()
  336. odb = scheduled_backup(ignore_files=not with_files, backup_path_db=backup_path_db, backup_path_files=backup_path_files)
  337. if verbose:
  338. from frappe.utils import now
  339. print "database backup taken -", odb.backup_path_db, "- on", now()
  340. if with_files:
  341. print "files backup taken -", odb.backup_path_files, "- on", now()
  342. frappe.destroy()
  343. return odb
  344. @cmd
  345. def move(dest_dir=None):
  346. import os
  347. if not dest_dir:
  348. raise Exception, "--dest_dir is required for --move"
  349. if not os.path.isdir(dest_dir):
  350. raise Exception, "destination is not a directory or does not exist"
  351. old_path = frappe.utils.get_site()
  352. new_path = os.path.join(dest_dir, site)
  353. # check if site dump of same name already exists
  354. site_dump_exists = True
  355. count = 0
  356. while site_dump_exists:
  357. final_new_path = new_path + (count and str(count) or "")
  358. site_dump_exists = os.path.exists(final_new_path)
  359. count = int(count or 0) + 1
  360. os.rename(old_path, final_new_path)
  361. frappe.destroy()
  362. return os.path.basename(final_new_path)
  363. @cmd
  364. def domain(host_url=None):
  365. frappe.connect()
  366. if host_url:
  367. frappe.db.set_value("Website Settings", None, "subdomain", host_url)
  368. frappe.db.commit()
  369. else:
  370. print frappe.db.get_value("Website Settings", None, "subdomain")
  371. frappe.destroy()
  372. @cmd
  373. def make_conf(db_name=None, db_password=None, site_config=None):
  374. from frappe.install_lib.install import make_conf
  375. make_conf(db_name=db_name, db_password=db_password, site_config=site_config)
  376. @cmd
  377. def make_custom_server_script(doctype):
  378. from frappe.core.doctype.custom_script.custom_script import make_custom_server_script_file
  379. frappe.connect()
  380. make_custom_server_script_file(doctype)
  381. frappe.destroy()
  382. # clear
  383. @cmd
  384. def clear_cache():
  385. import frappe.sessions
  386. frappe.connect()
  387. frappe.clear_cache()
  388. frappe.destroy()
  389. @cmd
  390. def clear_web():
  391. import frappe.website.render
  392. frappe.connect()
  393. frappe.website.render.clear_cache()
  394. frappe.destroy()
  395. @cmd
  396. def build_sitemap():
  397. from frappe.website import rebuild_config
  398. frappe.connect()
  399. rebuild_config()
  400. frappe.destroy()
  401. @cmd
  402. def sync_statics():
  403. from frappe.website import statics
  404. frappe.connect()
  405. statics.sync_statics()
  406. frappe.db.commit()
  407. frappe.destroy()
  408. @cmd
  409. def reset_perms():
  410. frappe.connect()
  411. for d in frappe.db.sql_list("""select name from `tabDocType`
  412. where ifnull(istable, 0)=0 and ifnull(custom, 0)=0"""):
  413. frappe.clear_cache(doctype=d)
  414. frappe.reset_perms(d)
  415. frappe.destroy()
  416. @cmd
  417. def execute(method):
  418. frappe.connect()
  419. ret = frappe.get_attr(method)()
  420. frappe.db.commit()
  421. frappe.destroy()
  422. if ret:
  423. print ret
  424. @cmd
  425. def celery(arg):
  426. import frappe
  427. import commands, os
  428. python = commands.getoutput('which python')
  429. os.execv(python, [python, "-m", "frappe.celery_app"] + arg.split())
  430. frappe.destroy()
  431. @cmd
  432. def run_scheduler_event(event, force=False):
  433. import frappe.utils.scheduler
  434. frappe.connect()
  435. frappe.utils.scheduler.trigger(frappe.local.site, event, now=force)
  436. frappe.destroy()
  437. # replace
  438. @cmd
  439. def replace(search_regex, replacement, extn, force=False):
  440. print search_regex, replacement, extn
  441. replace_code('.', search_regex, replacement, extn, force=force)
  442. # import/export
  443. @cmd
  444. def export_doc(doctype, docname):
  445. import frappe.modules
  446. frappe.connect()
  447. frappe.modules.export_doc(doctype, docname)
  448. frappe.destroy()
  449. @cmd
  450. def export_doclist(doctype, name, path):
  451. from frappe.core.page.data_import_tool import data_import_tool
  452. frappe.connect()
  453. data_import_tool.export_json(doctype, name, path)
  454. frappe.destroy()
  455. @cmd
  456. def export_csv(doctype, path):
  457. from frappe.core.page.data_import_tool import data_import_tool
  458. frappe.connect()
  459. data_import_tool.export_csv(doctype, path)
  460. frappe.destroy()
  461. @cmd
  462. def export_fixtures():
  463. from frappe.utils.fixtures import export_fixtures
  464. frappe.connect()
  465. export_fixtures()
  466. frappe.destroy()
  467. @cmd
  468. def import_doclist(path, force=False):
  469. from frappe.core.page.data_import_tool import data_import_tool
  470. frappe.connect()
  471. data_import_tool.import_doclist(path, overwrite=force)
  472. frappe.destroy()
  473. # translation
  474. @cmd
  475. def build_message_files():
  476. import frappe.translate
  477. frappe.connect()
  478. frappe.translate.rebuild_all_translation_files()
  479. frappe.destroy()
  480. @cmd
  481. def get_untranslated(lang, untranslated_file):
  482. import frappe.translate
  483. frappe.connect()
  484. frappe.translate.get_untranslated(lang, untranslated_file)
  485. frappe.destroy()
  486. @cmd
  487. def update_translations(lang, untranslated_file, translated_file):
  488. import frappe.translate
  489. frappe.connect()
  490. frappe.translate.update_translations(lang, untranslated_file, translated_file)
  491. frappe.destroy()
  492. # git
  493. @cmd
  494. def git(param):
  495. if isinstance(param, (list, tuple)):
  496. param = " ".join(param)
  497. import os
  498. os.system("""cd lib && git %s""" % param)
  499. os.system("""cd app && git %s""" % param)
  500. def get_remote_and_branch(remote=None, branch=None):
  501. if not (remote and branch):
  502. if not frappe.conf.branch:
  503. raise Exception("Please specify remote and branch")
  504. remote = remote or "origin"
  505. branch = branch or frappe.conf.branch
  506. frappe.destroy()
  507. return remote, branch
  508. @cmd
  509. def pull(remote=None, branch=None):
  510. remote, branch = get_remote_and_branch(remote, branch)
  511. git(("pull", remote, branch))
  512. @cmd
  513. def push(remote=None, branch=None):
  514. remote, branch = get_remote_and_branch(remote, branch)
  515. git(("push", remote, branch))
  516. @cmd
  517. def status():
  518. git("status")
  519. @cmd
  520. def commit(message):
  521. git("""commit -a -m "%s" """ % message.replace('"', '\"'))
  522. @cmd
  523. def checkout(branch):
  524. git(("checkout", branch))
  525. @cmd
  526. def set_admin_password(admin_password):
  527. import frappe
  528. frappe.connect()
  529. frappe.db.sql("""update __Auth set `password`=password(%s)
  530. where user='Administrator'""", (admin_password,))
  531. frappe.db.commit()
  532. frappe.destroy()
  533. @cmd
  534. def mysql():
  535. import frappe
  536. import commands, os
  537. msq = commands.getoutput('which mysql')
  538. os.execv(msq, [msq, '-u', frappe.conf.db_name, '-p'+frappe.conf.db_password, frappe.conf.db_name, '-h', frappe.conf.db_host or "localhost", "-A"])
  539. frappe.destroy()
  540. @cmd
  541. def python(site):
  542. import frappe
  543. import commands, os
  544. python = commands.getoutput('which python')
  545. if site:
  546. os.environ["site"] = site
  547. os.environ["PYTHONSTARTUP"] = os.path.join(os.path.dirname(frappe.__file__), "pythonrc.py")
  548. os.execv(python, [python])
  549. frappe.destroy()
  550. @cmd
  551. def ipython(site):
  552. import frappe
  553. frappe.connect(site=site)
  554. import IPython
  555. IPython.embed()
  556. @cmd
  557. def smtp_debug_server():
  558. import commands, os
  559. python = commands.getoutput('which python')
  560. os.execv(python, [python, '-m', "smtpd", "-n", "-c", "DebuggingServer", "localhost:25"])
  561. @cmd
  562. def run_tests(app=None, module=None, doctype=None, verbose=False):
  563. import frappe.test_runner
  564. ret = frappe.test_runner.main(app and app[0], module and module[0], doctype and doctype[0], verbose)
  565. if len(ret.failures) > 0 or len(ret.errors) > 0:
  566. exit(1)
  567. @cmd
  568. def serve(port=8000, profile=False, sites_path='.', site=None):
  569. import frappe.app
  570. frappe.app.serve(port=port, profile=profile, site=frappe.local.site, sites_path=sites_path)
  571. @cmd
  572. def request(args):
  573. import frappe.handler
  574. import frappe.api
  575. frappe.connect()
  576. if "?" in args:
  577. frappe.local.form_dict = frappe._dict([a.split("=") for a in args.split("?")[-1].split("&")])
  578. else:
  579. frappe.local.form_dict = frappe._dict()
  580. if args.startswith("/api/method"):
  581. frappe.local.form_dict.cmd = args.split("?")[0].split("/")[-1]
  582. frappe.handler.execute_cmd(frappe.form_dict.cmd)
  583. print frappe.response
  584. frappe.destroy()
  585. @cmd
  586. def flush_memcache():
  587. frappe.cache().flush_all()
  588. def replace_code(start, txt1, txt2, extn, search=None, force=False):
  589. """replace all txt1 by txt2 in files with extension (extn)"""
  590. import frappe.utils
  591. import os, re
  592. esc = frappe.utils.make_esc('[]')
  593. if not search: search = esc(txt1)
  594. for wt in os.walk(start, followlinks=1):
  595. for fn in wt[2]:
  596. if fn.split('.')[-1]==extn:
  597. fpath = os.path.join(wt[0], fn)
  598. with open(fpath, 'r') as f:
  599. content = f.read()
  600. if re.search(search, content):
  601. res = search_replace_with_prompt(fpath, txt1, txt2, force)
  602. if res == 'skip':
  603. return 'skip'
  604. def search_replace_with_prompt(fpath, txt1, txt2, force=False):
  605. """ Search and replace all txt1 by txt2 in the file with confirmation"""
  606. from termcolor import colored
  607. with open(fpath, 'r') as f:
  608. content = f.readlines()
  609. tmp = []
  610. for c in content:
  611. if c.find(txt1) != -1:
  612. print fpath
  613. print colored(txt1, 'red').join(c[:-1].split(txt1))
  614. a = ''
  615. if force:
  616. c = c.replace(txt1, txt2)
  617. else:
  618. while a.lower() not in ['y', 'n', 'skip']:
  619. a = raw_input('Do you want to Change [y/n/skip]?')
  620. if a.lower() == 'y':
  621. c = c.replace(txt1, txt2)
  622. elif a.lower() == 'skip':
  623. return 'skip'
  624. tmp.append(c)
  625. with open(fpath, 'w') as f:
  626. f.write(''.join(tmp))
  627. print colored('Updated', 'green')
  628. @cmd
  629. def get_site_status(verbose=False):
  630. import frappe
  631. import frappe.utils
  632. from frappe.utils.user import get_system_managers
  633. from frappe.core.doctype.user.user import get_total_users, get_active_users, \
  634. get_website_users, get_active_website_users
  635. import json
  636. frappe.connect()
  637. ret = {
  638. 'last_backup_on': frappe.local.conf.last_backup_on,
  639. 'active_users': get_active_users(),
  640. 'total_users': get_total_users(),
  641. 'active_website_users': get_active_website_users(),
  642. 'website_users': get_website_users(),
  643. 'system_managers': "\n".join(get_system_managers()),
  644. 'default_company': frappe.db.get_default("company"),
  645. 'disk_usage': frappe.utils.get_disk_usage(),
  646. 'working_directory': frappe.local.site_path
  647. }
  648. # country, timezone, industry
  649. control_panel_details = frappe.db.get_value("Control Panel", "Control Panel",
  650. ["country", "time_zone", "industry"], as_dict=True)
  651. if control_panel_details:
  652. ret.update(control_panel_details)
  653. # basic usage/progress analytics
  654. for doctype in ("Company", "Customer", "Item", "Quotation", "Sales Invoice",
  655. "Journal Voucher", "Stock Ledger Entry"):
  656. key = doctype.lower().replace(" ", "_") + "_exists"
  657. ret[key] = 1 if frappe.db.count(doctype) else 0
  658. frappe.destroy()
  659. if verbose:
  660. print json.dumps(ret, indent=1, sort_keys=True)
  661. return ret
  662. @cmd
  663. def update_site_config(site_config, verbose=False):
  664. import json
  665. if isinstance(site_config, basestring):
  666. site_config = json.loads(site_config)
  667. config = frappe.get_site_config()
  668. config.update(site_config)
  669. site_config_path = os.path.join(frappe.local.site_path, "site_config.json")
  670. with open(site_config_path, "w") as f:
  671. json.dump(config, f, indent=1, sort_keys=True)
  672. frappe.destroy()
  673. @cmd
  674. def bump(repo, bump_type):
  675. import json
  676. assert repo in ['lib', 'app']
  677. assert bump_type in ['minor', 'major', 'patch']
  678. def validate(repo_path):
  679. import git
  680. repo = git.Repo(repo_path)
  681. if repo.active_branch != 'master':
  682. raise Exception, "Current branch not master in {}".format(repo_path)
  683. def bump_version(version, version_type):
  684. import semantic_version
  685. v = semantic_version.Version(version)
  686. if version_type == 'minor':
  687. v.minor += 1
  688. elif version_type == 'major':
  689. v.major += 1
  690. elif version_type == 'patch':
  691. v.patch += 1
  692. return unicode(v)
  693. def add_tag(repo_path, version):
  694. import git
  695. repo = git.Repo(repo_path)
  696. repo.index.add(['config.json'])
  697. repo.index.commit('bumped to version {}'.format(version))
  698. repo.create_tag('v' + version, repo.head)
  699. def update_framework_requirement(version):
  700. with open('app/config.json') as f:
  701. config = json.load(f)
  702. config['requires_framework_version'] = '==' + version
  703. with open('app/config.json', 'w') as f:
  704. json.dump(config, f, indent=1, sort_keys=True)
  705. validate('lib/')
  706. validate('app/')
  707. if repo == 'app':
  708. with open('app/config.json') as f:
  709. config = json.load(f)
  710. new_version = bump_version(config['app_version'], bump_type)
  711. config['app_version'] = new_version
  712. with open('app/config.json', 'w') as f:
  713. json.dump(config, f, indent=1, sort_keys=True)
  714. add_tag('app/', new_version)
  715. elif repo == 'lib':
  716. with open('lib/config.json') as f:
  717. config = json.load(f)
  718. new_version = bump_version(config['framework_version'], bump_type)
  719. config['framework_version'] = new_version
  720. with open('lib/config.json', 'w') as f:
  721. json.dump(config, f, indent=1, sort_keys=True)
  722. add_tag('lib/', new_version)
  723. update_framework_requirement(new_version)
  724. bump('app', bump_type)
  725. if __name__=="__main__":
  726. main()