CampaignController.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548
  1. import time
  2. import json
  3. from Model.models import AppAdvertiseCampaign, DeviceTypeModel, CountryModel, OpenScreenCampaign
  4. from Ansjer.config import AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, SERVER_TYPE, LOGGER
  5. from django.core.paginator import Paginator
  6. from django.views import View
  7. from django.db.models import Prefetch
  8. from django.db.models import Q, F
  9. from Object.AWS.AmazonS3Util import AmazonS3Util
  10. from Object.ResponseObject import ResponseObject
  11. from Object.TokenObject import TokenObject
  12. class CampaignView(View):
  13. def get(self, request, *args, **kwargs):
  14. request.encoding = 'utf-8'
  15. operation = kwargs.get('operation')
  16. request_dict = request.GET
  17. return self.validation(request_dict, request, operation)
  18. def post(self, request, *args, **kwargs):
  19. request.encoding = 'utf-8'
  20. operation = kwargs.get('operation')
  21. request_dict = request.POST
  22. return self.validation(request_dict, request, operation)
  23. def validation(self, request_dict, request, operation):
  24. language = request_dict.get('language', 'en')
  25. response = ResponseObject(language, 'pc')
  26. if operation == 'getCountryList':
  27. return self.get_country_list(response)
  28. else:
  29. tko = TokenObject(
  30. request.META.get('HTTP_AUTHORIZATION'),
  31. returntpye='pc')
  32. if tko.code != 0:
  33. return response.json(tko.code)
  34. response.lang = tko.lang
  35. if operation == 'getCampaignList': # 获取广告活动列表
  36. return self.get_campaign_list(request_dict, response)
  37. elif operation == 'addCampaign': # 添加广告活动
  38. return self.add_campaign(request, request_dict, response)
  39. elif operation == 'updateCampaign': # 更新广告活动
  40. return self.update_campaign(request, request_dict, response)
  41. elif operation == 'deleteCampaign': # 删除广告活动
  42. return self.delete_campaign(request_dict, response)
  43. elif operation == 'switchCampaign': # 广告活动开关
  44. return self.switch_campaign(request_dict, response)
  45. elif operation == 'getUserBehaviorLog': # 获取用户行为日志
  46. return self.get_user_behavior_log(request_dict, response)
  47. else:
  48. return response.json(414)
  49. def get_campaign_list(self, request_dict, response):
  50. """
  51. 查询广告活动列表
  52. @param request_dict: 请求参数
  53. @param response: 响应对象
  54. @return: 响应对象包含广告活动列表
  55. """
  56. campaign_name = request_dict.get('campaign_name', None)
  57. campaign_country = request_dict.get('campaign_country', None)
  58. status = request_dict.get('status', None)
  59. pageNo = request_dict.get('pageNo', 1)
  60. pageSize = request_dict.get('pageSize', 20)
  61. unknown_country = 0
  62. try:
  63. # 连接并获取国家和设备类型
  64. country_prefetch = Prefetch('country', queryset=CountryModel.objects.only('country_name'),
  65. to_attr='country_list')
  66. device_type_prefetch = Prefetch('device_type', queryset=DeviceTypeModel.objects.only('name'),
  67. to_attr='device_type_list')
  68. app_advertise_campaign_qs = AppAdvertiseCampaign.objects.prefetch_related(country_prefetch,
  69. device_type_prefetch)
  70. # 过滤
  71. if campaign_name:
  72. app_advertise_campaign_qs = app_advertise_campaign_qs.filter(campaign_name=campaign_name)
  73. if status:
  74. app_advertise_campaign_qs = app_advertise_campaign_qs.filter(status=status)
  75. if campaign_country:
  76. campaign_country_list = campaign_country.split(',')
  77. if "未知地区" in campaign_country_list:
  78. unknown_country = 1
  79. app_advertise_campaign_qs = app_advertise_campaign_qs.filter(
  80. Q(country__country_name__in=campaign_country_list) | Q(unknown_country=unknown_country)).distinct()
  81. app_advertise_campaign_qs = app_advertise_campaign_qs.filter(~Q(status=2))
  82. # 分页
  83. paginator = Paginator(app_advertise_campaign_qs.order_by('id'), pageSize)
  84. campaigns = paginator.page(pageNo)
  85. # 添加设备名和地区返回
  86. campaign_list = []
  87. for campaign in campaigns.object_list:
  88. if campaign.unknown_country == 0:
  89. countries = ",".join([country.country_name for country in campaign.country_list])
  90. else:
  91. country_list = campaign.country_list
  92. country_names = []
  93. for country in country_list:
  94. country_names.append(country.country_name)
  95. country_names.append("未知地区")
  96. countries = ",".join(country_names)
  97. s3_url, _, _ = self.s3_server()
  98. # 轮播图处理
  99. banner_campaign_list = campaign.banner_campaign
  100. for banner_campaign in banner_campaign_list:
  101. banner_campaign["image"] = s3_url + banner_campaign["image"]
  102. banner_campaign_dict = {str(index): banner_campaign_list for index, banner_campaign_list in
  103. enumerate(banner_campaign_list, start=1)}
  104. campaign_data = {
  105. 'id': campaign.id,
  106. 'image_url': s3_url + campaign.image_url,
  107. 'banner_campaign': banner_campaign_dict,
  108. 'campaign_name': campaign.campaign_name,
  109. 'campaign_url': campaign.campaign_url,
  110. 'campaign_type': campaign.campaign_type,
  111. 'status': campaign.status,
  112. 'campaign_start_date': campaign.campaign_start_date,
  113. 'campaign_end_date': campaign.campaign_end_date,
  114. 'campaign_show_stime': campaign.campaign_show_stime,
  115. 'campaign_show_etime': campaign.campaign_show_etime,
  116. 'app_bundle_type': campaign.app_bundle_type,
  117. 'countries': countries,
  118. 'device_types': ",".join([device.name for device in campaign.device_type_list]),
  119. }
  120. campaign_list.append(campaign_data)
  121. data = {
  122. 'list': campaign_list,
  123. 'total': paginator.count,
  124. }
  125. return response.json(0, data)
  126. except Exception as e:
  127. print(e)
  128. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  129. def add_campaign(self, request, request_dict, response):
  130. """
  131. 添加新的广告活动
  132. @param request: 包含文件信息
  133. @param request_dict: 包含所有请求参数的字典
  134. @param response: 响应对象
  135. @return: 响应对象
  136. """
  137. try:
  138. campaign_name = request_dict.get('campaign_name', None)
  139. campaign_url = request_dict.get('campaign_url', "")
  140. status = request_dict.get('status', 2)
  141. campaign_start_time = request_dict.get('campaign_start_time', None)
  142. campaign_end_time = request_dict.get('campaign_end_time', None)
  143. campaign_show_stime = request_dict.get('campaign_show_stime', 0)
  144. campaign_show_etime = request_dict.get('campaign_show_etime', 86399)
  145. app_bundle_type = request_dict.get('app_bundle_type', None)
  146. poster_file = request.FILES.get('posterFile', None)
  147. banner_files = request.FILES.getlist('bannerFiles', None)
  148. banner_campaign_urls = json.loads(request_dict.get('banner_campaign_url', "[]"))
  149. device_type_names = json.loads(request_dict.get('device_type_list', "[]")) # 设备类型名称列表
  150. country_name_list = json.loads(request_dict.get('country_name_list', "[]")) # 地区列表
  151. required_fields = [campaign_name, campaign_start_time, campaign_end_time,
  152. app_bundle_type]
  153. list_fields = [device_type_names, country_name_list]
  154. # 检查基本字段是否为None
  155. if any(field is None for field in required_fields):
  156. return response.json(444)
  157. # 检查列表类型的字段是否为空
  158. if any(not field for field in list_fields):
  159. return response.json(444)
  160. # 确保至少提供了一个文件
  161. if not poster_file and not banner_files:
  162. return response.json(444)
  163. # 针对特殊地区的处理,表没设计好用这个处理挽救一下
  164. unknown_country = 0
  165. if "未知地区" in country_name_list:
  166. unknown_country = 1
  167. country_name_list.remove("未知地区")
  168. # 上传文件到S3
  169. banner_image_urls = self.upload_files_to_s3(banner_files,
  170. "BannerAdvertise") if banner_files is not None else []
  171. poster_image_url = self.upload_files_to_s3(poster_file,
  172. "OpenScreenAdvertise")[0] if poster_file is not None else ""
  173. campaign_type = []
  174. if poster_image_url != "":
  175. campaign_type.append(1)
  176. if banner_image_urls:
  177. campaign_type.append(2)
  178. banner_campaign = [{"image": image_url, "url": campaign_url}
  179. for image_url, campaign_url in zip(banner_image_urls, banner_campaign_urls)]
  180. else:
  181. banner_campaign = []
  182. create_time = int(time.time())
  183. update_time = int(time.time())
  184. # 创建 AppAdvertiseCampaign 实例
  185. new_campaign = AppAdvertiseCampaign.objects.create(
  186. image_url=poster_image_url,
  187. campaign_name=campaign_name,
  188. campaign_url=campaign_url,
  189. banner_campaign=banner_campaign,
  190. campaign_type=campaign_type,
  191. status=status,
  192. unknown_country=unknown_country,
  193. app_bundle_type=app_bundle_type,
  194. campaign_start_date=campaign_start_time,
  195. campaign_end_date=campaign_end_time,
  196. campaign_show_stime=campaign_show_stime,
  197. campaign_show_etime=campaign_show_etime,
  198. create_time=create_time,
  199. update_time=update_time,
  200. )
  201. # 根据名称查找 DeviceTypeModel 的实例并建立关系
  202. device_type_instances = DeviceTypeModel.objects.filter(name__in=device_type_names)
  203. for device_type_instance in device_type_instances:
  204. new_campaign.device_type.add(device_type_instance)
  205. # 根据 ID 关联 CountryModel 实例
  206. country_instances = CountryModel.objects.filter(country_name__in=country_name_list)
  207. for country_instance in country_instances:
  208. new_campaign.country.add(country_instance)
  209. return response.json(0)
  210. except Exception as e:
  211. print(e)
  212. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  213. def update_campaign(self, request, request_dict, response):
  214. """
  215. 更新广告活动
  216. @param request: 包含文件信息
  217. @param request_dict: 包含所有请求参数的字典
  218. @param response: 响应对象
  219. @return: 响应对象
  220. """
  221. try:
  222. campaign_id = request_dict.get('id', None)
  223. campaign_name = request_dict.get('campaign_name', None)
  224. campaign_url = request_dict.get('campaign_url', None)
  225. campaign_start_time = request_dict.get('campaign_start_time', None)
  226. campaign_end_time = request_dict.get('campaign_end_time', None)
  227. campaign_show_stime = request_dict.get('campaign_show_stime', None)
  228. campaign_show_etime = request_dict.get('campaign_show_etime', None)
  229. app_bundle_type = request_dict.get('app_bundle_type', None)
  230. device_type_names = json.loads(request_dict.get('device_type_list', "[]")) # 设备类型名称列表
  231. country_name_list = json.loads(request_dict.get('country_name_list', "[]")) # 地区列表
  232. poster_file = request.FILES.get('posterFile', None)
  233. # 更改轮播图广告图片
  234. sort_files = json.loads(request_dict.get('sort_files', "[]"))
  235. banner_files = request.FILES.getlist('bannerFiles', None)
  236. # 更改轮播图链接
  237. sort_urls = json.loads(request_dict.get('sort_url', "[]"))
  238. banner_urls = json.loads(request_dict.get('banner_urls', "[]"))
  239. # 删除轮播图广告
  240. sort_banner_del = json.loads(request_dict.get('sort_banner_del', "[]"))
  241. poster_del = request_dict.get('poster_del', None)
  242. if not campaign_id:
  243. return response.json(444)
  244. update_time = int(time.time())
  245. campaign = AppAdvertiseCampaign.objects.filter(pk=campaign_id).first()
  246. # 未知地区特殊处理
  247. if country_name_list is not None:
  248. if "未知地区" in country_name_list:
  249. campaign.unknown_country = 1
  250. country_name_list.remove("未知地区")
  251. else:
  252. campaign.unknown_country = 0
  253. # 开屏广告图片
  254. if poster_file is not None:
  255. self.del_file_to_s3(campaign.image_url)
  256. poster_image_url = self.upload_files_to_s3(poster_file,
  257. "OpenScreenAdvertise")[0]
  258. campaign.image_url = poster_image_url
  259. if 1 not in campaign.campaign_type:
  260. campaign.campaign_type.append(1)
  261. # 更改轮播广告图片
  262. if banner_files:
  263. for sort in sort_files:
  264. if sort <= len(campaign.banner_campaign):
  265. old_image = campaign.banner_campaign[sort - 1]["image"]
  266. self.del_file_to_s3(old_image)
  267. else:
  268. campaign.banner_campaign.append({})
  269. if sort not in sort_urls:
  270. return response.json(10, "图片需要和链接对应")
  271. banner_image_urls = self.upload_files_to_s3(banner_files, "BannerAdvertise")
  272. image_num = 0
  273. for sort in sort_files:
  274. campaign.banner_campaign[sort - 1]["image"] = banner_image_urls[image_num]
  275. image_num = image_num + 1
  276. if 2 not in campaign.campaign_type:
  277. campaign.campaign_type.append(2)
  278. # 更改轮播广告链接
  279. if banner_urls:
  280. url_num = 0
  281. for sort in sort_urls:
  282. if sort > len(campaign.banner_campaign):
  283. campaign.banner_campaign.append({})
  284. if sort not in sort_files:
  285. return response.json(10, "图片需要和链接对应")
  286. campaign.banner_campaign[sort - 1]["url"] = banner_urls[url_num]
  287. url_num = url_num + 1
  288. # 添加删除轮播图列表
  289. if sort_banner_del:
  290. sort_banner_del.sort(reverse=True)
  291. for sort in sort_banner_del:
  292. if sort <= len(campaign.banner_campaign):
  293. self.del_file_to_s3(campaign.banner_campaign[sort - 1]["image"])
  294. campaign.banner_campaign.pop(sort - 1)
  295. if not campaign.banner_campaign:
  296. campaign.campaign_type.remove(2)
  297. if poster_del is not None:
  298. campaign.campaign_url = ""
  299. self.del_file_to_s3(campaign.image_url)
  300. campaign.image_url = ""
  301. campaign.campaign_type.remove(1)
  302. if not campaign.campaign_type:
  303. campaign.status = 2
  304. # 常规字段
  305. if campaign_name is not None:
  306. campaign.campaign_name = campaign_name
  307. if campaign_url is not None:
  308. campaign.campaign_url = campaign_url
  309. if app_bundle_type is not None:
  310. campaign.app_bundle_type = app_bundle_type
  311. if campaign_start_time is not None:
  312. campaign.campaign_start_date = campaign_start_time
  313. if campaign_end_time is not None:
  314. campaign.campaign_end_date = campaign_end_time
  315. if campaign_show_stime is not None:
  316. campaign.campaign_show_stime = campaign_show_stime
  317. if campaign_show_etime is not None:
  318. campaign.campaign_show_etime = campaign_show_etime
  319. # 更新多对多字段 - 设备类型
  320. if device_type_names:
  321. device_types = DeviceTypeModel.objects.filter(name__in=device_type_names)
  322. campaign.device_type.set(device_types)
  323. # 更新多对多字段 - 国家/地区
  324. if country_name_list:
  325. countries = CountryModel.objects.filter(country_name__in=country_name_list)
  326. campaign.country.set(countries)
  327. campaign.update_time = update_time
  328. # 保存更新
  329. campaign.save()
  330. return response.json(0)
  331. except Exception as e:
  332. print(e)
  333. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  334. def delete_campaign(self, request_dict, response):
  335. """
  336. 删除广告活动
  337. @param request_dict: 包含所有请求参数的字典
  338. @param response: 响应对象
  339. @return: 响应对象
  340. """
  341. campaign_id = request_dict.get('id')
  342. if not campaign_id:
  343. return response.json(444)
  344. try:
  345. campaign = AppAdvertiseCampaign.objects.get(pk=campaign_id)
  346. if campaign.image_url != "":
  347. self.del_file_to_s3(campaign.image_url)
  348. if campaign.banner_campaign:
  349. banner_campaign_list = []
  350. for banner_campaign in campaign.banner_campaign:
  351. banner = banner_campaign["image"]
  352. banner_campaign_list.append(banner)
  353. self.del_file_to_s3(banner_campaign_list)
  354. # 清除多对多关系
  355. campaign.device_type.clear()
  356. campaign.country.clear()
  357. # 保留在广告表中
  358. campaign.status = 2
  359. campaign.update_time = int(time.time())
  360. campaign.save()
  361. return response.json(0)
  362. except Exception as e:
  363. print(e)
  364. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  365. def switch_campaign(self, request_dict, response):
  366. """
  367. 广告活动 开启/关闭
  368. @param request_dict: 包含所有请求参数的字典
  369. @param response: 响应对象
  370. @return: 响应对象
  371. """
  372. campaign_id = request_dict.get('id')
  373. status = request_dict.get('status')
  374. if not all([campaign_id, status]):
  375. return response.json(444)
  376. try:
  377. AppAdvertiseCampaign.objects.filter(pk=campaign_id).update(status=status, update_time=int(time.time()))
  378. return response.json(0)
  379. except Exception as e:
  380. print(e)
  381. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  382. def get_user_behavior_log(self, request_dict, response):
  383. campaign_ids = request_dict.get('campaign_ids', None)
  384. start_time = request_dict.get('start_time', None)
  385. end_time = request_dict.get('end_time', None)
  386. if not all([start_time, end_time]):
  387. return response.json(444)
  388. try:
  389. open_screen_campaign_qs = OpenScreenCampaign.objects.filter(
  390. update_time__range=[int(start_time), int(end_time)])
  391. if campaign_ids is not None:
  392. open_screen_campaign_qs = open_screen_campaign_qs.filter(Q(campaign_id_id__in=json.loads(campaign_ids)))
  393. open_screen_campaign_qs = open_screen_campaign_qs.select_related('campaign_id').annotate(
  394. campaign_name=F('campaign_id__campaign_name'),
  395. campaign_type=F('campaign_id__campaign_type'),
  396. campaign_status=F('campaign_id__status'),
  397. start_date=F('campaign_id__campaign_start_date'),
  398. end_date=F('campaign_id__campaign_end_date'),
  399. start_time=F('campaign_id__campaign_show_stime'),
  400. end_time=F('campaign_id__campaign_show_etime')
  401. ).values('id', 'user_id', 'status', 'update_time', 'create_time', 'campaign_id', 'campaign_name',
  402. 'campaign_type', 'campaign_status', 'start_date', 'end_date', 'start_time', 'end_time')
  403. if not open_screen_campaign_qs.exists():
  404. return response.json(0, {'list': []})
  405. campaigns_list = list(open_screen_campaign_qs)
  406. return response.json(0, {'list': campaigns_list})
  407. except Exception as e:
  408. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  409. def get_country_list(self, response):
  410. try:
  411. if SERVER_TYPE == 'Ansjer.us_config.formal_settings':
  412. region_api = 'https://www.dvema.com/'
  413. elif SERVER_TYPE == 'Ansjer.eur_config.formal_settings':
  414. region_api = 'https://api.zositeche.com/'
  415. elif SERVER_TYPE == 'Ansjer.cn_config.formal_settings':
  416. region_api = 'https://www.zositechc.cn/'
  417. else:
  418. region_api = 'https://test.zositechc.cn/'
  419. country_qs = CountryModel.objects.filter(region__api=region_api).values('country_name')
  420. if not country_qs.exists():
  421. return response.json(173)
  422. country_list = []
  423. for country in country_qs:
  424. country_list.append(country['country_name'])
  425. return response.json(0, {'list': country_list})
  426. except Exception as e:
  427. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  428. @staticmethod
  429. def upload_files_to_s3(files, path_prefix):
  430. """
  431. 广告图片文件上传到S3存储桶
  432. @param files: 文件 request.FILES获取
  433. @param path_prefix: 存储路径
  434. @return image_urls: 列表
  435. """
  436. try:
  437. _, regin, AWS_SES_ACCESS_REGION = CampaignView.s3_server()
  438. # 确保files是一个列表,统一处理单个文件和多个文件
  439. if not isinstance(files, list):
  440. files = [files]
  441. image_urls = []
  442. bucket_name = "ansjerfilemanager"
  443. s3 = AmazonS3Util(AWS_ACCESS_KEY_ID[regin], AWS_SECRET_ACCESS_KEY[regin], AWS_SES_ACCESS_REGION)
  444. for file in files:
  445. timestamp = int(time.time())
  446. file_key = f'app/campaign/{path_prefix}/{timestamp}_{file.name}'
  447. s3.upload_file_obj(bucket_name, file_key, file,
  448. {'ContentType': file.content_type, 'ACL': 'public-read'})
  449. image_urls.append(f"{path_prefix}/{timestamp}_{file.name}")
  450. return image_urls
  451. except Exception as e:
  452. LOGGER.info('存储桶添加异常:error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  453. return None
  454. @staticmethod
  455. def del_file_to_s3(path_list):
  456. """
  457. S3存储桶 删除文件
  458. @param path_list: 单一字符串或者列表
  459. @return
  460. """
  461. try:
  462. _, regin, AWS_SES_ACCESS_REGION = CampaignView.s3_server()
  463. if not isinstance(path_list, list):
  464. path_list = [path_list]
  465. for path in path_list:
  466. # 删除存储桶原来的图片
  467. s3 = AmazonS3Util(AWS_ACCESS_KEY_ID[regin], AWS_SECRET_ACCESS_KEY[regin], AWS_SES_ACCESS_REGION)
  468. bucket_name = 'ansjerfilemanager'
  469. s3.delete_obj(bucket_name, f"app/campaign/{path}")
  470. except Exception as e:
  471. LOGGER.info('存储桶删除异常:error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  472. @staticmethod
  473. def s3_server():
  474. if SERVER_TYPE == 'Ansjer.cn_config.formal_settings' or SERVER_TYPE == 'Ansjer.cn_config.test_settings':
  475. s3_url = "https://ansjerfilemanager.s3.cn-northwest-1.amazonaws.com.cn/app/campaign/"
  476. regin = 0
  477. AWS_SES_ACCESS_REGION = "cn-northwest-1"
  478. else:
  479. s3_url = "https://ansjerfilemanager.s3.amazonaws.com/app/campaign/"
  480. regin = 1
  481. AWS_SES_ACCESS_REGION = 'us-east-1'
  482. return s3_url, regin, AWS_SES_ACCESS_REGION