拍爱

views.py 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. # -*- coding: utf-8 -*-
  2. import xmltodict
  3. from django.conf import settings
  4. from django.db import transaction
  5. from django.shortcuts import HttpResponse
  6. from logit import logit
  7. from TimeConvert import TimeConvert as tc
  8. from wechatpy import WeChatPay, WeChatPayException
  9. from account.models import UserIncomeExpensesInfo, UserInfo
  10. from group.models import GroupPhotoInfo, GroupPhotoOrderInfo
  11. from pay.models import OrderInfo
  12. from photo.models import PhotosInfo
  13. from utils.error.errno_utils import (GroupPhotoStatusCode, OrderStatusCode, UserStatusCode, WechatStatusCode,
  14. WithdrawStatusCode)
  15. from utils.error.response_utils import response
  16. from utils.page_utils import pagination
  17. from utils.redis.rbrief import set_brief_info
  18. from utils.redis.rorder import set_lensman_order_record
  19. r = settings.REDIS_CACHE
  20. WECHAT = settings.WECHAT
  21. @logit
  22. @transaction.atomic
  23. def wx_order_create_api(request):
  24. """
  25. 订单创建
  26. :param request:
  27. :return:
  28. """
  29. group_id = request.POST.get('group_id', '')
  30. user_id = request.POST.get('user_id', '')
  31. photo_id = request.POST.get('photo_id', '')
  32. photo_type = request.POST.get('photo_type', 'nomark') # nomark for 去除水印, origin for 获取高清图
  33. photo_type_ = OrderInfo.ORIGIN if photo_type == 'origin' else OrderInfo.NOMARK
  34. # 群组照片校验
  35. try:
  36. group_photo = GroupPhotoInfo.objects.get(pk=photo_id)
  37. except GroupPhotoInfo.DoesNotExist:
  38. return response(GroupPhotoStatusCode.GROUP_PHOTO_NOT_FOUND)
  39. # 判断是否重复购买
  40. if OrderInfo.objects.filter(photo_id=photo_id, photo_type=photo_type_, from_uid=user_id, pay_status=OrderInfo.PAID).exists():
  41. return response(OrderStatusCode.WX_ORDER_PAID_ALREADY_EXISTS)
  42. body = request.POST.get('body', '') # 商品描述
  43. total_fee = int(request.POST.get('total_fee', 0)) # 总金额,单位分
  44. # 金额校验
  45. # if int(r.get(LENSMAN_PHOTO_PRICE % (user_id, photo_id, photo_type)) or 0) != total_fee:
  46. # return response(OrderStatusCode.FEE_CHECK_FAIL)
  47. # 获取 from_uid, to_uid
  48. from_uid = user_id
  49. to_uid = group_photo.lensman_id or group_photo.user_id
  50. # JSAPI--公众号支付、NATIVE--原生扫码支付、APP--app支付,统一下单接口trade_type的传参可参考这里
  51. trade_type = request.POST.get('trade_type', '')
  52. # 根据 trade_type 获取 wechat 配置
  53. wechat = WECHAT.get(trade_type, {})
  54. # WeChatPay 初始化
  55. wxpay = WeChatPay(wechat.get('appID'), wechat.get('apiKey'), wechat.get('mchID'))
  56. # 生成订单
  57. order = OrderInfo.objects.create(
  58. group_id=group_id,
  59. photo_id=photo_id,
  60. lensman_photo_id=group_photo.lensman_photo_id,
  61. photo_type=1 if photo_type == 'origin' else 0,
  62. from_uid=from_uid,
  63. to_uid=to_uid,
  64. session_id=group_photo.session_id,
  65. total_fee=total_fee,
  66. trade_type=trade_type,
  67. )
  68. try:
  69. prepay_data = wxpay.order.create(
  70. body=body,
  71. notify_url=settings.API_DOMAIN + '/wx/notify_url',
  72. out_trade_no=order.order_id,
  73. total_fee=total_fee,
  74. trade_type=trade_type,
  75. # user_id=None, # 可选,用户在商户appid下的唯一标识。trade_type=JSAPI,此参数必传
  76. )
  77. except WeChatPayException:
  78. return response(OrderStatusCode.WX_UNIFIED_ORDER_FAIL)
  79. prepay_id = prepay_data.get('prepay_id', '')
  80. if trade_type == 'JSAPI':
  81. wxpay_params = wxpay.jsapi.get_jsapi_params(prepay_id)
  82. elif trade_type == 'APP':
  83. wxpay_params = wxpay.order.get_appapi_params(prepay_id)
  84. return response(200, 'Order Create Success', u'订单创建成功', {
  85. 'order_id': order.order_id,
  86. 'prepay_id': prepay_id,
  87. 'wxpay_params': wxpay_params,
  88. })
  89. def order_paid_success(order):
  90. if order.pay_status == OrderInfo.PAID:
  91. return
  92. if order.photo_type == OrderInfo.NOMARK:
  93. order.photo_status = OrderInfo.FETCHED
  94. order.pay_status = OrderInfo.PAID
  95. order.paid_at = tc.utc_datetime()
  96. order.save()
  97. porder, created = GroupPhotoOrderInfo.objects.select_for_update().get_or_create(
  98. group_id=order.group_id,
  99. session_id=order.session_id,
  100. user_id=order.from_uid,
  101. photo_id=order.photo_id,
  102. lensman_photo_id=order.lensman_photo_id,
  103. )
  104. photo = PhotosInfo.objects.get(
  105. photo_id=order.lensman_photo_id,
  106. )
  107. if order.photo_type == OrderInfo.NOMARK:
  108. porder.m_photo_path = photo.m_photo_path
  109. elif order.photo_type == OrderInfo.ORIGIN:
  110. porder.r_photo_path = photo.r_photo_path
  111. porder.save()
  112. set_lensman_order_record(porder)
  113. to_uid = order.to_uid
  114. total_fee = order.total_fee
  115. try:
  116. user = UserInfo.objects.select_for_update().get(user_id=to_uid)
  117. except UserInfo.DoesNotExist:
  118. return
  119. if order.photo_type == OrderInfo.NOMARK:
  120. # 余额增加
  121. amount, freeze_income_amount = total_fee, 0
  122. user.balance += amount
  123. # Redis 数值更新
  124. set_brief_info(to_uid, order.photo_type, total_fee)
  125. # 余额记录
  126. UserIncomeExpensesInfo.objects.create(
  127. user_id=to_uid,
  128. photo_id=order.photo_id,
  129. type=UserIncomeExpensesInfo.INCOME,
  130. amount=amount,
  131. balance=user.balance,
  132. freeze_income_amount=freeze_income_amount,
  133. freeze_income_balance=user.freeze_income_balance,
  134. remark=u'图片购买',
  135. )
  136. elif order.photo_type == OrderInfo.ORIGIN:
  137. amount, freeze_income_amount = 0, total_fee
  138. user.freeze_income_balance += freeze_income_amount
  139. user.save()
  140. def order_paid_fail(order):
  141. if order.pay_status == OrderInfo.FAIL:
  142. return
  143. order.pay_status = OrderInfo.FAIL
  144. order.save()
  145. @logit
  146. @transaction.atomic
  147. def wx_order_query_api(request):
  148. """
  149. 订单查询
  150. :param request:
  151. :return:
  152. """
  153. order_id = request.POST.get('order_id', '')
  154. transaction_id = request.POST.get('transaction_id', '')
  155. try:
  156. order = OrderInfo.objects.select_for_update().get(order_id=order_id)
  157. except OrderInfo.DoesNotExist:
  158. return response(OrderStatusCode.WX_ORDER_NOT_FOUND)
  159. if order.pay_status == OrderInfo.PAID:
  160. return response(200, 'Order Pay Success', u'订单支付成功')
  161. elif order.pay_status == OrderInfo.FAIL:
  162. return response(OrderStatusCode.WX_ORDER_PAY_FAIL)
  163. # 根据 trade_type 获取 wechat 配置
  164. wechat = WECHAT.get(order.trade_type, {})
  165. # WeChatPay 初始化
  166. wxpay = WeChatPay(wechat.get('appID'), wechat.get('apiKey'), wechat.get('mchID'))
  167. # 订单查询
  168. query_data = wxpay.order.query(transaction_id, order_id)
  169. # 签名校验
  170. if not wxpay.check_signature(query_data):
  171. return response(OrderStatusCode.SIGN_CHECK_FAIL)
  172. # 交易状态
  173. trade_state = query_data.get('trade_state')
  174. # 订单状态判断更新
  175. if trade_state == 'SUCCESS': # 订单支付成功
  176. order_paid_success(order)
  177. return response(200, 'Order Pay Success', u'订单支付成功')
  178. elif trade_state == 'NOTPAY': # 订单未支付
  179. return response(OrderStatusCode.WX_ORDER_NOT_PAY)
  180. elif trade_state == 'USERPAYING': # 订单支付中
  181. return response(OrderStatusCode.WX_ORDER_PAYING)
  182. else: # 订单支付失败
  183. order_paid_fail(order)
  184. return response(OrderStatusCode.WX_ORDER_PAY_FAIL)
  185. @logit
  186. @transaction.atomic
  187. def wx_order_list_api(request):
  188. """
  189. 订单列表
  190. :param request:
  191. :return:
  192. """
  193. user_id = request.POST.get('user_id', '')
  194. page = int(request.POST.get('page', 1))
  195. num = int(request.POST.get('num', settings.ORDER_NUM_PER_PAGE))
  196. orders = OrderInfo.objects.filter(from_uid=user_id, pay_status=OrderInfo.PAID, status=True).order_by('-pk')
  197. orders, left = pagination(orders, page, num)
  198. orders = [order.data(user_id) for order in orders]
  199. return response(200, 'Get Order List Success', u'获取订单列表成功', {
  200. 'orders': orders,
  201. 'left': left,
  202. })
  203. @logit
  204. @transaction.atomic
  205. def wx_order_detail_api(request):
  206. """
  207. 订单详情
  208. :param request:
  209. :return:
  210. """
  211. user_id = request.POST.get('user_id', '')
  212. order_id = request.POST.get('order_id', '')
  213. try:
  214. order = OrderInfo.objects.get(order_id=order_id)
  215. except OrderInfo.DoesNotExist:
  216. return response(OrderStatusCode.WX_ORDER_NOT_FOUND)
  217. if user_id not in [order.from_uid, order.to_uid]:
  218. return response(OrderStatusCode.NO_DETAIL_PERMISSION)
  219. return response(200, 'Get Order Detail Success', u'获取订单详情成功')
  220. @logit
  221. @transaction.atomic
  222. def wx_notify_url_api(request):
  223. """
  224. 支付异步通知回调地址
  225. :param request:
  226. :return:
  227. """
  228. try:
  229. data = xmltodict.parse(request.body)['xml']
  230. except xmltodict.ParsingInterrupted:
  231. # 解析 XML 失败
  232. return HttpResponse(settings.WXPAY_NOTIFY_FAIL)
  233. trade_type = data.get('trade_type', '')
  234. # 根据 trade_type 获取 wechat 配置
  235. wechat = WECHAT.get(trade_type, {})
  236. # WeChatPay 初始化
  237. wxpay = WeChatPay(wechat.get('appID'), wechat.get('apiKey'), wechat.get('mchID'))
  238. # 签名校验
  239. if not wxpay.check_signature(data):
  240. return response(OrderStatusCode.SIGN_CHECK_FAIL)
  241. out_trade_no = data.get('out_trade_no', '')
  242. return_code = data.get('return_code', '')
  243. result_code = data.get('result_code', '')
  244. if return_code != 'SUCCESS' or result_code != 'SUCCESS':
  245. return HttpResponse(settings.WXPAY_NOTIFY_FAIL)
  246. try:
  247. order = OrderInfo.objects.get(order_id=out_trade_no)
  248. except OrderInfo.DoesNotExist:
  249. return response(OrderStatusCode.WX_ORDER_NOT_FOUND)
  250. order_paid_success(order)
  251. return HttpResponse(settings.WXPAY_NOTIFY_SUCCESS)
  252. @logit
  253. @transaction.atomic
  254. def wx_balance_withdraw_api(request):
  255. user_id = request.POST.get('user_id', '')
  256. # 用户校验
  257. try:
  258. user = UserInfo.objects.select_for_update().get(user_id=user_id)
  259. except UserInfo.DoesNotExist:
  260. return response(UserStatusCode.USER_NOT_FOUND)
  261. # JSAPI--公众号支付、NATIVE--原生扫码支付、APP--app支付,统一下单接口trade_type的传参可参考这里
  262. trade_type = request.POST.get('trade_type', '')
  263. # TRANSFER--企业付款、PACKET--现金红包, 余额提现接口withdraw_type的传参可参考这里
  264. withdraw_type = request.POST.get('withdraw_type', 'TRANSFER')
  265. amount = int(request.POST.get('amount', 0))
  266. if not user.openid:
  267. return response(WechatStatusCode.OPENID_NOT_FOUND)
  268. if user.balance < amount:
  269. return response(WithdrawStatusCode.BALANCE_NOT_ENOUGH)
  270. # 根据 trade_type 获取 wechat 配置
  271. wechat = WECHAT.get(trade_type, {})
  272. # WeChatPay 初始化
  273. wxpay = WeChatPay(wechat.get('appID'), wechat.get('apiKey'), wechat.get('mchID'), mch_cert=wechat.get('mch_cert'), mch_key=wechat.get('mch_key'))
  274. if withdraw_type == 'TRANSFER':
  275. ret_data = wxpay.transfer.transfer(user.openid, amount, u'摄影师余额提现,企业付款', check_name='NO_CHECK')
  276. elif withdraw_type == 'PACKET':
  277. ret_data = wxpay.redpack.send(
  278. user.openid,
  279. amount,
  280. send_name=wechat.get('redpacket', {}).get('SEND_NAME'),
  281. nick_name=wechat.get('redpacket', {}).get('NICK_NAME'),
  282. act_name=wechat.get('redpacket', {}).get('ACT_NAME'),
  283. wishing=wechat.get('redpacket', {}).get('WISHING'),
  284. remark=wechat.get('redpacket', {}).get('REMARK'),
  285. )
  286. # 根据 ret_data 判断是否提现成功, 成功则减余额, 失败则提示
  287. user.balance -= amount
  288. user.save()
  289. return response(200, 'Withdraw Success', u'提现成功', {})