|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566 |
- # Copyright (c) 2013, Web Notes Technologies Pvt. Ltd. and Contributors
- # MIT License. See license.txt
- """
- globals attached to webnotes module
- + some utility functions that should probably be moved
- """
-
- from __future__ import unicode_literals
-
- from werkzeug.local import Local, release_local
- from werkzeug.exceptions import NotFound
- from MySQLdb import ProgrammingError as SQLError
-
- import os, sys, importlib
- import json
- import semantic_version
-
- from webnotes.core.doctype.print_format.print_format import get_html as get_print_html
-
- local = Local()
-
- class _dict(dict):
- """dict like object that exposes keys as attributes"""
- def __getattr__(self, key):
- return self.get(key)
- def __setattr__(self, key, value):
- self[key] = value
- def __getstate__(self):
- return self
- def __setstate__(self, d):
- self.update(d)
- def update(self, d):
- """update and return self -- the missing dict feature in python"""
- super(_dict, self).update(d)
- return self
- def copy(self):
- return _dict(dict(self).copy())
-
- def __getattr__(self, key):
- return local.get("key", None)
-
- def _(msg):
- """translate object in current lang, if exists"""
- if local.lang == "en":
- return msg
-
- from webnotes.translate import get_full_dict
- return get_full_dict(local.lang).get(msg, msg)
-
- def get_lang_dict(fortype, name=None):
- if local.lang=="en":
- return {}
- from webnotes.translate import get_dict
- return get_dict(fortype, name)
-
- def set_user_lang(user, user_language=None):
- from webnotes.translate import get_lang_dict
-
- if not user_language:
- user_language = conn.get_value("Profile", user, "language")
-
- if user_language:
- lang_dict = get_lang_dict()
- if user_language in lang_dict:
- local.lang = lang_dict[user_language]
-
- # local-globals
- conn = local("conn")
- conf = local("conf")
- form = form_dict = local("form_dict")
- request = local("request")
- request_method = local("request_method")
- response = local("response")
- _response = local("_response")
- session = local("session")
- user = local("user")
- flags = local("flags")
- restrictions = local("restrictions")
-
- error_log = local("error_log")
- debug_log = local("debug_log")
- message_log = local("message_log")
-
- lang = local("lang")
-
- def init(site, sites_path=None):
- if getattr(local, "initialised", None):
- return
-
- if not sites_path:
- sites_path = '.'
-
- local.error_log = []
- local.site = site
- local.sites_path = sites_path
- local.site_path = os.path.join(sites_path, site)
- local.message_log = []
- local.debug_log = []
- local.response = _dict({})
- local.lang = "en"
- local.request_method = request.method if request else None
- local.conf = _dict(get_site_config())
- local.initialised = True
- local.flags = _dict({})
- local.rollback_observers = []
- local.module_app = None
- local.app_modules = None
- local.user = None
- local.restrictions = None
- local.user_perms = {}
- local.test_objects = {}
-
- setup_module_map()
-
- def get_site_config():
- site_filepath = os.path.join(local.site_path, "site_config.json")
- if os.path.exists(site_filepath):
- with open(site_filepath, 'r') as f:
- return json.load(f)
- else:
- return _dict()
-
- def destroy():
- """closes connection and releases werkzeug local"""
- if conn:
- conn.close()
-
- release_local(local)
-
- _memc = None
-
- # memcache
- def cache():
- global _memc
- if not _memc:
- from webnotes.memc import MClient
- _memc = MClient(['localhost:11211'])
- return _memc
-
- class DuplicateEntryError(Exception): pass
- class ValidationError(Exception): pass
- class AuthenticationError(Exception): pass
- class PermissionError(Exception): pass
- class DataError(Exception): pass
- class UnknownDomainError(Exception): pass
- class SessionStopped(Exception): pass
- class MappingMismatchError(ValidationError): pass
- class InvalidStatusError(ValidationError): pass
- class DoesNotExistError(ValidationError): pass
- class MandatoryError(ValidationError): pass
- class InvalidSignatureError(ValidationError): pass
- class RateLimitExceededError(ValidationError): pass
- class OutgoingEmailError(Exception): pass
-
- def get_traceback():
- import utils
- return utils.get_traceback()
-
- def errprint(msg):
- from utils import cstr
- if not request:
- print cstr(msg)
-
- error_log.append(cstr(msg))
-
- def log(msg):
- if not request:
- if conf.get("logging") or False:
- print repr(msg)
-
- from utils import cstr
- debug_log.append(cstr(msg))
-
- def msgprint(msg, small=0, raise_exception=0, as_table=False):
- def _raise_exception():
- if raise_exception:
- if flags.rollback_on_exception:
- conn.rollback()
- import inspect
- if inspect.isclass(raise_exception) and issubclass(raise_exception, Exception):
- raise raise_exception, msg
- else:
- raise ValidationError, msg
-
- if flags.mute_messages:
- _raise_exception()
- return
-
- from utils import cstr
- if as_table and type(msg) in (list, tuple):
- msg = '<table border="1px" style="border-collapse: collapse" cellpadding="2px">' + ''.join(['<tr>'+''.join(['<td>%s</td>' % c for c in r])+'</tr>' for r in msg]) + '</table>'
-
- if flags.print_messages:
- print "Message: " + repr(msg)
-
- message_log.append((small and '__small:' or '')+cstr(msg or ''))
- _raise_exception()
-
- def throw(msg, exc=ValidationError):
- msgprint(msg, raise_exception=exc)
-
- def create_folder(path):
- if not os.path.exists(path): os.makedirs(path)
-
- def connect(site=None, db_name=None):
- from db import Database
- if site:
- init(site)
- local.conn = Database(user=db_name or local.conf.db_name)
- local.response = _dict()
- local.form_dict = _dict()
- local.session = _dict()
- set_user("Administrator")
-
- def set_user(username):
- import webnotes.profile
- local.session["user"] = username
- local.user = webnotes.profile.Profile(username)
- local.restrictions = None
- local.user_perms = {}
-
- def get_request_header(key, default=None):
- return request.headers.get(key, default)
-
- def sendmail(recipients=[], sender="", subject="No Subject", message="No Message", as_markdown=False):
- import webnotes.utils.email_lib
- if as_markdown:
- webnotes.utils.email_lib.sendmail_md(recipients, sender=sender, subject=subject, msg=message)
- else:
- webnotes.utils.email_lib.sendmail(recipients, sender=sender, subject=subject, msg=message)
-
- logger = None
- whitelisted = []
- guest_methods = []
- def whitelist(allow_guest=False):
- """
- decorator for whitelisting a function
-
- Note: if the function is allowed to be accessed by a guest user,
- it must explicitly be marked as allow_guest=True
-
- for specific roles, set allow_roles = ['Administrator'] etc.
- """
- def innerfn(fn):
- global whitelisted, guest_methods
- whitelisted.append(fn)
-
- if allow_guest:
- guest_methods.append(fn)
-
- return fn
-
- return innerfn
-
- def only_for(roles):
- if not isinstance(roles, (tuple, list)):
- roles = (roles,)
- roles = set(roles)
- myroles = set(get_roles())
- if not roles.intersection(myroles):
- raise PermissionError
-
- def clear_cache(user=None, doctype=None):
- """clear cache"""
- import webnotes.sessions
- if doctype:
- import webnotes.model.doctype
- webnotes.model.doctype.clear_cache(doctype)
- reset_metadata_version()
- elif user:
- webnotes.sessions.clear_cache(user)
- else: # everything
- import translate
- webnotes.sessions.clear_cache()
- translate.clear_cache()
- reset_metadata_version()
-
- def get_roles(username=None):
- import webnotes.profile
- if not local.session:
- return ["Guest"]
- elif not username or username==local.session.user:
- return local.user.get_roles()
- else:
- return webnotes.profile.Profile(username).get_roles()
-
- def has_permission(doctype, ptype="read", refdoc=None):
- import webnotes.permissions
- return webnotes.permissions.has_permission(doctype, ptype, refdoc)
-
- def clear_perms(doctype):
- conn.sql("""delete from tabDocPerm where parent=%s""", doctype)
-
- def reset_perms(doctype):
- clear_perms(doctype)
- reload_doc(conn.get_value("DocType", doctype, "module"),
- "DocType", doctype, force=True)
-
- def generate_hash(txt=None):
- """Generates random hash for session id"""
- import hashlib, time
- return hashlib.sha224((txt or "") + str(time.time())).hexdigest()
-
- def reset_metadata_version():
- v = generate_hash()
- cache().set_value("metadata_version", v)
- return v
-
- def get_obj(dt = None, dn = None, doc=None, doclist=None, with_children = True):
- from webnotes.model.code import get_obj
- return get_obj(dt, dn, doc, doclist, with_children)
-
- def doc(doctype=None, name=None, fielddata=None):
- from webnotes.model.doc import Document
- return Document(doctype, name, fielddata)
-
- def new_doc(doctype, parent_doc=None, parentfield=None):
- from webnotes.model.create_new import get_new_doc
- return get_new_doc(doctype, parent_doc, parentfield)
-
- def new_bean(doctype):
- from webnotes.model.create_new import get_new_doc
- return bean([get_new_doc(doctype)])
-
- def doclist(lst=None):
- from webnotes.model.doclist import DocList
- return DocList(lst)
-
- def bean(doctype=None, name=None, copy=None):
- """return an instance of the object, wrapped as a Bean (webnotes.model.bean)"""
- from webnotes.model.bean import Bean
- if copy:
- return Bean(copy_doclist(copy))
- else:
- return Bean(doctype, name)
-
- def set_value(doctype, docname, fieldname, value):
- import webnotes.client
- return webnotes.client.set_value(doctype, docname, fieldname, value)
-
- def get_doclist(doctype, name=None):
- return bean(doctype, name).doclist
-
- def get_doctype(doctype, processed=False):
- import webnotes.model.doctype
- return webnotes.model.doctype.get(doctype, processed)
-
- def delete_doc(doctype=None, name=None, doclist = None, force=0, ignore_doctypes=None,
- for_reload=False, ignore_permissions=False):
- import webnotes.model.delete_doc
-
- if not ignore_doctypes:
- ignore_doctypes = []
-
- if isinstance(name, list):
- for n in name:
- webnotes.model.delete_doc.delete_doc(doctype, n, doclist, force, ignore_doctypes,
- for_reload, ignore_permissions)
- else:
- webnotes.model.delete_doc.delete_doc(doctype, name, doclist, force, ignore_doctypes,
- for_reload, ignore_permissions)
-
- def reload_doc(module, dt=None, dn=None, force=False):
- import webnotes.modules
- return webnotes.modules.reload_doc(module, dt, dn, force=force)
-
- def rename_doc(doctype, old, new, debug=0, force=False, merge=False):
- from webnotes.model.rename_doc import rename_doc
- return rename_doc(doctype, old, new, force=force, merge=merge)
-
- def insert(doclist):
- import webnotes.model
- return webnotes.model.insert(doclist)
-
- def get_module(modulename):
- return importlib.import_module(modulename)
-
- def scrub(txt):
- return txt.replace(' ','_').replace('-', '_').replace('/', '_').lower()
-
- def get_module_path(module, *joins):
- module = scrub(module)
- return get_pymodule_path(local.module_app[module] + "." + module, *joins)
-
- def get_pymodule_path(modulename, *joins):
- joins = [scrub(part) for part in joins]
- return os.path.join(os.path.dirname(get_module(scrub(modulename)).__file__), *joins)
-
- def get_module_list(app_name):
- return get_file_items(os.path.join(os.path.dirname(get_module(app_name).__file__), "modules.txt"))
-
- def get_all_apps(with_webnotes=False):
- apps = get_file_items(os.path.join(local.sites_path, "apps.txt")) \
- + get_file_items(os.path.join(local.site_path, "apps.txt"))
- if with_webnotes:
- apps.insert(0, 'webnotes')
- return apps
-
- def get_installed_apps():
- if flags.in_install_db:
- return []
- installed = json.loads(conn.get_global("installed_apps") or "[]")
- return installed
-
- def get_hooks(hook=None, app_name=None):
- def load_app_hooks(app_name=None):
- hooks = {}
- for app in [app_name] if app_name else get_installed_apps():
- for item in get_file_items(get_pymodule_path(app, "hooks.txt")):
- key, value = item.split("=", 1)
- key, value = key.strip(), value.strip()
- hooks.setdefault(key, [])
- hooks[key].append(value)
- return hooks
- if app_name:
- hooks = _dict(load_app_hooks(app_name))
- else:
- hooks = _dict(cache().get_value("app_hooks", load_app_hooks))
-
- if hook:
- return hooks.get(hook) or []
- else:
- return hooks
-
- def setup_module_map():
- _cache = cache()
-
- if conf.db_name:
- local.app_modules = _cache.get_value("app_modules")
- local.module_app = _cache.get_value("module_app")
-
- if not local.app_modules:
- local.module_app, local.app_modules = {}, {}
- for app in get_all_apps(True):
- local.app_modules.setdefault(app, [])
- for module in get_module_list(app):
- local.module_app[module] = app
- local.app_modules[app].append(module)
-
- if conf.db_name:
- _cache.set_value("app_modules", local.app_modules)
- _cache.set_value("module_app", local.module_app)
-
- def get_file_items(path):
- content = read_file(path)
- if content:
- return [p.strip() for p in content.splitlines() if p.strip() and not p.startswith("#")]
- else:
- return []
-
- def read_file(path):
- if os.path.exists(path):
- with open(path, "r") as f:
- return unicode(f.read(), encoding="utf-8")
- else:
- return None
-
- def get_attr(method_string):
- modulename = '.'.join(method_string.split('.')[:-1])
- methodname = method_string.split('.')[-1]
- return getattr(get_module(modulename), methodname)
-
- def make_property_setter(args):
- args = _dict(args)
- bean([{
- 'doctype': "Property Setter",
- 'doctype_or_field': args.doctype_or_field or "DocField",
- 'doc_type': args.doctype,
- 'field_name': args.fieldname,
- 'property': args.property,
- 'value': args.value,
- 'property_type': args.property_type or "Data",
- '__islocal': 1
- }]).save()
-
- def get_application_home_page(user='Guest'):
- """get home page for user"""
- hpl = conn.sql("""select home_page
- from `tabDefault Home Page`
- where parent='Control Panel'
- and role in ('%s') order by idx asc limit 1""" % "', '".join(get_roles(user)))
- if hpl:
- return hpl[0][0]
- else:
- return conn.get_value("Control Panel", None, "home_page")
-
- def copy_doclist(in_doclist):
- new_doclist = []
- parent_doc = None
- for i, d in enumerate(in_doclist):
- is_dict = False
- if isinstance(d, dict):
- is_dict = True
- values = _dict(d.copy())
- else:
- values = _dict(d.fields.copy())
-
- newd = new_doc(values.doctype, parent_doc=(None if i==0 else parent_doc), parentfield=values.parentfield)
- newd.fields.update(values)
-
- if i==0:
- parent_doc = newd
-
- new_doclist.append(newd.fields if is_dict else newd)
-
- return doclist(new_doclist)
-
- def compare(val1, condition, val2):
- import webnotes.utils
- return webnotes.utils.compare(val1, condition, val2)
-
- def repsond_as_web_page(title, html):
- local.message_title = title
- local.message = "<h3>" + title + "</h3>" + html
- local.response['type'] = 'page'
- local.response['page_name'] = 'message.html'
-
- return obj
-
- def build_match_conditions(doctype, fields=None, as_condition=True):
- import webnotes.widgets.reportview
- return webnotes.widgets.reportview.build_match_conditions(doctype, fields, as_condition)
-
- def get_list(doctype, filters=None, fields=None, docstatus=None,
- group_by=None, order_by=None, limit_start=0, limit_page_length=None,
- as_list=False, debug=False):
- import webnotes.widgets.reportview
- return webnotes.widgets.reportview.execute(doctype, filters=filters, fields=fields, docstatus=docstatus,
- group_by=group_by, order_by=order_by, limit_start=limit_start, limit_page_length=limit_page_length,
- as_list=as_list, debug=debug)
-
- jenv = None
-
- def get_jenv():
- global jenv
- if not jenv:
- from jinja2 import Environment, ChoiceLoader, PackageLoader
-
- apps = get_installed_apps()
- apps.remove("webnotes")
-
- # webnotes will be loaded last, so app templates will get precedence
- jenv = Environment(loader = ChoiceLoader([PackageLoader(app, ".") \
- for app in apps + ["webnotes"]]))
-
- set_filters(jenv)
-
- return jenv
-
- def set_filters(jenv):
- from webnotes.utils import global_date_format
- from markdown2 import markdown
- from json import dumps
-
- jenv.filters["global_date_format"] = global_date_format
- jenv.filters["markdown"] = markdown
- jenv.filters["json"] = dumps
-
- # load jenv_filters from hooks.txt
- for app in get_all_apps(True):
- for jenv_filter in (get_hooks(app_name=app).jenv_filter or []):
- filter_name, filter_function = jenv_filter.split(":")
- jenv.filters[filter_name] = get_attr(filter_function)
-
- def get_template(path):
- return get_jenv().get_template(path)
|