| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335 | # -*- coding: utf-8 -*-# @Time: 2023/11/03import jsonimport osimport randomimport sysimport timeimport requestsimport urllib3sys.path.append(os.getcwd())from common import AliyunLoggerfrom common.mq import MQfrom common.common import Commonfrom common.scheduling_db import MysqlHelperfrom common.public import get_config_from_mysql, download_ruleproxies = {"http": None, "https": None}class KanyikanRecommend:    platform = "看一看-feed流"    strategy = "feed流"    @classmethod    def repeat_video(cls, log_type, crawler, video_id, env):        sql = f""" select * from crawler_video where platform in ("{crawler}","{cls.platform}") and create_time>='2023-10-09' and out_video_id="{video_id}"; """        repeat_video = MysqlHelper.get_values(log_type, crawler, sql, env)        return len(repeat_video)    @classmethod    def get_vid(cls,session, log_type, crawler, env, our_uid, rule_dict):        url = 'https://search.weixin.qq.com/cgi-bin/recwxa/recwxavideolist?'        header = {            'Host': 'search.weixin.qq.com',            'Content-Type': 'application/json',            'X-WX-ClientVersion': '0x33050520',            'X-WECHAT-UIN': 'b2hfbTQ1WGNjSzQxemdfanpMSml1TEtfbEtsVQ==',            'Accept': '*/*',            'User-Agent': 'Mozilla/5.0 (iPhone; CPU iPhone OS 11_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E217 MicroMessenger/6.8.0(0x16080000) NetType/WIFI Language/en Branch/Br_trunk MiniProgramEnv/Mac',            'Referer': 'https://servicewechat.com/wxbb9a805eb4f9533c/268/page-frame.html',            'Accept-Language': 'zh-cn'        }        params = {            "session": session,            "offset": 0,            "count": "10",            "channelid": "200",            "scene": '310',            "subscene": '1074',            "sharesearchid": '0',            "nettype": 'wifi',            "switchprofile": "0",            "switchnewuser": "0",            "ad": 0        }        urllib3.disable_warnings()        response = requests.get(url=url, headers=header, params=params, proxies=proxies, verify=False)        if "data" not in response.text:            Common.logger(log_type, crawler).info("获取视频list时,session过期,随机睡眠 31-50 秒")            Common.logging(log_type, crawler, env, "获取视频list时,session过期,随机睡眠 31-50 秒")            AliyunLogger.logging(                code="2000",                platform=crawler,                mode=log_type,                env=env,                message=f"获取视频list时,session过期,随机睡眠 31-50 秒"            )            # 如果返回空信息,则随机睡眠 31-40 秒            time.sleep(random.randint(31, 40))            cls.get_videoList(log_type, crawler, our_uid, rule_dict, env)        elif "items" not in response.json()["data"]:            Common.logger(log_type, crawler).info(f"get_feeds:{response.json()},随机睡眠 1-3 分钟")            Common.logging(log_type, crawler, env, f"get_feeds:{response.json()},随机睡眠 1-3 分钟")            AliyunLogger.logging(                code="2000",                platform=crawler,                mode=log_type,                env=env,                message=f"get_feeds:{response.json()},随机睡眠 1-3 分钟"            )            # 如果返回空信息,则随机睡眠 1-3 分钟            time.sleep(random.randint(60, 180))            cls.get_videoList(log_type, crawler, our_uid, rule_dict, env)        feeds = response.json().get("data", {}).get("items", "")        return feeds    @classmethod    def get_videoList(cls, log_type, crawler, our_uid, rule_dict, env):        mq = MQ(topic_name="topic_crawler_etl_" + env)        try:            session = Common.get_session(log_type, crawler, env)            if session is None:                time.sleep(1)                cls.get_videoList(log_type, crawler, our_uid, rule_dict, env)            feeds = cls.get_vid(session,log_type, crawler, env, our_uid, rule_dict)            if feeds == "":                Common.logger(log_type, crawler).info(f"feeds:{feeds}")                Common.logging(log_type, crawler, env, f"feeds:{feeds}")                return            for i in range(len(feeds)):                vid = feeds[i].get("videoId", "")                shared_cnt = int(feeds[i].get("shared_cnt", 0))                liked_cnt = int(feeds[i].get("liked_cnt", 0))                playCount = int(feeds[i].get("playCount", 0))                video_percent = '%.2f' % (liked_cnt / playCount)                if playCount < 100000:                    if float(video_percent) < 0.01 and playCount < 10000:                        Common.logger(log_type, crawler).info(f"分享/播放:{video_percent},播放量:{playCount}\n")                        Common.logging(log_type, crawler, env, f"分享/播放:{video_percent},播放量:{playCount}\n")                        continue                time.sleep(random.randint(4, 10))                url1 = "https://search.weixin.qq.com/cgi-bin/recwxa/recwxavideolist?"                header = {                    'Host': 'search.weixin.qq.com',                    'Content-Type': 'application/json',                    'X-WX-ClientVersion': '0x33050520',                    'X-WECHAT-UIN': 'b2hfbTQ1WGNjSzQxemdfanpMSml1TEtfbEtsVQ==',                    'Accept': '*/*',                    'User-Agent': 'Mozilla/5.0 (iPhone; CPU iPhone OS 11_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E217 MicroMessenger/6.8.0(0x16080000) NetType/WIFI Language/en Branch/Br_trunk MiniProgramEnv/Mac',                    'Referer': 'https://servicewechat.com/wxbb9a805eb4f9533c/269/page-frame.html',                    'Accept-Language': 'zh-cn'                }                params = {                    "session": session,                    "offset": 0,                    "count": "30",                    "channelid": "200201",                    "vid": vid,                    "scene": "310",                    "subscene": '1098'                }                urllib3.disable_warnings()                response = requests.get(url=url1, headers=header, params=params, proxies=proxies, verify=False)                if "data" not in response.text:                    Common.logger(log_type, crawler).info("获取视频list时,session过期,随机睡眠 31-50 秒")                    Common.logging(log_type, crawler, env, "获取视频list时,session过期,随机睡眠 31-50 秒")                    AliyunLogger.logging(                        code="2000",                        platform=crawler,                        mode=log_type,                        env=env,                        message=f"获取视频list时,session过期,随机睡眠 31-50 秒"                    )                    # 如果返回空信息,则随机睡眠 31-40 秒                    time.sleep(random.randint(31, 40))                    cls.get_videoList(log_type, crawler, our_uid, rule_dict, env)                elif "items" not in response.json()["data"]:                    Common.logger(log_type, crawler).info(f"get_feeds:{response.json()},随机睡眠 1-3 分钟")                    Common.logging(log_type, crawler, env, f"get_feeds:{response.json()},随机睡眠 1-3 分钟")                    AliyunLogger.logging(                        code="2000",                        platform=crawler,                        mode=log_type,                        env=env,                        message=f"get_feeds:{response.json()},随机睡眠 1-3 分钟"                    )                    # 如果返回空信息,则随机睡眠 1-3 分钟                    time.sleep(random.randint(60, 180))                    cls.get_videoList(log_type, crawler, our_uid, rule_dict, env)                feeds = response.json().get("data", {}).get("items", "")                if feeds == "":                    Common.logger(log_type, crawler).info(f"feeds:{feeds}")                    Common.logging(log_type, crawler, env, f"feeds:{feeds}")                    AliyunLogger.logging(                        code="2001",                        platform=crawler,                        mode=log_type,                        env=env,                        message=f"没有更多视频啦 ~\n"                    )                    return                for j in range(len(feeds)):                    try:                        AliyunLogger.logging(                            code="1001",                            platform=crawler,                            mode=log_type,                            env=env,                            message='扫描到一条视频\n'                        )                        video_title = feeds[j].get("title", "").strip().replace("\n", "") \                            .replace("/", "").replace("\\", "").replace("\r", "") \                            .replace(":", "").replace("*", "").replace("?", "") \                            .replace("?", "").replace('"', "").replace("<", "") \                            .replace(">", "").replace("|", "").replace(" ", "") \                            .replace("&NBSP", "").replace(".", "。").replace(" ", "") \                            .replace("'", "").replace("#", "").replace("Merge", "")                        publish_time_stamp = feeds[j].get("date", 0)                        publish_time_str = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(publish_time_stamp))                        # 获取播放地址                        if "videoInfo" not in feeds[j]:                            video_url = ""                        elif "mpInfo" in feeds[j]["videoInfo"]["videoCdnInfo"]:                            if len(feeds[j]["videoInfo"]["videoCdnInfo"]["mpInfo"]["urlInfo"]) > 2:                                video_url = feeds[j]["videoInfo"]["videoCdnInfo"]["mpInfo"]["urlInfo"][2]["url"]                            else:                                video_url = feeds[j]["videoInfo"]["videoCdnInfo"]["mpInfo"]["urlInfo"][0]["url"]                        elif "ctnInfo" in feeds[j]["videoInfo"]["videoCdnInfo"]:                            video_url = feeds[j]["videoInfo"]["videoCdnInfo"]["ctnInfo"]["urlInfo"][0]["url"]                        else:                            video_url = feeds[j]["videoInfo"]["videoCdnInfo"]["urlInfo"][0]["url"]                        video_id = feeds[j].get("videoId", "")+"feed"                        s_cnt = int(feeds[j].get("shared_cnt", 0))                        p_count = int(feeds[j].get("playCount", 0))                        l_cnt = int(feeds[j].get("liked_cnt", 0))                        if p_count < 100000:                            if s_cnt < 200 and p_count < 15000:                                Common.logger(log_type, crawler).info(f"分享:{video_percent},播放量:{playCount}\n")                                Common.logging(log_type, crawler, env, f"分享:{video_percent},播放量:{playCount}\n")                                continue                        video_dict = {                            "video_title": video_title,                            "video_id": video_id,                            "play_cnt": feeds[j].get("playCount", 0),                            "like_cnt": feeds[j].get("liked_cnt", 0),                            "comment_cnt": feeds[j].get("comment_cnt", 0),                            "share_cnt": feeds[j].get("shared_cnt", 0),                            "duration": feeds[j].get("mediaDuration", 0),                            "video_width": feeds[j].get("short_video_info", {}).get("width", 0),                            "video_height": feeds[j].get("short_video_info", {}).get("height", 0),                            "publish_time_stamp": publish_time_stamp,                            "publish_time_str": publish_time_str,                            "user_name": feeds[j].get("source", "").strip().replace("\n", ""),                            "user_id": feeds[j].get("openid", ""),                            "avatar_url": feeds[j].get("bizIcon", ""),                            "cover_url": feeds[j].get("thumbUrl", ""),                            "video_url": video_url,                            "session": session,                        }                        for k, v in video_dict.items():                            Common.logger(log_type, crawler).info(f"{k}:{v}")                        Common.logging(log_type, crawler, env, f"video_dict:{video_dict}")                        AliyunLogger.logging(                            code="1000",                            platform=crawler,                            mode=log_type,                            env=env,                            message=f"{video_dict}\n"                        )                        if video_dict["video_id"] == "" or video_dict["video_title"] == "" or video_dict[                            "video_url"] == "":                            Common.logger(log_type, crawler).info("无效视频\n")                            Common.logging(log_type, crawler, env, "无效视频\n")                            AliyunLogger.logging(                                code="2004",                                platform=crawler,                                mode=log_type,                                env=env,                                message='无效视频\n'                            )                        elif download_rule(log_type=log_type, crawler=crawler, video_dict=video_dict,                                           rule_dict=rule_dict) is False:                            Common.logger(log_type, crawler).info("不满足抓取规则\n")                            Common.logging(log_type, crawler, env, "不满足抓取规则\n")                            AliyunLogger.logging(                                code="2004",                                platform=crawler,                                mode=log_type,                                env=env,                                message='不满足抓取规则\n'                            )                        elif any(str(word) if str(word) in video_dict["video_title"] else False                                 for word in get_config_from_mysql(log_type=log_type,                                                                   source=crawler,                                                                   env=env,                                                                   text="filter",                                                                   action="")) is True:                            Common.logger(log_type, crawler).info('已中过滤词\n')                            Common.logging(log_type, crawler, env, '已中过滤词\n')                            AliyunLogger.logging(                                code="2004",                                platform=crawler,                                mode=log_type,                                env=env,                                message='已中过滤词\n'                            )                        elif cls.repeat_video(log_type, crawler, video_dict["video_id"], env) != 0:                            Common.logger(log_type, crawler).info('视频已下载\n')                            Common.logging(log_type, crawler, env, '视频已下载\n')                            AliyunLogger.logging(                                code="2002",                                platform=crawler,                                mode=log_type,                                env=env,                                message='视频已下载\n'                            )                        else:                            video_dict["out_user_id"] = video_dict["user_id"]                            video_dict["platform"] = crawler                            video_dict["strategy"] = log_type                            video_dict["strategy_type"] = "hcm"                            video_dict["out_video_id"] = video_dict["video_id"]                            video_dict["width"] = video_dict["video_width"]                            video_dict["height"] = video_dict["video_height"]                            video_dict["crawler_rule"] = json.dumps(rule_dict)                            video_dict["user_id"] = our_uid                            video_dict["publish_time"] = video_dict["publish_time_str"]                            mq.send_msg(video_dict)                    except Exception as e:                        Common.logger(log_type, crawler).error(f"抓取单条视频异常:{e}\n")                        Common.logging(log_type, crawler, env, f"抓取单条视频异常:{e}\n")                        AliyunLogger.logging(                            code="3000",                            platform=crawler,                            mode=log_type,                            env=env,                            message=f"抓取单条视频异常:{e}\n"                        )        except Exception as e:            Common.logger(log_type, crawler).error(f"抓取列表页时异常:{e}\n")            Common.logging(log_type, crawler, env, f"抓取列表页时异常:{e}\n")            AliyunLogger.logging(                code="3000",                platform=crawler,                mode=log_type,                env=env,                message=f"抓取列表页时异常:{e}\n"            )if __name__ == "__main__":    KanyikanRecommend.get_videoList(        log_type="recommend",        crawler="kanyikan",        env="prod",        rule_dict={'share_cnt': {'min': 300, 'max': 0}},        our_uid=64080779    )
 |