Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.
 
 
 
 
 
 

361 rinda
10 KiB

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