You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

webutils.py 10 KiB

12 years ago
11 years ago
12 years ago
12 years ago
12 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365
  1. # Copyright (c) 2013, Web Notes Technologies Pvt. Ltd.
  2. # MIT License. See license.txt
  3. from __future__ import unicode_literals
  4. from webnotes import conf
  5. import webnotes
  6. import json
  7. from webnotes import _
  8. import webnotes.utils
  9. import mimetypes
  10. class PageNotFoundError(Exception): pass
  11. def render(page_name):
  12. """render html page"""
  13. try:
  14. html = render_page(page_name or "index")
  15. except PageNotFoundError:
  16. html = render_page("404")
  17. except Exception:
  18. html = render_page("error")
  19. webnotes._response.data = html
  20. def render_page(page_name):
  21. """get page html"""
  22. set_content_type(page_name)
  23. page_name = scrub_page_name(page_name)
  24. html = ''
  25. if not conf.auto_cache_clear:
  26. html = webnotes.cache().get_value("page:" + page_name)
  27. from_cache = True
  28. if not html:
  29. html = build_page(page_name)
  30. from_cache = False
  31. if not html:
  32. raise PageNotFoundError
  33. if page_name=="error":
  34. html = html.replace("%(error)s", webnotes.getTraceback())
  35. elif "text/html" in webnotes._response.headers["Content-Type"]:
  36. comments = "\npage:"+page_name+\
  37. "\nload status: " + (from_cache and "cache" or "fresh")
  38. html += """\n<!-- %s -->""" % webnotes.utils.cstr(comments)
  39. return html
  40. def set_content_type(page_name):
  41. webnotes._response.headers["Content-Type"] = "text/html; charset: utf-8"
  42. if "." in page_name and not page_name.endswith(".html"):
  43. content_type, encoding = mimetypes.guess_type(page_name)
  44. webnotes._response.headers["Content-Type"] = content_type
  45. def build_page(page_name):
  46. if not webnotes.conn:
  47. webnotes.connect()
  48. sitemap = get_website_sitemap()
  49. page_options = sitemap.get(page_name)
  50. if not page_options:
  51. if page_name=="index":
  52. # page not found, try home page
  53. home_page = get_home_page()
  54. page_options = sitemap.get(home_page)
  55. if not page_options:
  56. raise PageNotFoundError
  57. page_options["page_name"] = home_page
  58. else:
  59. raise PageNotFoundError
  60. else:
  61. page_options["page_name"] = page_name
  62. basepath = webnotes.utils.get_base_path()
  63. module = None
  64. no_cache = False
  65. if page_options.get("controller"):
  66. module = webnotes.get_module(page_options["controller"])
  67. no_cache = getattr(module, "no_cache", False)
  68. # if generator, then load bean, pass arguments
  69. if page_options.get("is_generator"):
  70. if not module:
  71. raise Exception("Generator controller not defined")
  72. name = webnotes.conn.get_value(module.doctype, {
  73. page_options.get("page_name_field", "page_name"): page_options["page_name"]})
  74. obj = webnotes.get_obj(module.doctype, name, with_children=True)
  75. if hasattr(obj, 'get_context'):
  76. obj.get_context()
  77. context = webnotes._dict(obj.doc.fields)
  78. context["obj"] = obj
  79. else:
  80. # page
  81. context = webnotes._dict({ 'name': page_name })
  82. if module and hasattr(module, "get_context"):
  83. context.update(module.get_context())
  84. context.update(get_website_settings())
  85. jenv = webnotes.get_jenv()
  86. context["base_template"] = jenv.get_template(webnotes.get_config().get("base_template"))
  87. template_name = page_options['template']
  88. html = jenv.get_template(template_name).render(context)
  89. if not no_cache:
  90. webnotes.cache().set_value("page:" + page_name, html)
  91. return html
  92. def build_sitemap():
  93. sitemap = {}
  94. config = webnotes.cache().get_value("website_sitemap_config", build_website_sitemap_config)
  95. sitemap.update(config["pages"])
  96. # pages
  97. for p in config["pages"].values():
  98. if p.get("controller"):
  99. module = webnotes.get_module(p["controller"])
  100. p["no_cache"] = getattr(module, "no_cache", False)
  101. p["no_sitemap"] = getattr(module, "no_sitemap", False) or p["no_cache"]
  102. # generators
  103. for g in config["generators"].values():
  104. g["is_generator"] = True
  105. module = webnotes.get_module(g["controller"])
  106. condition = ""
  107. page_name_field = "page_name"
  108. if hasattr(module, "page_name_field"):
  109. page_name_field = module.page_name_field
  110. if hasattr(module, "condition_field"):
  111. condition = " where ifnull(%s, 0)=1" % module.condition_field
  112. for page_name, name, modified in webnotes.conn.sql("""select %s, name, modified from
  113. `tab%s` %s""" % (page_name_field, module.doctype, condition)):
  114. opts = g.copy()
  115. opts["doctype"] = module.doctype
  116. opts["no_cache"] = getattr(module, "no_cache", False)
  117. opts["page_name"] = page_name
  118. if page_name_field != "page_name":
  119. opts["page_name_field"] = page_name_field
  120. opts["docname"] = name
  121. opts["lastmod"] = modified.strftime("%Y-%m-%d %H:%M:%S")
  122. sitemap[page_name] = opts
  123. return sitemap
  124. def get_home_page():
  125. if not webnotes.conn:
  126. webnotes.connect()
  127. doc_name = webnotes.conn.get_value('Website Settings', None, 'home_page')
  128. if doc_name:
  129. page_name = webnotes.conn.get_value('Web Page', doc_name, 'page_name')
  130. else:
  131. page_name = 'login'
  132. return page_name
  133. def build_website_sitemap_config():
  134. import os, time
  135. config = {"pages": {}, "generators":{}}
  136. basepath = webnotes.utils.get_base_path()
  137. def get_options(path, fname):
  138. name = fname
  139. if fname.endswith(".html"):
  140. name = fname[:-5]
  141. template_path = os.path.relpath(os.path.join(path, fname), basepath)
  142. options = webnotes._dict({
  143. "link_name": name,
  144. "template": template_path,
  145. "lastmod": time.ctime(os.path.getmtime(template_path))
  146. })
  147. controller_name = fname.split(".")[0].replace("-", "_") + ".py"
  148. controller_path = os.path.join(path, controller_name)
  149. if os.path.exists(controller_path):
  150. options.controller = os.path.relpath(controller_path[:-3], basepath).replace(os.path.sep, ".")
  151. options.controller = ".".join(options.controller.split(".")[1:])
  152. return options
  153. for path, folders, files in os.walk(basepath, followlinks=True):
  154. if 'locale' in folders: folders.remove('locale')
  155. if os.path.basename(path)=="pages" and os.path.basename(os.path.dirname(path))=="templates":
  156. for fname in files:
  157. fname = webnotes.utils.cstr(fname)
  158. if fname.split(".")[-1] in ("html", "xml", "js", "css"):
  159. options = get_options(path, fname)
  160. config["pages"][options.link_name] = options
  161. if os.path.basename(path)=="generators" and os.path.basename(os.path.dirname(path))=="templates":
  162. for fname in files:
  163. if fname.endswith(".html"):
  164. options = get_options(path, fname)
  165. config["generators"][fname] = options
  166. return config
  167. def get_website_settings():
  168. from webnotes.utils import get_request_site_address, encode, cint
  169. from urllib import quote
  170. all_top_items = webnotes.conn.sql("""\
  171. select * from `tabTop Bar Item`
  172. where parent='Website Settings' and parentfield='top_bar_items'
  173. order by idx asc""", as_dict=1)
  174. top_items = [d for d in all_top_items if not d['parent_label']]
  175. # attach child items to top bar
  176. for d in all_top_items:
  177. if d['parent_label']:
  178. for t in top_items:
  179. if t['label']==d['parent_label']:
  180. if not 'child_items' in t:
  181. t['child_items'] = []
  182. t['child_items'].append(d)
  183. break
  184. context = webnotes._dict({
  185. 'top_bar_items': top_items,
  186. 'footer_items': webnotes.conn.sql("""\
  187. select * from `tabTop Bar Item`
  188. where parent='Website Settings' and parentfield='footer_items'
  189. order by idx asc""", as_dict=1),
  190. "webnotes": webnotes,
  191. "utils": webnotes.utils,
  192. "post_login": [
  193. {"label": "Reset Password", "url": "update-password", "icon": "icon-key"},
  194. {"label": "Logout", "url": "/?cmd=web_logout", "icon": "icon-signout"}
  195. ]
  196. })
  197. settings = webnotes.doc("Website Settings", "Website Settings")
  198. for k in ["banner_html", "brand_html", "copyright", "twitter_share_via",
  199. "favicon", "facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  200. "disable_signup"]:
  201. if k in settings.fields:
  202. context[k] = settings.fields.get(k)
  203. if settings.address:
  204. context["footer_address"] = settings.address
  205. for k in ["facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  206. "disable_signup"]:
  207. context[k] = cint(context.get(k) or 0)
  208. context.url = quote(str(get_request_site_address(full_address=True)), str(""))
  209. context.encoded_title = quote(encode(context.title or ""), str(""))
  210. try:
  211. import startup.webutils
  212. if hasattr(startup.webutils, "get_website_settings"):
  213. startup.webutils.get_website_settings(context)
  214. except:
  215. pass
  216. return context
  217. def clear_cache(page_name=None):
  218. if page_name:
  219. delete_page_cache(page_name)
  220. else:
  221. cache = webnotes.cache()
  222. for p in get_all_pages():
  223. if p is not None:
  224. cache.delete_value("page:" + p)
  225. cache.delete_value("page:index")
  226. cache.delete_value("website_sitemap")
  227. cache.delete_value("website_sitemap_config")
  228. def get_website_sitemap():
  229. return webnotes.cache().get_value("website_sitemap", build_sitemap)
  230. def get_all_pages():
  231. return get_website_sitemap().keys()
  232. def delete_page_cache(page_name):
  233. if page_name:
  234. cache = webnotes.cache()
  235. cache.delete_value("page:" + page_name)
  236. cache.delete_value("website_sitemap")
  237. def get_hex_shade(color, percent):
  238. def p(c):
  239. v = int(c, 16) + int(int('ff', 16) * (float(percent)/100))
  240. if v < 0:
  241. v=0
  242. if v > 255:
  243. v=255
  244. h = hex(v)[2:]
  245. if len(h) < 2:
  246. h = "0" + h
  247. return h
  248. r, g, b = color[0:2], color[2:4], color[4:6]
  249. avg = (float(int(r, 16) + int(g, 16) + int(b, 16)) / 3)
  250. # switch dark and light shades
  251. if avg > 128:
  252. percent = -percent
  253. # stronger diff for darker shades
  254. if percent < 25 and avg < 64:
  255. percent = percent * 2
  256. return p(r) + p(g) + p(b)
  257. def scrub_page_name(page_name):
  258. if page_name.endswith('.html'):
  259. page_name = page_name[:-5]
  260. return page_name
  261. def is_signup_enabled():
  262. if getattr(webnotes.local, "is_signup_enabled", None) is None:
  263. webnotes.local.is_signup_enabled = True
  264. if webnotes.utils.cint(webnotes.conn.get_value("Website Settings",
  265. "Website Settings", "disable_signup")):
  266. webnotes.local.is_signup_enabled = False
  267. return webnotes.local.is_signup_enabled
  268. def update_page_name(doc, title):
  269. """set page_name and check if it is unique"""
  270. new_page_name = page_name(title)
  271. sitemap = get_website_sitemap()
  272. if new_page_name in sitemap and \
  273. not (sitemap[new_page_name].doctype == doc.doctype and sitemap[new_page_name].docname == doc.name):
  274. webnotes.throw("%s: %s. %s: %s" % (new_page_name, _("Page already exists"),
  275. _("Please change the value"), title))
  276. if doc.page_name: delete_page_cache(doc.page_name)
  277. webnotes.conn.set(doc, "page_name", new_page_name)
  278. delete_page_cache(doc.page_name)
  279. def page_name(title):
  280. """make page name from title"""
  281. import re
  282. name = title.lower()
  283. name = re.sub('[~!@#$%^&*+()<>,."\']', '', name)
  284. name = re.sub('[:/]', '-', name)
  285. name = '-'.join(name.split())
  286. # replace repeating hyphens
  287. name = re.sub(r"(-)\1+", r"\1", name)
  288. return name