爬虫采集去重优化探讨
优采云 发布时间: 2020-08-13 10:52以前在做漏洞Fuzz爬虫时,曾做过URL去重相关的工作,当时是参考了seay牧师的文章以及网上零碎的一些资料,感觉做的很简单。近来又遇见相关问题,于是乎有了再度改进算法的念头。
首先,针对URL本身的去重,可以直接对整块URL进行处理。在参考网上的一些文章时,发现它们大多采用了 URL 压缩储存的方式。不过使用这种算法在数据量较大的时侯,能急剧减少储存的空间:
基于c盘的次序储存。
基于Hash算法的储存。
基于MD5压缩映射的储存。
基于嵌入式Berkeley DB的储存。
基于布隆过滤器(Bloom Filter)的储存。
对于 URL 直接去重,主要涉及的是储存优化方面,对于本文不是重点,这里不再细说。
而对于 URL 逻辑上的去重,则须要更多地追求数据的可用性,这是做测试工作须要去审视的。
这里先给出 seay 文章中的相似度去重算法,大致是下边这样的:
def urlsimilar(url):
hash_size=199999
tmp=urlparse.urlparse(url)
scheme=tmp[0]
netloc=tmp[1]
path=tmp[2][1:]
query=tmp[4]
#First get tail
if len(path.split('/'))>1:
tail=path.split('/')[-1].split('.')[-1]
#print tail
elif len(path.split('/'))==1:
tail=path
else:
tail='1'
#Second get path_length
path_length=len(path.split('/'))-1
#Third get directy list except last
path_list=path.split('/')[:-1]+[tail]
#Fourth hash
path_value=0
for i in range(path_length+1):
if path_length-i==0:
path_value+=hash(path_list[path_length-i])%98765
else:
path_value+=len(path_list[path_length-i])*(10**(i+1))
#get host hash value
netloc_value=hash(hashlib.new("md5",netloc).hexdigest())%hash_size
url_value=hash(hashlib.new("md5",str(path_value+netloc_value)).hexdigest())%hash_size
return url_value
这段函数的大约作用是,最后它会依照算法返回一个hash值,这个hash值也就是该URL的hash相似度。如果两个URL估算出的hash值最后比较相等,我们则可以判定两个URL是具有较高的相似度的。
但是这个函数应当是seay举例时随手提出的(这里指出下,免得被喷,后文不再细说),只是简单做了demo,并没有进行细化检验。在比较粗糙的情况下,该算法确实能剔除一些简单的参数重复的情况,但一旦参数复杂或者url不规范,是不太能挺好的进行去重的。
那么在针对URL获取的过程中,我们还可以做的小优化有什么呢?
日期时间命名
首先,我们可以依据日期来去重。我们晓得,在爬取一些Blog和和门户等系统时,经常会遇见以日期命名的目录。
这些目录大约归纳上去,存在类似下边的方式:
2010-11-11
10-11-11
20101111
当然,还有些文件会以时间+随机值命名,也可能是用unix时间戳命名,这些可能是依据上传和编辑时间来定义的。
笔者建议是,使用redis或则memcache之类等缓存型数据库,将其直接储存;或者在数据量较大的时侯,考虑将其作临时储存,需要的时侯再进行对比。
比如,一旦出现日期时间命名的目录或静态文件,我们可以考虑将其储存为下边的格式:
目录层级
命名格式
URL地址(或压缩过的hash值)
有人可能说,在上面seay提出的那种案例里,好像是可以解决类似日期相似度的问题。那我们先瞧瞧下边的事例,此处输出依旧基于前面哪个函数:
print urlsimilar('http://www.baidu.com/blog/2010-10-11/')
print urlsimilar('http://www.baidu.com/blog/2010-10-13/')
print urlsimilar('http://www.baidu.com/blog/2010-9-13/')
print urlsimilar('http://www.baidu.com/whisper/2010-10-11/')
输出结果如下:
110086
110086
37294
4842
我们可以看见,在普通情况下,确实于相同父级目录下,相似度算法是可以判定正确的。 但是一旦日期格式不规范,或者父级目录存在一定的差别,这里是不能挺好的判定的。
当然,我们也可以通过机器学习来完成去重的工作。不过就简化工作而言,还是可以使用一些小Tips,根据规则匹配来做到。
静态文件的去重
我们晓得,在爬取URL的过程中,也会碰到许多静态文件,如shtml、html、css等等。这些文件在大多数的情况下,是没有很大意义的。除非测试者倾向于使用“宁可错杀一百,绝不放过一个”的全量采集手法。
这时候,我们可以配置黑名单,建立文件后缀规则库进行过滤。
当然,在这种静态后缀的URL链接,也可能带上参数混淆的情况。 个人建议是,用于反弹的json、xml等URL,里面可能存储敏感内容,尽量别动;其他类型的静态文件,仍然采取将参数分离的方法,最后对URL进行去重储存。
特定情况的过滤
在爬取特定网站时,我们可以预先做好配置,指定过滤一些目录和页面,以节约大量时间资源。
反过来,我们也可以指定只爬取指定目录下的页面,定向获取我们想要的内容。
敏感页面的感知
在上面seay提出的demo算法中,在这些情况下是有一定局限的。比如我们须要在敏感目录下,尽可能多的领到文件信息。比如我们爬取到了后台管理目录,可能会遇见下边的情况:
print urlsimilar('http://www.baidu.com/blog/admin/login.php')
print urlsimilar('http://www.baidu.com/blog/admin/manage_index.php')
print urlsimilar('http://www.baidu.com/blog/admin/test.css')
输出结果如下:
40768
40768
40768
很明显有问题不是么?
当然,我们可以通过对敏感页面关键词进行监控;或者也可以指定后缀文件,进行白名单监控。
但是一旦这样做,而且还想采用上面的hash算法的话,大家自行定义的过滤函数的优先级,肯定须要小于该算法。并且,我们在这样做的过程中,也应当考虑过滤成本的问题,建议采用选择性启用。
高频敏感目录的优待
可能在爬取的过程中,部分爬虫是兼用了目录爆破的手段的。如果采用了这些手法而且匹配成功后,我们可以将该目录下的内容单独使用一份过滤规则,从而防止去重算法的错判。
响应页面的过滤
对于个别网站来讲,可能有不少页面由于链接是失效的,会被冠以404页面和50x错误。另外,在无权访问的时侯,可能网站会做30x跳转和403目录限制。
这些页面没有实质性内容,在大多数时侯是没有意义的,我们可以在配置文件里对须要爬取的这类页面做白名单,比如保留403页面,或者存取30x跳转前(后)的页面。
WAF(警告)页面过滤
某些网站可能被装上了WAF,在访问频度过快时,可能会得到一个WAF的警告页面。而在CMS本身就做了限制的情况下,会以20x的响应码展示一些没有不存在的页面。
当然,我们可以通过分布式换代理的方法,去解决部份这样的问题,这里先不多做讨论。
这时候,我们可以配置相应的次数阀值,如果个别页面出现的次数过多,可以将其标记为警告(WAF)页面,进而作出过滤处理。这里对某页面的辨识,可以通过黑名单关键字标记;或者尝试估算页面hash值,比如下边这样:
content = urllib2.urlopen('http://www.test.com/').read()
md5_sum = hashlib.md5()
md5_sum.update(content)
print md5_sum.hexdigest()
当然,我们在实际估算页面hash值和做关键字监控时,也可能因为反爬虫机制的存在(如添加随机值),需要适时调整相似度来估算hash值或则采用其他手段。当然这也会消耗更多的时间和机器资源。但个别特定的情况下,可能也会带来意想不到的收获。
无意义参数页面去重
我们在采集页面的过程中,同样有可能会碰到一些毫无意义的、高频出现的多参数页面。这类页面可能是反弹页面,也可能是临时渲染的随机页面。
在这里,大家可以通过上面处理WAF(警告)的方式进行过滤。当然,使用上面的hash算法也是可以应对大部分情况的。毕竟网站的这类的URL有限,不必为了几种特型去消耗更多的资源,这样得不偿失。
JS代码中的URL
在我们提取js代码,也就是遇见ajax之类的交互情况时,可能会遇见须要拼接的GET恳求,或者直接可以取用的POST恳求。
这类的URL地址,最好是结合phantomjs等webkit,更方便地进行动态拼接。
它们会变得比较特殊,可能仅仅返回状态码,也可能会返回实质性的敏感内容。这种情况,就须要依照爬取者的要求,对爬取的过滤规则进行适应性调整。
总结
笔者这儿致力提出一些对相像URL去重的小优化,可能疗效有限,也可能存在未尽人意之处。
欢迎你们提出建议,希望少一些爱喷的童鞋,多一点讨论的大牛,与诸位自勉。
参考文章
如何防止重复抓取同一个网页
浅谈动态爬虫与去重
网络爬虫:URL去重策略之布隆过滤器(BloomFilter)的使用
实用科普:爬虫技术探讨 编写爬虫应注意的点
网络爬虫 (spider) URL消重设计 URL去重设计