#!/usr/bin/python3
# -*- coding: utf-8 -*-
# @Time : 2022/4/15 12:06
# @Author : shaofei
# @Email : shaochenshuo@126.com
# @File : 内核月报搜索exe版.py
# @Software: PyCharm
#pip3 install beautifulsoup4
#pip3 install lxml
#pip install requests
#import urllib2
import urllib.request
import requests
import threading
import argparse
import re
import os
import sys
import datetime
import time
from bs4 import BeautifulSoup
url = "http://mysql.taobao.org/monthly"
#加上请求头,模拟成浏览器取访问网站,避免请求头反爬策略:HTTP Error 418,注意headers是dic
USER_AGENT = {'user-agent': 'Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.75 Mobile Safari/537.36'}
URL_TIMEOUT = 15
def get_input():
key_str = input('请输入关键字,最多支持3个关键字,多个关键字间用逗号分隔:')
key = []
key_relation = 0
key = key_str.split(',')
if 1 < len(key) <= 3:
key_relation = input('\n指定多个过滤关键字时,需要同时指定关键字间关系(请输入or 或者 and):')
print('\n')
elif len(key) == 1:
if key[0] == '':
print('你还未指定搜索关键字,请指定!!!')
key, key_relation = get_input()
else:
#print('本次过滤指定的关键字为:{}'.format(key[0]))
key_relation = 0
else:
print('\nERROR:最多只支持指定三个关键字')
sys.exit()
return key, key_relation
def parameter_parse():
# 解析命令行参数(-k 指定过滤的关键字,-l指定多个关键字间的关系)
parse = argparse.ArgumentParser()
parse.add_argument('-k', '--key', nargs='+', type=str, default=None,
help='指定搜索的关键字,必填参数,支持指定多个关键字,多个关键字间以空格分隔', required=True)
parse.add_argument('-l', '--relation', type=str, choices=['or', 'and'], help='多个关键词之间的关系 or或者and')
args = parse.parse_args()
# print(args)
# print(args.key[0])
if 1 < len(args.key) <= 3:
if args.relation:
print('本次过滤指定的关键字列表为:{},关键字间关系为:{}'.format(args.key, args.relation))
else:
print('当你使用 -k 指定多个关键字时,必须使用 -l 指定关键字间的关系(and 或者 or)')
sys.exit()
elif len(args.key) == 1:
print('本次过滤指定的关键字为:{}'.format(args.key[0]))
else:
print(' -k 最多只支持指定三个关键字')
sys.exit()
return args
def get_html(url, USER_AGENT, URL_TIMEOUT):
"""
访问url返回html
"""
response = requests.get(url, headers=USER_AGENT, timeout=URL_TIMEOUT)
response.encoding = response.apparent_encoding
#text = response.text
html = response.content.decode('utf-8')
#print(text)
return html
def call_gethtml(url, USER_AGENT, URL_TIMEOUT):
try:
html = get_html(url, USER_AGENT, URL_TIMEOUT)
except Exception as e:
error = str(e)
if "Connection to mysql.taobao.org timed out" in error:
print('访问网页超时,重新访问')
html = get_html(url, USER_AGENT, URL_TIMEOUT)
else:
print('访问网页报错,报错代码如下:{}'.format(e))
return html
def get_month_list():
"""
用户生成淘宝内核月报,每月url 列表
"""
html = call_gethtml(url, USER_AGENT, URL_TIMEOUT)
#通过BeautifulSoup对html进行格式化,便于访问
soup = BeautifulSoup(html, 'lxml')
h3_list = soup.find_all('h3')
#上面的 soup 是 'bs4.element.Tag' 类型,下面的 h3 也是 'bs4.element.Tag'类型
month_url = []
for h3 in h3_list:
#h3.a['href'] 访问h3的a标签,h3.a.string 获取a标签的文本内容(如果有多个a标签,下面获取的是第一个a标签)
child_list = '{}{}'.format('http://mysql.taobao.org', h3.a['href'])
#print('链接为:{}, 链接标题为:{}'.format(child_list, h3.a.string))
month_url.append(child_list)
#print('月报列表长度为:{}。月报列表如下:{}'.format(len(month_url), month_url))
return month_url
def key_check(url_month, key: list, relation = 0):
"""
匹配标题,正文,代码中的关键字,匹配成功输出文章标题及url
:param url_month: 如http://mysql.taobao.org/monthly/2022/03/
:param key:
:return:
"""
#访问某月的内核月报列表(如http://mysql.taobao.org/monthly/2022/03/)
html_title = call_gethtml(url_month, USER_AGENT, URL_TIMEOUT)
soup = BeautifulSoup(html_title, 'lxml')
article_h3_list = soup.find_all('h3')
for h3_in in article_h3_list:
#指定3个关键字参数时对月报进行过滤
if len(key) == 3:
#对所有的关键字进行标题,正文,代码匹配,只要有一个关键字匹配上,则匹配成功
if relation == 'or':
url_page = '{}{}'.format('http://mysql.taobao.org', h3_in.a['href'])
url_string = h3_in.a.string.strip()
# 访问具体某篇内核月报(如http://mysql.taobao.org/monthly/2022/03/01/)
html_page = call_gethtml(url_page, USER_AGENT, URL_TIMEOUT)
soup_page = BeautifulSoup(html_page, 'lxml')
if (h3_in.find('a', text=re.compile(key[0], re.I)) or soup_page.find_all('p', text=re.compile(key[0],
re.I)) or soup_page.find_all(
'code', text=re.compile(key[0], re.I))
or h3_in.find('a', text=re.compile(key[1], re.I)) or soup_page.find_all('p', text=re.compile(key[1],
re.I)) or soup_page.find_all(
'code', text=re.compile(key[1], re.I))
or h3_in.find('a', text=re.compile(key[2], re.I)) or soup_page.find_all('p', text=re.compile(key[2],
re.I)) or soup_page.find_all(
'code', text=re.compile(key[2], re.I))):
print('{},链接地址为:{}'.format(url_string, url_page))
article_url['{}'.format(url_string)] = url_page
else:
pass
#对多个关键字同时进行标题,正文,代码过滤,几个关键字同时匹配上,则匹配成功
if relation == 'and':
url_page = '{}{}'.format('http://mysql.taobao.org', h3_in.a['href'])
url_string = h3_in.a.string.strip()
# 访问具体某篇内核月报(如http://mysql.taobao.org/monthly/2022/03/01/)
html_page = call_gethtml(url_page, USER_AGENT, URL_TIMEOUT)
soup_page = BeautifulSoup(html_page, 'lxml')
if (h3_in.find('a', text=re.compile(key[0], re.I)) or soup_page.find_all('p', text=re.compile(key[0], re.I)) or soup_page.find_all('code', text=re.compile(key[0], re.I))
) and (h3_in.find('a', text=re.compile(key[1], re.I)) or soup_page.find_all('p', text=re.compile(key[1], re.I)) or soup_page.find_all('code', text=re.compile(key[1], re.I))
) and (h3_in.find('a', text=re.compile(key[2], re.I)) or soup_page.find_all('p', text=re.compile(key[2], re.I)) or soup_page.find_all('code', text=re.compile(key[2], re.I))):
print('{},链接地址为:{}'.format(url_string, url_page))
article_url['{}'.format(url_string)] = url_page
else:
pass
#指定两个关键字参数时,对月报进行过滤
elif len(key) == 2:
# 只要标题,正文或者代码中有一处匹配上第一个或是第二个关键字,则匹配成功
if relation == 'or':
url_page = '{}{}'.format('http://mysql.taobao.org', h3_in.a['href'])
url_string = h3_in.a.string.strip()
# 访问具体某篇内核月报(如http://mysql.taobao.org/monthly/2022/03/01/)
html_page = call_gethtml(url_page, USER_AGENT, URL_TIMEOUT)
soup_page = BeautifulSoup(html_page, 'lxml')
if (h3_in.find('a', text=re.compile(key[0], re.I)) or soup_page.find_all('p', text=re.compile(key[0], re.I)) or
soup_page.find_all('code', text=re.compile(key[0], re.I)) or h3_in.find('a', text=re.compile(key[1], re.I))
or soup_page.find_all('p', text=re.compile(key[1], re.I)) or soup_page.find_all('code', text=re.compile(key[1], re.I))):
article_url['{}'.format(url_string)] = url_page
print('{},链接地址为:{}'.format(url_string, url_page))
# 对多个关键字同时进行标题,正文,代码过滤,几个关键字同时匹配上,则匹配成功
if relation == 'and':
url_page = '{}{}'.format('http://mysql.taobao.org', h3_in.a['href'])
url_string = h3_in.a.string.strip()
# 访问具体某篇内核月报(如http://mysql.taobao.org/monthly/2022/03/01/)
html_page = call_gethtml(url_page, USER_AGENT, URL_TIMEOUT)
soup_page = BeautifulSoup(html_page, 'lxml')
if (h3_in.find('a', text=re.compile(key[0], re.I)) or soup_page.find_all('p', text=re.compile(key[0], re.I)) or soup_page.find_all('code', text=re.compile(key[0], re.I))
) and (h3_in.find('a', text=re.compile(key[1], re.I)) or soup_page.find_all('p', text=re.compile(key[1], re.I)) or soup_page.find_all('code', text=re.compile(key[1], re.I))):
article_url['{}'.format(url_string)] = url_page
print('{},链接地址为:{}'.format(url_string, url_page))
else:
pass
#只指定一个关键字参数时对月报进行过滤
else:
#过滤标题,正文,代码中是否有相关关键字,如果存在,则匹配成功
url_page = '{}{}'.format('http://mysql.taobao.org', h3_in.a['href'])
url_string = h3_in.a.string.strip()
# 访问具体某篇内核月报(如http://mysql.taobao.org/monthly/2022/03/01/)
html_page = call_gethtml(url_page, USER_AGENT, URL_TIMEOUT)
soup_page = BeautifulSoup(html_page, 'lxml')
if h3_in.find('a', text=re.compile(key[0], re.I)) or soup_page.find_all('p', text=re.compile(key[0], re.I)) or soup_page.find_all('code', text=re.compile(key[0], re.I)):
url_page = '{}{}'.format('http://mysql.taobao.org', h3_in.a['href'])
url_string = h3_in.a.string.strip()
print('{},链接地址为:{}'.format(url_string, url_page))
article_url['{}'.format(url_string)] = url_page
continue
else:
pass
if __name__ == '__main__':
# args = parameter_parse()
# key = args.key
# if len(key) > 1:
# key_relation = args.relation
# key_str = input('请输入关键字,最多支持3个关键字,多个关键字间用逗号分隔:')
# key = key_str.split(',')
# if 1 < len(key) <= 3:
# key_relation = input('\n指定多个过滤关键字时,需要同时指定关键字间关系(请输入or 或者 and):')
# print('\n')
# elif len(key) == 1:
# print('本次过滤指定的关键字为:{}'.format(key[0]))
# key_relation = 0
# elif len(key) == 0:
# print('你还未指定搜索关键字,程序自动退出')
# else:
# print('\nERROR:最多只支持指定三个关键字')
# sys.exit()
key, key_relation = get_input()
print('关键字为:{},关系为:{}'.format(key,key_relation))
#这里不比考虑python 中字典是否线程安全,因为我的并发线程只会对自字典增加不同key:value 键值对
article_url = {}
i_parallel = 2
month_list = get_month_list()
list_count = len(month_list)
operate_count = 1
print('开始搜索:{} ......'.format(datetime.datetime.now()))
if list_count > 1:
while operate_count <= list_count:
threads = []
for i in range(i_parallel):
t_check = threading.Thread(target=key_check, args=(month_list[operate_count - 1], key, key_relation))
threads.append(t_check)
operate_count += 1
if operate_count > (list_count):
break
for s in threads:
s.start()
for j in threads:
j.join()
print('当前过滤分析进度为 ---------------------{}%---------------------'.format(round(((operate_count - 1)/list_count)*100, 2)))
else:
key_check(month_list[0], key)
print('搜索结束:{}'.format(datetime.datetime.now()))
print('\n最终检查结果为:{}'.format(article_url))
filename = '搜索结果_{}.txt'.format(datetime.datetime.strftime(datetime.datetime.now(), "%Y%m%d%H%M%S"))
with open(filename, 'a', encoding='utf-8') as fileobj:
title = article_url.keys()
for i in title:
fileobj.writelines('{}:{}'.format(i, os.linesep))
fileobj.writelines('{0}{1}{1}'.format(article_url[i], os.linesep))
print('\n 请注意结果保存在当前目录的\'{}\'文件中,该窗口将在5分钟后自动关闭!!!'.format(filename))
time.sleep(300)
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)