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.
 
 
 
 
 
 

324 line
9.6 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. from webnotes import conf
  5. import webnotes
  6. import json, os, time
  7. from webnotes import _
  8. import webnotes.utils
  9. from webnotes.model.controller import DocListController
  10. import mimetypes
  11. from webnotes.website.doctype.website_sitemap.website_sitemap import add_to_sitemap, update_sitemap, remove_sitemap
  12. class PageNotFoundError(Exception): pass
  13. def render(page_name):
  14. """render html page"""
  15. if not page_name:
  16. page_name = "index"
  17. if "/" in page_name:
  18. page_name = page_name.split("/")[0]
  19. try:
  20. html = render_page(page_name)
  21. except Exception:
  22. html = render_page("error")
  23. webnotes._response.data = html
  24. def render_page(page_name):
  25. """get page html"""
  26. set_content_type(page_name)
  27. if page_name.endswith('.html'):
  28. page_name = page_name[:-5]
  29. html = ''
  30. if not conf.disable_website_cache:
  31. html = webnotes.cache().get_value("page:" + page_name)
  32. from_cache = True
  33. if not html:
  34. html = build_page(page_name)
  35. from_cache = False
  36. if page_name=="error":
  37. html = html.replace("%(error)s", webnotes.get_traceback())
  38. elif "text/html" in webnotes._response.headers["Content-Type"]:
  39. comments = "\npage:"+page_name+\
  40. "\nload status: " + (from_cache and "cache" or "fresh")
  41. html += """\n<!-- %s -->""" % webnotes.utils.cstr(comments)
  42. return html
  43. def set_content_type(page_name):
  44. webnotes._response.headers["Content-Type"] = "text/html; charset: utf-8"
  45. if "." in page_name and not page_name.endswith(".html"):
  46. content_type, encoding = mimetypes.guess_type(page_name)
  47. webnotes._response.headers["Content-Type"] = content_type
  48. def build_page(page_name):
  49. if not webnotes.conn:
  50. webnotes.connect()
  51. if page_name=="index":
  52. page_name = get_home_page()
  53. try:
  54. sitemap_options = webnotes.doc("Website Sitemap", page_name).fields
  55. page_options = webnotes.doc("Website Sitemap Config",
  56. sitemap_options.get("website_sitemap_config")).fields.update({
  57. "page_name":sitemap_options.page_name,
  58. "docname":sitemap_options.docname
  59. })
  60. except webnotes.DoesNotExistError:
  61. hooks = webnotes.get_hooks()
  62. if hooks.website_catch_all:
  63. return build_page(hooks.website_catch_all[0])
  64. else:
  65. return build_page("404")
  66. page_options["page_name"] = page_name
  67. no_cache = page_options.get("no_cache")
  68. # if generator, then load bean, pass arguments
  69. if page_options.get("page_or_generator")=="Generator":
  70. bean = webnotes.bean(page_options.get("ref_doctype"), page_options["docname"])
  71. bean.run_method("get_context")
  72. context = webnotes._dict(bean.doc.fields)
  73. context["obj"] = bean.get_controller()
  74. else:
  75. # page
  76. context = webnotes._dict({ 'name': page_name })
  77. if page_options.get("controller"):
  78. module = webnotes.get_module(page_options.get("controller"))
  79. if module and hasattr(module, "get_context"):
  80. context.update(module.get_context())
  81. context.update(get_website_settings())
  82. jenv = webnotes.get_jenv()
  83. context["base_template"] = jenv.get_template("templates/base.html")
  84. template_name = page_options['template_path']
  85. context["_"] = webnotes._
  86. html = jenv.get_template(template_name).render(context)
  87. if not no_cache:
  88. webnotes.cache().set_value("page:" + page_name, html)
  89. return html
  90. def get_home_page():
  91. return webnotes.cache().get_value("home_page", \
  92. lambda: webnotes.conn.get_value("Website Settings", None, "home_page") or "login")
  93. def get_website_settings():
  94. from webnotes.utils import get_request_site_address, encode, cint
  95. from urllib import quote
  96. hooks = webnotes.get_hooks()
  97. all_top_items = webnotes.conn.sql("""\
  98. select * from `tabTop Bar Item`
  99. where parent='Website Settings' and parentfield='top_bar_items'
  100. order by idx asc""", as_dict=1)
  101. top_items = [d for d in all_top_items if not d['parent_label']]
  102. # attach child items to top bar
  103. for d in all_top_items:
  104. if d['parent_label']:
  105. for t in top_items:
  106. if t['label']==d['parent_label']:
  107. if not 'child_items' in t:
  108. t['child_items'] = []
  109. t['child_items'].append(d)
  110. break
  111. context = webnotes._dict({
  112. 'top_bar_items': top_items,
  113. 'footer_items': webnotes.conn.sql("""\
  114. select * from `tabTop Bar Item`
  115. where parent='Website Settings' and parentfield='footer_items'
  116. order by idx asc""", as_dict=1),
  117. "webnotes": webnotes,
  118. "utils": webnotes.utils,
  119. "post_login": [
  120. {"label": "Reset Password", "url": "update-password", "icon": "icon-key"},
  121. {"label": "Logout", "url": "/?cmd=web_logout", "icon": "icon-signout"}
  122. ]
  123. })
  124. settings = webnotes.doc("Website Settings", "Website Settings")
  125. for k in ["banner_html", "brand_html", "copyright", "twitter_share_via",
  126. "favicon", "facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  127. "disable_signup"]:
  128. if k in settings.fields:
  129. context[k] = settings.fields.get(k)
  130. if settings.address:
  131. context["footer_address"] = settings.address
  132. for k in ["facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  133. "disable_signup"]:
  134. context[k] = cint(context.get(k) or 0)
  135. context.url = quote(str(get_request_site_address(full_address=True)), str(""))
  136. context.encoded_title = quote(encode(context.title or ""), str(""))
  137. for update_website_context in hooks.update_website_context or []:
  138. webnotes.get_attr(update_website_context)(context)
  139. context.web_include_js = hooks.web_include_js or []
  140. context.web_include_css = hooks.web_include_css or []
  141. return context
  142. def clear_cache(page_name=None):
  143. if page_name:
  144. delete_page_cache(page_name)
  145. else:
  146. cache = webnotes.cache()
  147. for p in webnotes.conn.sql_list("""select name from `tabWebsite Sitemap`"""):
  148. if p is not None:
  149. cache.delete_value("page:" + p)
  150. cache.delete_value("home_page")
  151. cache.delete_value("page:index")
  152. cache.delete_value("website_sitemap")
  153. cache.delete_value("website_sitemap_config")
  154. def delete_page_cache(page_name):
  155. if page_name:
  156. cache = webnotes.cache()
  157. cache.delete_value("page:" + page_name)
  158. cache.delete_value("website_sitemap")
  159. def is_signup_enabled():
  160. if getattr(webnotes.local, "is_signup_enabled", None) is None:
  161. webnotes.local.is_signup_enabled = True
  162. if webnotes.utils.cint(webnotes.conn.get_value("Website Settings",
  163. "Website Settings", "disable_signup")):
  164. webnotes.local.is_signup_enabled = False
  165. return webnotes.local.is_signup_enabled
  166. def call_website_generator(bean, method):
  167. getattr(WebsiteGenerator(bean.doc, bean.doclist), method)()
  168. class WebsiteGenerator(DocListController):
  169. def setup_generator(self):
  170. if webnotes.flags.in_install_app:
  171. return
  172. self._website_config = webnotes.conn.get_values("Website Sitemap Config",
  173. {"ref_doctype": self.doc.doctype}, "*")[0]
  174. def on_update(self):
  175. self.update_sitemap()
  176. def after_rename(self, olddn, newdn, merge):
  177. webnotes.conn.sql("""update `tabWebsite Sitemap`
  178. set docname=%s where ref_doctype=%s and docname=%s""", (newdn, self.doc.doctype, olddn))
  179. if merge:
  180. self.setup_generator()
  181. remove_sitemap(ref_doctype=self.doc.doctype, docname=olddn)
  182. def on_trash(self):
  183. self.setup_generator()
  184. remove_sitemap(ref_doctype=self.doc.doctype, docname=self.doc.name)
  185. def update_sitemap(self):
  186. if webnotes.flags.in_install_app:
  187. return
  188. self.setup_generator()
  189. if self._website_config.condition_field and \
  190. not self.doc.fields.get(self._website_config.condition_field):
  191. # condition field failed, remove and return!
  192. remove_sitemap(ref_doctype=self.doc.doctype, docname=self.doc.name)
  193. return
  194. self.add_or_update_sitemap()
  195. def add_or_update_sitemap(self):
  196. page_name = self.get_page_name()
  197. existing_page_name = webnotes.conn.get_value("Website Sitemap", {"ref_doctype": self.doc.doctype,
  198. "docname": self.doc.name})
  199. opts = webnotes._dict({
  200. "page_or_generator": "Generator",
  201. "ref_doctype":self.doc.doctype,
  202. "docname": self.doc.name,
  203. "page_name": page_name,
  204. "link_name": self._website_config.name,
  205. "lastmod": webnotes.utils.get_datetime(self.doc.modified).strftime("%Y-%m-%d"),
  206. "parent_website_sitemap": self.doc.parent_website_sitemap
  207. })
  208. if self.meta.get_field("public_read"):
  209. opts.public_read = self.doc.public_read
  210. opts.public_write = self.doc.public_write
  211. else:
  212. opts.public_read = 1
  213. if existing_page_name:
  214. if existing_page_name != page_name:
  215. webnotes.rename_doc("Website Sitemap", existing_page_name, page_name, ignore_permissions=True)
  216. update_sitemap(page_name, opts)
  217. else:
  218. add_to_sitemap(opts)
  219. def get_page_name(self):
  220. if not self.doc.fields.get(self._website_config.page_name_field):
  221. new_page_name = cleanup_page_name(self.get_page_title() \
  222. if hasattr(self, "get_page_title") else (self.doc.title or self.doc.name))
  223. webnotes.conn.set(self.doc, self._website_config.page_name_field, new_page_name)
  224. return self.doc.fields.get(self._website_config.page_name_field)
  225. def cleanup_page_name(title):
  226. """make page name from title"""
  227. import re
  228. name = title.lower()
  229. name = re.sub('[~!@#$%^&*+()<>,."\'\?]', '', name)
  230. name = re.sub('[:/]', '-', name)
  231. name = '-'.join(name.split())
  232. # replace repeating hyphens
  233. name = re.sub(r"(-)\1+", r"\1", name)
  234. return name
  235. def get_hex_shade(color, percent):
  236. def p(c):
  237. v = int(c, 16) + int(int('ff', 16) * (float(percent)/100))
  238. if v < 0:
  239. v=0
  240. if v > 255:
  241. v=255
  242. h = hex(v)[2:]
  243. if len(h) < 2:
  244. h = "0" + h
  245. return h
  246. r, g, b = color[0:2], color[2:4], color[4:6]
  247. avg = (float(int(r, 16) + int(g, 16) + int(b, 16)) / 3)
  248. # switch dark and light shades
  249. if avg > 128:
  250. percent = -percent
  251. # stronger diff for darker shades
  252. if percent < 25 and avg < 64:
  253. percent = percent * 2
  254. return p(r) + p(g) + p(b)