FAQController.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. import json
  4. import os
  5. import shutil
  6. import time
  7. import traceback
  8. from django.core import serializers
  9. from django.http import HttpResponse
  10. from django.utils.decorators import method_decorator
  11. from django.views.decorators.csrf import csrf_exempt
  12. from django.views.generic.base import View
  13. import Ansjer
  14. from Ansjer.config import BASE_DIR, SERVER_TYPE, SERVER_DOMAIN, SERVER_DOMAIN_SSL
  15. from Model.models import FAQModel, HelpLink
  16. from Object.Enums.RedisKeyConstant import RedisKeyConstant
  17. from Object.RedisObject import RedisObject
  18. from Object.ResponseObject import ResponseObject
  19. from Object.TokenObject import TokenObject
  20. from var_dump import var_dump
  21. from Service.CommonService import CommonService
  22. from Service.ModelService import ModelService, ZositechHelpModel
  23. class FAQUploadView(View):
  24. @method_decorator(csrf_exempt)
  25. def dispatch(self, request, *args, **kwargs):
  26. return super(FAQUploadView, self).dispatch(request, *args, **kwargs)
  27. def get(self, request, *args, **kwargs):
  28. request.encoding = 'utf-8'
  29. request_dict = request.GET
  30. fileName = request.FILES.get('fileName', None)
  31. return self.validate(fileName, request_dict)
  32. def post(self, request, *args, **kwargs):
  33. request.encoding = 'utf-8'
  34. request_dict = request.POST
  35. fileName = request.FILES.get('fileName', None)
  36. return self.validate(fileName, request_dict)
  37. def validate(self, fileName, request_dict):
  38. token = TokenObject(request_dict.get('token', None))
  39. response = ResponseObject()
  40. if token.code != 0:
  41. return response.json(token.code)
  42. own_permission = ModelService.check_perm(userID=token.userID, permID=120)
  43. if own_permission is not True:
  44. return response.json(404)
  45. try:
  46. redisObject = RedisObject()
  47. path = '/'.join((BASE_DIR, 'static/FAQImages/tmp')).replace('\\', '/') + '/'
  48. # path = '/'.join((BASE_DIR, 'static/{user}/FAQImages'.format(user=token.userID))).replace('\\', '/') + '/'
  49. if not os.path.exists(path):
  50. os.makedirs(path)
  51. # 先从redis中取出token对应的图片信息
  52. images = redisObject.get_data(key=token.token)
  53. if images is not False:
  54. images = json.loads(images)
  55. # 判断此次编辑是否已经存在对应的图片
  56. if images.__contains__(str(fileName)):
  57. file_name = images[str(fileName)]
  58. index = file_name.find('static/')
  59. filePath = file_name[index:]
  60. if SERVER_TYPE == "Ansjer.formal_settings":
  61. filePath = SERVER_DOMAIN+'faq/image/' + filePath
  62. else:
  63. filePath = SERVER_DOMAIN_SSL+'faq/image/' + filePath
  64. return response.json(0, {'filePath': filePath})
  65. # redis中没有对应的图片信息
  66. tmp_name = str(fileName)
  67. suffix = tmp_name[tmp_name.find('.'):]
  68. tmp_file_name = int(time.time())
  69. tmp_file_name = '{file_name}{suffix}'.format(file_name=tmp_file_name, suffix=suffix)
  70. file_name = path + str(tmp_file_name)
  71. if os.path.exists(file_name):
  72. os.remove(file_name)
  73. destination = open(file_name, 'wb+')
  74. for chunk in fileName.chunks():
  75. destination.write(chunk)
  76. destination.close()
  77. # 把图片信息保存到redis中
  78. images = redisObject.get_data(token.token)
  79. if images is False:
  80. images = json.dumps({})
  81. images = json.loads(images)
  82. images[tmp_name] = file_name
  83. redisObject.set_data(key=token.token, val=json.dumps(images), expire=3600)
  84. except Exception as e:
  85. errorInfo = traceback.format_exc()
  86. print('上传文件错误: %s' % errorInfo)
  87. return response.json(700, {'details': repr(e)})
  88. else:
  89. index = file_name.find('static/')
  90. filePath = file_name[index:]
  91. if SERVER_TYPE == "Ansjer.formal_settings":
  92. filePath = 'http://www.zositechc.cn/faq/image/' + filePath
  93. else:
  94. filePath = 'http://test.www.zositechc.cn/faq/image/' + filePath
  95. # filePath = "http://192.168.136.35:8000/" + 'faq/image/' + filePath
  96. return response.json(0, {'filePath': filePath})
  97. class getFAQImage(View):
  98. def post(self, request, *args, **kwargs):
  99. request.encoding = 'utf-8'
  100. filePath = kwargs.get('filePath', None)
  101. filePath.encode(encoding='utf-8', errors='strict')
  102. response = ResponseObject()
  103. return self.getFile(filePath, response)
  104. def get(self, request, *args, **kwargs):
  105. request.encoding = 'gb2312'
  106. filePath = kwargs.get('filePath', None)
  107. response = ResponseObject()
  108. filePath.encode(encoding='gb2312', errors='strict')
  109. return self.getFile(filePath, response)
  110. def getFile(self, filePath, response):
  111. if filePath:
  112. pass
  113. else:
  114. return response.json(800)
  115. fullPath = os.path.join(BASE_DIR, filePath).replace('\\', '/')
  116. var_dump(fullPath)
  117. if os.path.isfile(fullPath):
  118. try:
  119. Imagedata = open(fullPath, 'rb').read()
  120. except Exception as e:
  121. return response.json(906, repr(e))
  122. else:
  123. return HttpResponse(Imagedata, content_type="image/jpeg")
  124. else:
  125. return response.json(907)
  126. class FAQView(View):
  127. @method_decorator(csrf_exempt)
  128. def dispatch(self, *args, **kwargs):
  129. return super(FAQView, self).dispatch(*args, **kwargs)
  130. def post(self, request, *args, **kwargs):
  131. request.encoding = 'utf-8'
  132. request_dict = request.POST
  133. operation = kwargs.get('operation', None)
  134. return self.validate(request_dict, operation)
  135. def get(self, request, *args, **kwargs):
  136. request.encoding = 'utf-8'
  137. request_dict = request.GET
  138. operation = kwargs.get('operation', None)
  139. return self.validate(request_dict, operation)
  140. def validate(self, request_dict, operation):
  141. token = TokenObject(request_dict.get('token', None))
  142. response = ResponseObject()
  143. if operation != 'zositechHelp':
  144. if token.code != 0:
  145. return response.json(token.code)
  146. if operation == 'add':
  147. return self.do_add(token, request_dict, response)
  148. elif operation == 'query':
  149. return self.do_query(token.userID, request_dict, response)
  150. elif operation == 'update':
  151. return self.do_update(token, request_dict, response)
  152. elif operation == 'delete':
  153. return self.do_delete(token.userID, request_dict, response)
  154. elif operation == 'zositechHelp':
  155. return self.do_zositechHelp(request_dict, response)
  156. elif operation == 'synZositechHelp':
  157. return self.do_synZositechHelp(request_dict, response)
  158. else:
  159. return response.json(404)
  160. def do_add(self, token, request_dict, response):
  161. own_permission = ModelService.check_perm(userID=token.userID, permID=120)
  162. if own_permission is not True:
  163. return response.json(404)
  164. title = request_dict.get('title', None)
  165. content = request_dict.get('content', None)
  166. if title and content:
  167. try:
  168. # 对content中的图片路径进行修改
  169. content = str(content)
  170. content = content.replace('faq/image/static/FAQImages/tmp', 'faq/image/static/FAQImages')
  171. # 取出redis中保存的此次上传的图片信息
  172. redisObject = RedisObject()
  173. images = redisObject.get_data(key=token.token)
  174. if images is not False:
  175. images = json.loads(images)
  176. # 把图片从临时文件移动到FAQ资源文件夹下
  177. for k, v in images.items():
  178. start_index1 = v.find('tmp/')
  179. start_index2 = start_index1 + 4
  180. new_path = v[0:start_index1] + v[start_index2:]
  181. shutil.move(v, new_path)
  182. now_time = int(time.time())
  183. FAQModel.objects.create(**{
  184. 'title': title,
  185. 'content': content,
  186. 'add_time': now_time,
  187. 'update_time': now_time
  188. })
  189. # 删除redis中token对应的信息
  190. redisObject.del_data(key=token.token)
  191. except Exception as e:
  192. print(e)
  193. return response.json(174)
  194. return response.json(0)
  195. else:
  196. return response.json(444)
  197. def do_query(self, userID, request_dict, response):
  198. page = request_dict.get('page', None)
  199. line = request_dict.get('line', None)
  200. search_key = request_dict.get('search_key', None)
  201. if page and line:
  202. if search_key:
  203. own_permission = ModelService.check_perm(userID=userID, permID=110)
  204. if own_permission is not True:
  205. return response.json(404)
  206. faq_qs = FAQModel.objects.filter(title__contains=search_key).order_by('-add_time')
  207. else:
  208. own_permission = ModelService.check_perm(userID=userID, permID=100)
  209. if own_permission is not True:
  210. return response.json(404)
  211. faq_qs = FAQModel.objects.filter().order_by('-add_time')
  212. if not faq_qs.exists():
  213. return response.json(0, {'count': 0, 'data': []})
  214. count = faq_qs.count()
  215. page = int(page)
  216. line = int(line)
  217. start = (page - 1) * line
  218. end = start + line
  219. faq_qs = faq_qs.values()[start:end]
  220. return response.json(0, {'count': count, 'data': list(faq_qs)})
  221. else:
  222. return response.json(444)
  223. def do_update(self, token, request_dict, response):
  224. own_permission = ModelService.check_perm(userID=token.userID, permID=130)
  225. if own_permission is not True:
  226. return response.json(404)
  227. id = request_dict.get('id', None)
  228. title = request_dict.get('title', None)
  229. content = request_dict.get('content', None)
  230. if id:
  231. now_time = int(time.time())
  232. data = {
  233. 'update_time': now_time
  234. }
  235. if title:
  236. data['title'] = title
  237. if content:
  238. content = str(content)
  239. content = content.replace('faq/image/static/FAQImages/tmp', 'faq/image/static/FAQImages')
  240. # 取出redis中保存的此次上传的图片信息
  241. redisObject = RedisObject()
  242. images = redisObject.get_data(key=token.token)
  243. if images is not False:
  244. images = json.loads(images)
  245. # 把图片从临时文件移动到FAQ资源文件夹下
  246. for k, v in images.items():
  247. start_index1 = v.find('tmp/')
  248. start_index2 = start_index1 + 4
  249. new_path = v[0:start_index1] + v[start_index2:]
  250. shutil.move(v, new_path)
  251. # 删除redis中token对应的信息
  252. redisObject.del_data(key=token.token)
  253. data['content'] = content
  254. FAQModel.objects.filter(id=id).update(**data)
  255. return response.json(0)
  256. else:
  257. return response.json(444)
  258. def do_delete(self, userID, request_dict, response):
  259. own_permission = ModelService.check_perm(userID=userID, permID=140)
  260. if own_permission is not True:
  261. return response.json(404)
  262. id = request_dict.get('id', None)
  263. if id:
  264. try:
  265. faq_qs = FAQModel.objects.filter(id=id)
  266. faq_qs.delete()
  267. except Exception as e:
  268. print(e)
  269. return response.json(173)
  270. else:
  271. return response.json(0)
  272. else:
  273. return response.json(444)
  274. def do_zositechHelp(self, request_dict, response):
  275. locale = request_dict.get('locale', None)
  276. label_names = request_dict.get('label_names', None)
  277. origin = request_dict.get('origin', None)
  278. help_qs = None
  279. if label_names:
  280. help = ZositechHelpModel.objects.filter(locale=locale, label_names__contains=label_names,
  281. origin=origin).values()
  282. else:
  283. help = ZositechHelpModel.objects.filter(locale=locale, origin=origin).values()
  284. if help.exists():
  285. # send_dict = CommonService.qs_to_dict(help)
  286. send_dict = list(help)
  287. return response.json(0, send_dict)
  288. else:
  289. return response.json(444)
  290. def do_synZositechHelp(self, request_dict, response):
  291. zhresults = request_dict.get('zhresults', None).replace("\'", "XX??????XX")
  292. #.replace("\"", "XX??????XX").replace("\'", "\"").replace("XX??????XX", "\'")
  293. zhresults = json.loads(zhresults)
  294. enresults = request_dict.get('enresults', None).replace("\'", "XX??????XX")
  295. enresults = json.loads(enresults)
  296. ZositechHelpModel.objects.all().delete()
  297. for data in zhresults['articles']:
  298. labname = ""
  299. if data['label_names']:
  300. for lab in data['label_names']:
  301. if lab:
  302. labname += ","
  303. labname = lab
  304. if not labname:
  305. labname = None
  306. ZositechHelpModel.objects.create(**{
  307. 'locale': data['locale'],
  308. 'label_names': labname,
  309. 'origin': 'web_widget',
  310. 'content': json.dumps(data).replace("\'", "\"").replace("XX??????XX", "\'")
  311. })
  312. for data in enresults['articles']:
  313. labname = ""
  314. if data['label_names']:
  315. for lab in data['label_names']:
  316. if lab:
  317. labname += ","
  318. labname = lab
  319. if not labname:
  320. labname = None
  321. ZositechHelpModel.objects.create(**{
  322. 'locale': data['locale'],
  323. 'label_names': labname,
  324. 'origin': 'web_widget',
  325. 'content': json.dumps(data).replace("\'", "\"").replace("XX??????XX", "\'")
  326. })
  327. return response.json(0)
  328. class HelpLinkView(View):
  329. def get(self, request, *args, **kwargs):
  330. request.encoding = 'utf-8'
  331. operation = kwargs.get('operation')
  332. return self.validation(request.GET, request, operation)
  333. def post(self, request, *args, **kwargs):
  334. request.encoding = 'utf-8'
  335. operation = kwargs.get('operation')
  336. return self.validation(request.POST, request, operation)
  337. def validation(self, request_dict, request, operation):
  338. response = ResponseObject('en')
  339. tko = TokenObject(request.META.get('HTTP_AUTHORIZATION'))
  340. if tko.code != 0:
  341. return response.json(tko.code)
  342. response.lang = tko.lang
  343. userID = tko.userID
  344. if operation == 'queryFAQByDeviceType': # 获取电池电量列表
  345. return HelpLinkView.query_faq_by_device_type(request, request_dict, response)
  346. else:
  347. return response.json(414)
  348. @staticmethod
  349. def query_faq_by_device_type(request, request_dict, response):
  350. """根据设备类型和语言查询帮助链接"""
  351. device_type = request_dict.get('device_type')
  352. lang = request_dict.get('lang')
  353. if not device_type or not lang:
  354. return response.json(444, {'message': 'device_type和lang参数不能为空'})
  355. try:
  356. redis = RedisObject()
  357. cache_key = RedisKeyConstant.HELP_LINK_TYPE.value + f'{device_type}:{lang}'
  358. # 先尝试从缓存获取
  359. cached_data = redis.get_data(cache_key)
  360. if cached_data:
  361. cached_data = json.loads(cached_data)
  362. return response.json(0, cached_data)
  363. # 查询匹配的帮助链接
  364. help_link = HelpLink.objects.filter(
  365. device_type=device_type,
  366. lang=lang,
  367. is_active=True
  368. ).first()
  369. if not help_link:
  370. return response.json(173)
  371. # 构建返回数据
  372. data = {
  373. 'url': help_link.url,
  374. 'title': help_link.title,
  375. 'description': help_link.description
  376. }
  377. # 设置缓存,过期时间30天
  378. try:
  379. redis.set_data(cache_key, json.dumps(data), RedisKeyConstant.EXPIRE_TIME_30_DAYS.value)
  380. except Exception as e:
  381. print(f"设置Redis缓存出错: {str(e)}")
  382. # 缓存失败不影响主流程
  383. return response.json(0, data)
  384. except Exception as e:
  385. print(f"查询帮助链接出错: {str(e)}")
  386. return response.json(500, {'message': '服务器内部错误'})