web.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. # -*- coding: utf-8 -*-
  2. """
  3. demeter
  4. name:web.py
  5. author:rabin
  6. """
  7. #from gevent import monkey
  8. #monkey.patch_all()
  9. #import gevent
  10. import functools
  11. import os
  12. import json
  13. import threading
  14. from demeter.core import *
  15. import tornado.web
  16. import tornado.ioloop
  17. import tornado.httpserver
  18. import tornado.httpclient
  19. import tornado.concurrent
  20. class Base(tornado.web.RequestHandler):
  21. def set_default_headers(self):
  22. self.set_header("Access-Control-Allow-Origin", "*")
  23. self.set_header("Access-Control-Allow-Headers", "x-requested-with")
  24. self.set_header('Access-Control-Allow-Methods', 'POST, GET, OPTIONS')
  25. def initialize(self):
  26. try:
  27. Demeter.request = self
  28. self.assign()
  29. self.page()
  30. self.cookie()
  31. self.setting()
  32. except Exception as e:
  33. return
  34. def get_current_user(self):
  35. return self.get_secure_cookie(self.KEYS[0])
  36. def assign(self):
  37. self.data = {}
  38. self.data['setting'] = Demeter.config['setting']
  39. self.data['base'] = Demeter.config['base']
  40. def cookie(self):
  41. for key in self.KEYS:
  42. cookie = None
  43. """
  44. if key in self.data['base']:
  45. cookie = self.data['base'][key]
  46. """
  47. if not cookie:
  48. cookie = self.get_secure_cookie(key)
  49. #cookie = self.get_cookie(key)
  50. if not cookie:
  51. value = self.input(key)
  52. if value:
  53. #self.set_secure_cookie(key, value)
  54. self.data['setting'][key] = value
  55. else:
  56. self.data['setting'][key] = 0
  57. else:
  58. self.data['setting'][key] = cookie
  59. def page(self):
  60. Demeter.config['page'] = {}
  61. page = self.input('page')
  62. if page:
  63. Demeter.config['page']['ajax'] = True
  64. else:
  65. Demeter.config['page']['ajax'] = False
  66. page = 1
  67. Demeter.config['page']['current'] = page
  68. Demeter.config['page']['total'] = 0
  69. self.data['page'] = Demeter.config['page']
  70. def search(self):
  71. data = self.request.arguments
  72. self.data['search'] = {}
  73. self.data['update'] = {}
  74. for key in data:
  75. if 'search_' in key:
  76. index = key.replace('search_', '')
  77. self.data['search'][index] = ",".join(data[key])
  78. if 'update_' in key:
  79. index = key.replace('update_', '')
  80. self.data['update'][index] = ",".join(data[key])
  81. def input(self, key, value=None):
  82. return self.get_argument(key, value)
  83. def inputs(self, key):
  84. return self.get_arguments(key)
  85. def service(self, name):
  86. return Demeter.service(name)
  87. def model(self, name):
  88. return Demeter.model(name)
  89. def set(self, **kwd):
  90. self.data['common'] = kwd
  91. self.data['common']['argvs'] = ''
  92. def show(self, name):
  93. self.view('common/'+name+'.html')
  94. def list(self, model, order = 'cdate desc'):
  95. self.data['state'] = self.input('state', True)
  96. self.data['list'] = self.service('common').list(model, state=self.data['state'], search=self.data['search'], page=True, order=order)
  97. def one(self, model, **kwd):
  98. id = self.input('id')
  99. self.data['info'] = {}
  100. if id:
  101. kwd['id'] = id
  102. if kwd:
  103. self.data['info'] = self.service('common').one(model, **kwd)
  104. def update(self, model, msg='', id=0, **kwd):
  105. if not self.data['auth']:
  106. self.auth()
  107. else:
  108. if id <= 0:
  109. id = self.input('id')
  110. if kwd:
  111. info = self.service('common').one(model, **kwd)
  112. if info and (id != info['id']):
  113. self.out(msg)
  114. return
  115. state = self.service('common').update(model, id, self.data['update'])
  116. self.log(model, 'update', self.data['update'])
  117. self.out('yes', {'id':state})
  118. return state
  119. def drop(self, model):
  120. if not self.data['auth']:
  121. self.auth()
  122. else:
  123. id = self.input('id')
  124. state = self.input('state', False)
  125. state = self.service('common').delete(model, id, state)
  126. self.log(model, 'delete', {id:id, state:state})
  127. self.out('yes', {'state':state})
  128. def log(self, model, method, data):
  129. if 'admin' in self.data['setting'] and self.data['setting']['admin'] > 0:
  130. insert = {}
  131. insert['admin_id'] = self.data['setting']['admin']
  132. insert['model'] = model
  133. insert['method'] = method
  134. insert['data'] = json.dumps(data)
  135. self.service('common').update('manage_log', None, insert)
  136. def view(self, name):
  137. if not self.data['auth']:
  138. self.auth()
  139. else:
  140. config = Demeter.config[Demeter.web]
  141. path = ''
  142. if 'mobile' in config:
  143. mobile = Demeter.checkMobile(self.request)
  144. if mobile:
  145. path = 'mobile/'
  146. else:
  147. path = 'pc/'
  148. self.render(path + name, data=self.data, Demeter=Demeter)
  149. def auth(self):
  150. self.out('您没有权限')
  151. def out(self, msg='', data={}, code=0):
  152. callback = self.input('callback')
  153. function = self.input('function')
  154. result = Demeter.out(msg=msg, data=data, code=code, callback=callback, function=function)
  155. self.write(result)
  156. if not data:
  157. from tornado.web import Finish
  158. raise Finish()
  159. else:
  160. self.finish()
  161. class Web(object):
  162. @classmethod
  163. def auth(self, method):
  164. return tornado.web.authenticated(method)
  165. @classmethod
  166. def setting(self, method):
  167. return self.async(method)
  168. @staticmethod
  169. def async(method):
  170. @tornado.web.asynchronous
  171. @tornado.gen.coroutine
  172. @functools.wraps(method)
  173. def callback(self, *args, **kwargs):
  174. #self._auto_finish = False
  175. try:
  176. result = method(self, *args, **kwargs)
  177. return result
  178. except Exception as e:
  179. import traceback
  180. error = traceback.format_exc()
  181. if 'Finish' in error:
  182. return
  183. else:
  184. traceback.print_exc()
  185. try:
  186. return self.view('404.html')
  187. except Exception as e:
  188. return self.out('404')
  189. #return gevent.spawn(method, self, *args, **kwargs)
  190. return callback
  191. @classmethod
  192. def init(self, application):
  193. for v in application:
  194. self.load(v)
  195. @classmethod
  196. def load(self, package):
  197. """
  198. path = os.path.split(os.path.realpath(file))[0] + '/'
  199. sys.path.append(path)
  200. files = self.file(path)
  201. """
  202. url = []
  203. for key in Demeter.getPackage(package):
  204. module = Demeter.getObject(key)
  205. url = self.url(module, key, url)
  206. Demeter.route = Demeter.route + url
  207. """
  208. @staticmethod
  209. def file(path):
  210. files = os.listdir(path)
  211. result = []
  212. for key in files:
  213. if key and '.DS_Store' not in key and '__' not in key and 'pyc' not in key:
  214. key = key.replace('.py', '')
  215. result.append(key)
  216. return result
  217. """
  218. @staticmethod
  219. def url(module, key, url):
  220. str = Demeter.getMethod(module)
  221. key = key.split('.')[-1]
  222. for i,j in str:
  223. act = ''
  224. if '_path' in i:
  225. act = i.replace('_path', '')
  226. if '_html' in i:
  227. act = i.replace('_html', '.html')
  228. if act:
  229. attr = getattr(module, i)
  230. if key == 'main' and act == 'index':
  231. url.append((r'/', attr))
  232. elif key == act or act == 'index':
  233. url.append((r'/'+key, attr))
  234. url.append((r'/'+key+'/'+act, attr))
  235. return url
  236. @classmethod
  237. def start(self, application=[]):
  238. t = threading.Thread(target=lambda: self.start_server(application))
  239. t.start()
  240. @classmethod
  241. def start_server(self, application=[]):
  242. self.init(application)
  243. if 'route' in Demeter.config['setting']:
  244. Demeter.echo(Demeter.route)
  245. config = Demeter.config[Demeter.web]
  246. cookie = False
  247. if 'tornado' not in Demeter.config:
  248. Demeter.config['tornado'] = {}
  249. if 'xsrf_cookies' in config:
  250. cookie = Demeter.bool(config['xsrf_cookies'])
  251. settings = dict({
  252. "static_path": Demeter.webPath + 'static',
  253. "template_path": Demeter.webPath + 'templates',
  254. "cookie_secret": 'demeter',
  255. "login_url": '/user/login',
  256. "xsrf_cookies": cookie,
  257. "debug": Demeter.bool(config['debug']),
  258. #"autoreload": Demeter.bool(config['autoreload']),
  259. "port": config['port'],
  260. "max_buffer_size": int(config['max_buffer_size']),
  261. "process": int(config['process'])
  262. }, **Demeter.config['tornado'])
  263. com = ('cookie_secret', 'login_url', 'static_path', 'template_path')
  264. for v in com:
  265. if v in config:
  266. settings[v] = config[v]
  267. handlers = []
  268. def application_setting():
  269. handlers.append((r"/upload/(.*)", tornado.web.StaticFileHandler, {"path": Demeter.path + 'runtime/upload/'}))
  270. handlers.append((r"/files/(.*[\.png|\.jpg|\.gif|\.js|\.css|\.font|\.fonts|\.ttc|\.ttf|\.woff|\.woff2|\.fon|\.eot|\.otf])", tornado.web.StaticFileHandler, {"path": Demeter.path + 'runtime/files/'}))
  271. handlers.append((r"/qrcode/(.*)", tornado.web.StaticFileHandler, {"path": Demeter.path + 'runtime/qrcode/'}))
  272. handlers.append((r"/camera/(.*)", tornado.web.StaticFileHandler, {"path": Demeter.path + 'runtime/camera/'}))
  273. handlers.append((r"/static/(.*)", tornado.web.StaticFileHandler, {"path": "static"}))
  274. handlers.append((r"/(apple-touch-icon\.png)", tornado.web.StaticFileHandler, dict(path=settings['static_path'])))
  275. handlers.extend(Demeter.route)
  276. application_setting()
  277. application = tornado.web.Application(handlers=handlers, **settings)
  278. if Demeter.checkPy3():
  279. import asyncio
  280. asyncio.set_event_loop(asyncio.new_event_loop())
  281. if settings['debug'] == True:
  282. application.listen(settings['port'])
  283. tornado.ioloop.IOLoop.instance().start()
  284. else:
  285. server = tornado.httpserver.HTTPServer(application, settings['max_buffer_size'])
  286. server.bind(settings['port'])
  287. server.start(settings['process'])
  288. try:
  289. Demeter.echo('running on port %s' % settings['port'])
  290. tornado.ioloop.IOLoop.instance().start()
  291. except KeyboardInterrupt:
  292. tornado.ioloop.IOLoop.instance().stop()