Non puoi selezionare più di 25 argomenti Gli argomenti devono iniziare con una lettera o un numero, possono includere trattini ('-') e possono essere lunghi fino a 35 caratteri.
 
 
 
 
 
 

430 righe
12 KiB

  1. # Copyright (c) 2013, Web Notes Technologies Pvt. Ltd. and Contributors
  2. # MIT License. See license.txt
  3. from __future__ import unicode_literals
  4. import webnotes
  5. import json, os, time
  6. from webnotes import _
  7. import webnotes.utils
  8. from webnotes.utils import get_request_site_address, encode, cint
  9. from webnotes.model import default_fields
  10. from webnotes.model.controller import DocListController
  11. from urllib import quote
  12. import mimetypes
  13. from webnotes.website.doctype.website_sitemap.website_sitemap import add_to_sitemap, update_sitemap, remove_sitemap
  14. from webnotes.website.doctype.website_sitemap_permission.website_sitemap_permission \
  15. import get_access
  16. class PageNotFoundError(Exception): pass
  17. def render(page_name):
  18. """render html page"""
  19. page_name = scrub_page_name(page_name)
  20. try:
  21. data = render_page(page_name)
  22. except Exception:
  23. page_name = "error"
  24. data = render_page(page_name)
  25. data = insert_traceback(data)
  26. data = set_content_type(data, page_name)
  27. webnotes._response.data = data
  28. webnotes._response.headers["Page Name"] = page_name
  29. def render_page(page_name):
  30. """get page html"""
  31. cache_key = ("page_context:{}" if is_ajax() else "page:{}").format(page_name)
  32. out = None
  33. # try memcache
  34. if can_cache():
  35. out = webnotes.cache().get_value(cache_key)
  36. if is_ajax():
  37. out = out.get("data")
  38. if out:
  39. webnotes._response.headers["From Cache"] = True
  40. return out
  41. return build(page_name)
  42. def build(page_name):
  43. if not webnotes.conn:
  44. webnotes.connect()
  45. build_method = (build_json if is_ajax() else build_page)
  46. try:
  47. return build_method(page_name)
  48. except webnotes.DoesNotExistError:
  49. hooks = webnotes.get_hooks()
  50. if hooks.website_catch_all:
  51. return build_method(hooks.website_catch_all[0])
  52. else:
  53. return build_method("404")
  54. def build_json(page_name):
  55. return get_context(page_name).data
  56. def build_page(page_name):
  57. context = get_context(page_name)
  58. html = webnotes.get_template(context.base_template_path).render(context)
  59. if can_cache(context.no_cache):
  60. webnotes.cache().set_value("page:" + page_name, html)
  61. return html
  62. def get_context(page_name):
  63. context = None
  64. cache_key = "page_context:{}".format(page_name)
  65. # try from memcache
  66. if can_cache():
  67. context = webnotes.cache().get_value(cache_key)
  68. if not context:
  69. sitemap_options = get_sitemap_options(page_name)
  70. context = build_context(sitemap_options)
  71. if can_cache(context.no_cache):
  72. webnotes.cache().set_value(cache_key, context)
  73. context.update(context.data or {})
  74. return context
  75. def get_sitemap_options(page_name):
  76. sitemap_options = None
  77. cache_key = "sitemap_options:{}".format(page_name)
  78. if can_cache():
  79. sitemap_options = webnotes.cache().get_value(cache_key)
  80. if sitemap_options:
  81. return sitemap_options
  82. sitemap_options = build_sitemap_options(page_name)
  83. if can_cache(sitemap_options.no_cache):
  84. webnotes.cache().set_value(cache_key, sitemap_options)
  85. return sitemap_options
  86. def build_sitemap_options(page_name):
  87. sitemap_options = webnotes.doc("Website Sitemap", page_name).fields
  88. # only non default fields
  89. for fieldname in default_fields:
  90. if fieldname in sitemap_options:
  91. del sitemap_options[fieldname]
  92. sitemap_config = webnotes.doc("Website Sitemap Config",
  93. sitemap_options.get("website_sitemap_config")).fields
  94. # get sitemap config fields too
  95. for fieldname in ("base_template_path", "template_path", "controller", "no_cache", "no_sitemap",
  96. "page_name_field", "condition_field"):
  97. sitemap_options[fieldname] = sitemap_config.get(fieldname)
  98. # establish hierarchy
  99. sitemap_options.parents = webnotes.conn.sql("""select name, page_title from `tabWebsite Sitemap`
  100. where lft < %s and rgt > %s order by lft asc""", (sitemap_options.lft, sitemap_options.rgt), as_dict=True)
  101. sitemap_options.children = webnotes.conn.sql("""select * from `tabWebsite Sitemap`
  102. where parent_website_sitemap=%s""", (sitemap_options.page_name,), as_dict=True)
  103. # determine templates to be used
  104. if not sitemap_options.base_template_path:
  105. sitemap_options.base_template_path = "templates/base.html"
  106. sitemap_options.template = webnotes.get_jenv().get_template(sitemap_options.template_path)
  107. return sitemap_options
  108. def build_context(sitemap_options):
  109. """get_context method of bean or module is supposed to render content templates and push it into context"""
  110. context = webnotes._dict({ "_": webnotes._ })
  111. context.update(sitemap_options)
  112. context.update(get_website_settings())
  113. if sitemap_options.get("controller"):
  114. module = webnotes.get_module(sitemap_options.get("controller"))
  115. if module and hasattr(module, "get_context"):
  116. context.data = module.get_context(context) or {}
  117. return context
  118. def can_cache(no_cache=False):
  119. return not (webnotes.conf.disable_website_cache or no_cache)
  120. def get_home_page():
  121. return webnotes.cache().get_value("home_page", \
  122. lambda: webnotes.conn.get_value("Website Settings", None, "home_page") or "login")
  123. def get_website_settings():
  124. # TODO Cache this
  125. hooks = webnotes.get_hooks()
  126. all_top_items = webnotes.conn.sql("""\
  127. select * from `tabTop Bar Item`
  128. where parent='Website Settings' and parentfield='top_bar_items'
  129. order by idx asc""", as_dict=1)
  130. top_items = [d for d in all_top_items if not d['parent_label']]
  131. # attach child items to top bar
  132. for d in all_top_items:
  133. if d['parent_label']:
  134. for t in top_items:
  135. if t['label']==d['parent_label']:
  136. if not 'child_items' in t:
  137. t['child_items'] = []
  138. t['child_items'].append(d)
  139. break
  140. context = webnotes._dict({
  141. 'top_bar_items': top_items,
  142. 'footer_items': webnotes.conn.sql("""\
  143. select * from `tabTop Bar Item`
  144. where parent='Website Settings' and parentfield='footer_items'
  145. order by idx asc""", as_dict=1),
  146. "webnotes": webnotes,
  147. "utils": webnotes.utils,
  148. "post_login": [
  149. {"label": "Reset Password", "url": "update-password", "icon": "icon-key"},
  150. {"label": "Logout", "url": "?cmd=web_logout", "icon": "icon-signout"}
  151. ]
  152. })
  153. settings = webnotes.doc("Website Settings", "Website Settings")
  154. for k in ["banner_html", "brand_html", "copyright", "twitter_share_via",
  155. "favicon", "facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  156. "disable_signup"]:
  157. if k in settings.fields:
  158. context[k] = settings.fields.get(k)
  159. if settings.address:
  160. context["footer_address"] = settings.address
  161. for k in ["facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  162. "disable_signup"]:
  163. context[k] = cint(context.get(k) or 0)
  164. context.url = quote(str(get_request_site_address(full_address=True)), safe="/:")
  165. context.encoded_title = quote(encode(context.title or ""), str(""))
  166. for update_website_context in hooks.update_website_context or []:
  167. webnotes.get_attr(update_website_context)(context)
  168. context.web_include_js = hooks.web_include_js or []
  169. context.web_include_css = hooks.web_include_css or []
  170. # get settings from site config
  171. if webnotes.conf.get("fb_app_id"):
  172. context.fb_app_id = webnotes.conf.fb_app_id
  173. return context
  174. def is_ajax():
  175. return webnotes.get_request_header("X-Requested-With")=="XMLHttpRequest"
  176. def scrub_page_name(page_name):
  177. if not page_name:
  178. page_name = "index"
  179. if "/" in page_name:
  180. page_name = page_name.split("/")[0]
  181. if page_name.endswith('.html'):
  182. page_name = page_name[:-5]
  183. if page_name == "index":
  184. page_name = get_home_page()
  185. return page_name
  186. def insert_traceback(data):
  187. traceback = webnotes.get_traceback()
  188. if isinstance(data, dict):
  189. data["error"] = traceback
  190. else:
  191. data = data % {"error": traceback}
  192. return data
  193. def set_content_type(data, page_name):
  194. if isinstance(data, dict):
  195. webnotes._response.headers["Content-Type"] = "application/json; charset: utf-8"
  196. data = json.dumps(data)
  197. return data
  198. webnotes._response.headers["Content-Type"] = "text/html; charset: utf-8"
  199. if "." in page_name and not page_name.endswith(".html"):
  200. content_type, encoding = mimetypes.guess_type(page_name)
  201. webnotes._response.headers["Content-Type"] = content_type
  202. return data
  203. def clear_cache(page_name=None):
  204. if page_name:
  205. delete_page_cache(page_name)
  206. else:
  207. cache = webnotes.cache()
  208. for p in webnotes.conn.sql_list("""select name from `tabWebsite Sitemap`"""):
  209. if p is not None:
  210. cache.delete_value("page:" + p)
  211. cache.delete_value("home_page")
  212. cache.delete_value("page:index")
  213. cache.delete_value("website_sitemap")
  214. cache.delete_value("website_sitemap_config")
  215. def delete_page_cache(page_name):
  216. if page_name:
  217. cache = webnotes.cache()
  218. cache.delete_value("page:" + page_name)
  219. cache.delete_value("website_sitemap")
  220. def is_signup_enabled():
  221. if getattr(webnotes.local, "is_signup_enabled", None) is None:
  222. webnotes.local.is_signup_enabled = True
  223. if webnotes.utils.cint(webnotes.conn.get_value("Website Settings",
  224. "Website Settings", "disable_signup")):
  225. webnotes.local.is_signup_enabled = False
  226. return webnotes.local.is_signup_enabled
  227. def call_website_generator(bean, method):
  228. getattr(WebsiteGenerator(bean.doc, bean.doclist), method)()
  229. class WebsiteGenerator(DocListController):
  230. def setup_generator(self):
  231. if webnotes.flags.in_install_app:
  232. return
  233. self._website_config = webnotes.conn.get_values("Website Sitemap Config",
  234. {"ref_doctype": self.doc.doctype}, "*")[0]
  235. def on_update(self):
  236. self.update_sitemap()
  237. def after_rename(self, olddn, newdn, merge):
  238. webnotes.conn.sql("""update `tabWebsite Sitemap`
  239. set docname=%s where ref_doctype=%s and docname=%s""", (newdn, self.doc.doctype, olddn))
  240. if merge:
  241. self.setup_generator()
  242. remove_sitemap(ref_doctype=self.doc.doctype, docname=olddn)
  243. def on_trash(self):
  244. self.setup_generator()
  245. remove_sitemap(ref_doctype=self.doc.doctype, docname=self.doc.name)
  246. def update_sitemap(self):
  247. if webnotes.flags.in_install_app:
  248. return
  249. self.setup_generator()
  250. if self._website_config.condition_field and \
  251. not self.doc.fields.get(self._website_config.condition_field):
  252. # condition field failed, remove and return!
  253. remove_sitemap(ref_doctype=self.doc.doctype, docname=self.doc.name)
  254. return
  255. self.add_or_update_sitemap()
  256. def add_or_update_sitemap(self):
  257. page_name = self.get_page_name()
  258. existing_page_name = webnotes.conn.get_value("Website Sitemap", {"ref_doctype": self.doc.doctype,
  259. "docname": self.doc.name})
  260. opts = webnotes._dict({
  261. "page_or_generator": "Generator",
  262. "ref_doctype":self.doc.doctype,
  263. "docname": self.doc.name,
  264. "page_name": page_name,
  265. "link_name": self._website_config.name,
  266. "lastmod": webnotes.utils.get_datetime(self.doc.modified).strftime("%Y-%m-%d"),
  267. "parent_website_sitemap": self.doc.parent_website_sitemap,
  268. "page_title": self.get_page_title() \
  269. if hasattr(self, "get_page_title") else (self.doc.title or self.doc.name)
  270. })
  271. if self.meta.get_field("public_read"):
  272. opts.public_read = self.doc.public_read
  273. opts.public_write = self.doc.public_write
  274. else:
  275. opts.public_read = 1
  276. if existing_page_name:
  277. if existing_page_name != page_name:
  278. webnotes.rename_doc("Website Sitemap", existing_page_name, page_name, ignore_permissions=True)
  279. update_sitemap(page_name, opts)
  280. else:
  281. add_to_sitemap(opts)
  282. def get_page_name(self):
  283. if not self.doc.fields.get(self._website_config.page_name_field):
  284. new_page_name = cleanup_page_name(self.get_page_title() \
  285. if hasattr(self, "get_page_title") else (self.doc.title or self.doc.name))
  286. webnotes.conn.set(self.doc, self._website_config.page_name_field, new_page_name)
  287. return self.doc.fields.get(self._website_config.page_name_field)
  288. def cleanup_page_name(title):
  289. """make page name from title"""
  290. import re
  291. name = title.lower()
  292. name = re.sub('[~!@#$%^&*+()<>,."\'\?]', '', name)
  293. name = re.sub('[:/]', '-', name)
  294. name = '-'.join(name.split())
  295. # replace repeating hyphens
  296. name = re.sub(r"(-)\1+", r"\1", name)
  297. return name
  298. def get_hex_shade(color, percent):
  299. def p(c):
  300. v = int(c, 16) + int(int('ff', 16) * (float(percent)/100))
  301. if v < 0:
  302. v=0
  303. if v > 255:
  304. v=255
  305. h = hex(v)[2:]
  306. if len(h) < 2:
  307. h = "0" + h
  308. return h
  309. r, g, b = color[0:2], color[2:4], color[4:6]
  310. avg = (float(int(r, 16) + int(g, 16) + int(b, 16)) / 3)
  311. # switch dark and light shades
  312. if avg > 128:
  313. percent = -percent
  314. # stronger diff for darker shades
  315. if percent < 25 and avg < 64:
  316. percent = percent * 2
  317. return p(r) + p(g) + p(b)
  318. def render_blocks(context):
  319. """returns a dict of block name and its rendered content"""
  320. from jinja2.utils import concat
  321. out = {}
  322. template = context["template"]
  323. # required as per low level API
  324. context = template.new_context(context)
  325. # render each block individually
  326. for block, render in template.blocks.items():
  327. out[block] = concat(render(context))
  328. return out