douyin_author_scheduling_new.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. # -*- coding: utf-8 -*-
  2. # @Time: 2023/11/07
  3. import os
  4. import random
  5. import sys
  6. import time
  7. import requests
  8. import json
  9. import urllib3
  10. sys.path.append(os.getcwd())
  11. from datetime import timedelta, date
  12. from common.common import Common
  13. from common import AliyunLogger
  14. from common.mq import MQ
  15. from requests.adapters import HTTPAdapter
  16. from common.scheduling_db import MysqlHelper
  17. from common.public import get_config_from_mysql, download_rule
  18. from douyin.douyin_author.douyin_author_scheduling_help import DouYinHelper
  19. class DouyinauthorScheduling:
  20. platform = "抖音"
  21. download_cnt = 0
  22. @classmethod
  23. def videos_cnt(cls, rule_dict):
  24. videos_cnt = rule_dict.get("videos_cnt", {}).get("min", 0)
  25. if videos_cnt == 0:
  26. videos_cnt = 1000
  27. return videos_cnt
  28. @classmethod
  29. def get_cookie(cls, log_type, crawler, env):
  30. select_sql = f""" select * from crawler_config where source="{crawler}" """
  31. configs = MysqlHelper.get_values(log_type, crawler, select_sql, env, action="")
  32. for config in configs:
  33. if "cookie" in config["config"]:
  34. cookie_dict = {
  35. "cookie_id": config["id"],
  36. "title": config["title"].strip(),
  37. "cookie": dict(eval(config["config"]))["cookie"].strip(),
  38. "update_time": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(int(config["update_time"] / 1000))),
  39. "operator": config["operator"].strip()
  40. }
  41. return cookie_dict
  42. @classmethod
  43. def get_videoList(cls, log_type, crawler, user_dict, rule_dict, env):
  44. mq = MQ(topic_name="topic_crawler_etl_" + env)
  45. next_cursor = 0
  46. while True:
  47. cookie = cls.get_cookie(log_type, crawler, env)["cookie"]
  48. time.sleep(random.randint(5, 10))
  49. url = 'https://www.douyin.com/aweme/v1/web/aweme/post/'
  50. account_id = user_dict["link"]
  51. headers = {
  52. 'Accept': 'application/json, text/plain, */*',
  53. 'Accept-Language': 'zh-CN,zh;q=0.9',
  54. 'Cache-Control': 'no-cache',
  55. 'Cookie': f"ttwid=" + cookie,
  56. 'Pragma': 'no-cache',
  57. 'Referer': f'https://www.douyin.com/user/{account_id}',
  58. 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) '
  59. 'Chrome/118.0.0.0 Safari/537.36',
  60. }
  61. query = DouYinHelper.get_full_query(ua=headers['User-Agent'], extra_data={
  62. 'sec_user_id': account_id,
  63. 'max_cursor': next_cursor,
  64. 'locate_query': 'false',
  65. 'show_live_replay_strategy': '1',
  66. 'need_time_list': '1',
  67. 'time_list_query': '0',
  68. 'whale_cut_token': '',
  69. 'cut_version': '1',
  70. 'count': '18',
  71. 'publish_video_strategy_type': '2',
  72. })
  73. urllib3.disable_warnings()
  74. s = requests.session()
  75. # max_retries=3 重试3次
  76. s.mount('http://', HTTPAdapter(max_retries=3))
  77. s.mount('https://', HTTPAdapter(max_retries=3))
  78. response = requests.request(method='GET', url=url, headers=headers, params=query)
  79. body = response.content.decode()
  80. obj = json.loads(body)
  81. has_more = True if obj.get('has_more', 0) == 1 else False
  82. next_cursor = str(obj.get('max_cursor')) if has_more else None
  83. data = obj.get('aweme_list', [])
  84. response.close()
  85. if response.status_code != 200:
  86. Common.logger(log_type, crawler).warning(f"data:{data}\n")
  87. AliyunLogger.logging(
  88. code="2000",
  89. platform=crawler,
  90. mode=log_type,
  91. env=env,
  92. message=f"data:{data}\n"
  93. )
  94. return
  95. elif len(data) == 0:
  96. Common.logger(log_type, crawler).warning(f"没有更多视频啦 ~\n")
  97. AliyunLogger.logging(
  98. code="2001",
  99. platform=crawler,
  100. mode=log_type,
  101. env=env,
  102. message=f"没有更多视频啦 ~\n"
  103. )
  104. return
  105. for i in range(len(data)):
  106. try:
  107. entity_type = data[i].get('search_impr').get('entity_type')
  108. if entity_type == 'GENERAL':
  109. Common.logger(log_type, crawler).info('扫描到一条视频\n')
  110. AliyunLogger.logging(
  111. code="1001",
  112. platform=crawler,
  113. mode=log_type,
  114. env=env,
  115. message='扫描到一条视频\n'
  116. )
  117. video_id = data[i].get('aweme_id') # 文章id
  118. video_title = data[i].get('desc', "").strip().replace("\n", "") \
  119. .replace("/", "").replace("\\", "").replace("\r", "") \
  120. .replace(":", "").replace("*", "").replace("?", "") \
  121. .replace("?", "").replace('"', "").replace("<", "") \
  122. .replace(">", "").replace("|", "").replace(" ", "") \
  123. .replace("&NBSP", "").replace(".", "。").replace(" ", "") \
  124. .replace("'", "").replace("#", "").replace("Merge", "")
  125. publish_time_stamp = data[i].get('create_time') # 发布时间
  126. publish_time_str = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(publish_time_stamp))
  127. video_url = data[i].get('video').get('play_addr').get('url_list')[0] # 视频链接
  128. cover_url = data[i].get('video').get('cover').get('url_list')[0] # 视频封面
  129. digg_count = int(data[i].get('statistics').get('digg_count')) # 点赞
  130. comment_count = int(data[i].get('statistics').get('comment_count')) # 评论
  131. # collect_count = data[i].get('statistics').get('collect_count') # 收藏
  132. share_count = int(data[i].get('statistics').get('share_count')) # 转发
  133. date_three_days_ago_string = (date.today() + timedelta(days=-5)).strftime("%Y-%m-%d %H:%M:%S")
  134. rule = publish_time_str > date_three_days_ago_string
  135. if i > 2:
  136. if rule == False:
  137. break
  138. if rule == False:
  139. Common.logger(log_type, crawler).info(f"发布时间小于5天,发布时间:{publish_time_str}\n")
  140. AliyunLogger.logging(
  141. code="2004",
  142. platform=crawler,
  143. mode=log_type,
  144. env=env,
  145. message=f"发布时间小于5天,发布时间:{publish_time_str}\n"
  146. )
  147. continue
  148. video_percent = '%.2f' % (share_count / digg_count)
  149. if digg_count < 50000 and digg_count < 50:
  150. if float(video_percent) < 0.01:
  151. Common.logger(log_type, crawler).info(f"不符合条件:分享/点赞-{video_percent},点赞量-{digg_count}\n")
  152. AliyunLogger.logging(
  153. code="2004",
  154. platform=crawler,
  155. mode=log_type,
  156. env=env,
  157. message=f"不符合条件:分享/点赞-{video_percent},点赞量-{digg_count}\n"
  158. )
  159. continue
  160. video_dict = {'video_title': video_title,
  161. 'video_id': video_id,
  162. 'play_cnt': 0,
  163. 'like_cnt': digg_count,
  164. 'comment_cnt': comment_count,
  165. 'share_cnt': share_count,
  166. 'video_width': 0,
  167. 'video_height': 0,
  168. 'duration': 0,
  169. 'publish_time_stamp': publish_time_stamp,
  170. 'publish_time_str': publish_time_str,
  171. 'user_name': "douyin",
  172. 'user_id': video_id,
  173. 'avatar_url': '',
  174. 'cover_url': cover_url,
  175. 'video_url': video_url,
  176. 'session': f"douyin-{int(time.time())}"}
  177. for k, v in video_dict.items():
  178. Common.logger(log_type, crawler).info(f"{k}:{v}")
  179. AliyunLogger.logging(
  180. code="1000",
  181. platform=crawler,
  182. mode=log_type,
  183. env=env,
  184. message=f"{video_dict}\n"
  185. )
  186. if int((int(time.time()) - int(publish_time_stamp)) / (3600*24)) > int(rule_dict.get("period", {}).get("max", 1000)):
  187. Common.logger(log_type, crawler).info(f'发布时间超过{int(rule_dict.get("period", {}).get("max", 1000))}天\n')
  188. AliyunLogger.logging(
  189. code="2004",
  190. platform=crawler,
  191. mode=log_type,
  192. env=env,
  193. message=f'发布时间超过{int(rule_dict.get("period", {}).get("max", 1000))}天\n'
  194. )
  195. return
  196. if video_dict["video_id"] == '' or video_dict["cover_url"] == '' or video_dict["video_url"] == '':
  197. Common.logger(log_type, crawler).info('无效视频\n')
  198. AliyunLogger.logging(
  199. code="2004",
  200. platform=crawler,
  201. mode=log_type,
  202. env=env,
  203. message='无效视频\n'
  204. )
  205. elif download_rule(log_type=log_type, crawler=crawler, video_dict=video_dict, rule_dict=rule_dict) is False:
  206. Common.logger(log_type, crawler).info("不满足抓取规则\n")
  207. AliyunLogger.logging(
  208. code="2004",
  209. platform=crawler,
  210. mode=log_type,
  211. env=env,
  212. message='不满足抓取规则\n'
  213. )
  214. elif any(str(word) if str(word) in video_dict["video_title"] else False
  215. for word in get_config_from_mysql(log_type=log_type,
  216. source=crawler,
  217. env=env,
  218. text="filter",
  219. action="")) is True:
  220. Common.logger(log_type, crawler).info('已中过滤词\n')
  221. AliyunLogger.logging(
  222. code="2004",
  223. platform=crawler,
  224. mode=log_type,
  225. env=env,
  226. message='已中过滤词\n'
  227. )
  228. elif cls.repeat_video(log_type, crawler, video_dict["video_id"], env) != 0:
  229. Common.logger(log_type, crawler).info('视频已下载\n')
  230. AliyunLogger.logging(
  231. code="2002",
  232. platform=crawler,
  233. mode=log_type,
  234. env=env,
  235. message='视频已下载\n'
  236. )
  237. else:
  238. video_dict["out_user_id"] = video_dict["user_id"]
  239. video_dict["platform"] = crawler
  240. video_dict["strategy"] = log_type
  241. video_dict["out_video_id"] = video_dict["video_id"]
  242. video_dict["width"] = video_dict["video_width"]
  243. video_dict["height"] = video_dict["video_height"]
  244. video_dict["crawler_rule"] = json.dumps(rule_dict)
  245. video_dict["user_id"] = user_dict["uid"]
  246. video_dict["publish_time"] = video_dict["publish_time_str"]
  247. video_dict["strategy_type"] = log_type
  248. mq.send_msg(video_dict)
  249. cls.download_cnt += 1
  250. except Exception as e:
  251. Common.logger(log_type, crawler).warning(f"抓取单条视频异常:{e}\n")
  252. AliyunLogger.logging(
  253. code="3000",
  254. platform=crawler,
  255. mode=log_type,
  256. env=env,
  257. message=f"抓取单条视频异常:{e}\n"
  258. )
  259. @classmethod
  260. def repeat_video(cls, log_type, crawler, video_id, env):
  261. sql = f""" select * from crawler_video where platform in ("{crawler}","{cls.platform}") and out_video_id="{video_id}"; """
  262. repeat_video = MysqlHelper.get_values(log_type, crawler, sql, env)
  263. return len(repeat_video)
  264. @classmethod
  265. def get_author_videos(cls, log_type, crawler, user_list, rule_dict, env):
  266. for user_dict in user_list:
  267. try:
  268. Common.logger(log_type, crawler).info(f"开始抓取 {user_dict['nick_name']} 主页视频")
  269. AliyunLogger.logging(
  270. code="2000",
  271. platform=crawler,
  272. mode=log_type,
  273. env=env,
  274. message=f"开始抓取 {user_dict['nick_name']} 主页视频"
  275. )
  276. cls.download_cnt = 0
  277. cls.get_videoList(log_type=log_type,
  278. crawler=crawler,
  279. user_dict=user_dict,
  280. rule_dict=rule_dict,
  281. env=env)
  282. except Exception as e:
  283. Common.logger(log_type, crawler).warning(f"抓取用户{user_dict['nick_name']}主页视频时异常:{e}\n")
  284. AliyunLogger.logging(
  285. code="3000",
  286. platform=crawler,
  287. mode=log_type,
  288. env=env,
  289. message=f"抓取用户{user_dict['nick_name']}主页视频时异常:{e}\n"
  290. )
  291. if __name__ == "__main__":
  292. print(DouyinauthorScheduling.get_cookie("author", "douyin", "prod")["cookie"])
  293. pass