摘要:前言自从之前爬取后公司要求对进行爬取,瞬间心中有一万只。毕竟这些社交网络的站点反爬机制做的很不错。但既然上面安排下来只能硬着头皮上了。通过抓包,发现登陆站点的数据相比要简单所有就写了一套利用爬取的爬虫。
前言
自从之前爬取twitter后公司要求对fancebook进行爬取,瞬间心中有一万只×××。毕竟这些社交网络的站点反爬机制做的很不错。但既然上面安排下来只能硬着头皮上了。通过抓包,发现登陆m.facebook.com站点psot的数据相比facebook.com要简单,所有就写了一套利用scrapy爬取facebook的爬虫。
模拟登陆from scrapy import Spider from scrapy.http import Request, FormRequest class FacebookLogin(Spider): download_delay = 0.5 usr = "××××" # your username/email/phone number pwd = "××××" #account password def start_requests(self): return [Request("https://m.facebook.com/", callback=self.parse)] def parse(self, response): return FormRequest.from_response(response, formdata={ "email": self.usr, "pass": self.pwd }, callback=self.remember_browser) def remember_browser(self, response): # if re.search(r"(checkpoint)", response.url): # Use "save_device" instead of "dont_save" to save device return FormRequest.from_response(response, formdata={"name_action_selected": "dont_save"}, callback=self.after_login) def after_login(self, response): pass
注:为了保险起见可以在seething文件中添加一个手机端的USER-AGENT
爬取用户基本信息# -*- coding: UTF-8 -*- import re from urlparse import urljoin from scrapy import Item, Field from scrapy.http import Request from scrapy.selector import Selector from facebook_login import FacebookLogin class FacebookItems(Item): id = Field() url = Field() name = Field() work = Field() education = Field() family = Field() skills = Field() address = Field() contact_info = Field() basic_info = Field() bio = Field() quote = Field() nicknames = Field() relationship = Field() image_urls = Field() class FacebookProfile(FacebookLogin): download_delay = 2 name = "fb" links = None start_ids = [ "plok74122", "bear.black.12","tabaco.wang","chaolin.chang.q","ahsien.liu","kaiwen.cheng.100","liang.kevin.92","bingheng.tsai.9","psppupu", "cscgbakery","hc.shiao.l","asusisbad","benjamin","franklin", # "RobertScoble" ] # "https://m.facebook.com/tabaco.wang?v=info","https://m.facebook.com/RobertScoble?v=info"] def after_login(self, response): for id in self.start_ids: url = "https://m.facebook.com/%s?v=info" %id yield Request(url, callback=self.parse_profile,meta={"id":id}) def parse_profile(self, response): item = FacebookItems() item["id"] = response.meta["id"] item["url"] = response.url item["name"] = "".join(response.css("#root strong *::text").extract()) item["work"] = self.parse_info_has_image(response, response.css("#work")) item["education"] = self.parse_info_has_image(response, response.css("#education")) item["family"] = self.parse_info_has_image(response, response.css("#family")) item["address"] = self.parse_info_has_table(response.css("#living")) item["contact_info"] = self.parse_info_has_table(response.css("#contact-info")) item["basic_info"] = self.parse_info_has_table(response.css("#basic-info")) item["nicknames"] = self.parse_info_has_table(response.css("#nicknames")) item["skills"] = self.parse_info_text_only(response.css("#skills")) item["bio"] = self.parse_info_text_only(response.css("#bio")) item["quote"] = self.parse_info_text_only(response.css("#quote")) item["relationship"] = self.parse_info_text_only(response.css("#relationship")) yield item def parse_info_has_image(self, response, css_path): info_list = [] for div in css_path.xpath("div/div[2]/div"): url = urljoin(response.url, "".join(div.css("div > a::attr(href)").extract())) title = "".join(div.css("div").xpath("span | h3").xpath("a/text()").extract()) info = " ".join(div.css("div").xpath("span | h3").xpath("text()").extract()) if url and title and info: info_list.append({"url": url, "title": title, "info": info}) return info_list def parse_info_has_table(self, css_path): info_dict = {} for div in css_path.xpath("div/div[2]/div"): key = "".join(div.css("td:first-child div").xpath("span | span/span[1]").xpath("text()").extract()) value = "".join(div.css("td:last-child").xpath("div//text()").extract()).strip() if key and value: if key in info_dict: info_dict[key] += ", %s" % value else: info_dict[key] = value return info_dict def parse_info_text_only(self, css_path): text = css_path.xpath("div/div[2]//text()").extract() text = [t.strip() for t in text] text = [t for t in text if re.search("w+", t) and t != "Edit"] return " ".join(text)爬取用户的所有图片
虽然图片在https://m.facebook.com/%s?v=info中会有显示,但是真正的图片链接却需要几次请求之后才能拿到,本作在spider中尽量少的操作原则故将抓取图片也多带带写成了一个爬虫,如下:
# -*- coding: UTF-8 -*- from scrapy.spider import CrawlSpider,Rule,Spider from scrapy.linkextractor import LinkExtractor from facebook_login import FacebookLogin from scrapy.http import Request from scrapy.selector import Selector from scrapy import Item, Field import re,hashlib import sys reload(sys) sys.setdefaultencoding("utf-8") class FacebookPhotoItems(Item): url = Field() id = Field() photo_links = Field() md5 = Field() class CrawlPhoto(FacebookLogin): name = "fbphoto" timelint_photo = None id = None links = [] start_ids = [ "plok74122", "bear.black.12", "tabaco.wang", "chaolin.chang.q", # "ashien.liu", "liang.kevin.92","qia.chen", "bingheng.tsai.9", "psppupu", "cscgbakery", "hc.shiao.l", "asusisbad", "benjamin", "franklin", # "RobertScoble" ] def after_login(self, response): for url in self.start_ids: yield Request("https://m.facebook.com/%s/photos"%url,callback=self.parse_item,meta={"id":url}) # yield Request("https://m.facebook.com/%s/photos"%self.id,callback=self.parse_item) def parse_item(self,response): # print response.body urls = response.xpath("//span").extract() next_page = None try: next_page = response.xpath("//div[@class="co"]/a/@href").extract()[0].strip() except: pass # urls = response.xpath("//div[@data-sigil="marea"]").extract() for i in urls: # if i.find(u"时间线照片")!=-1: try: self.timeline_photo = Selector(text=i).xpath("//span/a/@href").extract()[0] if self.timeline_photo is not None: yield Request("https://m.facebook.com/%s"%self.timeline_photo,callback=self.parse_photos,meta=response.meta) except: continue if next_page: print "-----------------------next image page -----------------------------------------" yield Request("https://m.facebook.com/%s"%next_page,callback=self.parse_item,meta=response.meta) def parse_photos(self,response): urls = response.xpath("//a[@class="bw bx"]/@href").extract() # urls = response.xpath("//a[@class="_39pi _4i6j"]/@href").extract() for i in urls: yield Request("https://m.facebook.com/%s"%i,callback=self.process_photo_url,meta=response.meta) if len(urls) == 12: next_page = response.xpath("//div[@id="m_more_item"]/a/@href").extract()[0] yield Request("https://m.facebook.com/%s"%next_page,callback=self.parse_photos,meta=response.meta) def process_photo_url(self,response): # photo_url = response.xpath("//i[@class="img img"]").extract() item = FacebookPhotoItems() item["url"] = response.url item["id"] = response.meta["id"] photo_url = response.xpath("//div[@style="text-align:center;"]/img/@src").extract()[0] item["photo_links"] = photo_url item["md5"] = self.getstr_md5(item["photo_links"])+".jpg" yield item def wirtefile(self,str): with open("temp2.html","w") as file: file.write(str) file.write(" ") def getstr_md5(self, input): if input is None: input = "" md = hashlib.md5() md.update(input) return md.hexdigest()
因为我的python水平也是半路出家,所有还没有找到一个好的办法将图片链接的抓取集成到抓取基本信息的那个爬虫上,如果有大神知道还请指点一二。
下载图片没有使用scrapy的imagePipline,而是使用的wget命令,原因就是上面所说,python水平太菜。。。
下面是自己写的一个下载图片的pipline:
class MyOwenImageDownload(object): def process_item(self, item,spider): if len(item) >6: pass else: file = "image/"+item["id"] if os.path.exists(file): pass else: os.makedirs(file) cmd = "wget "%s" -O %s -P %s --timeout=10 -q"%(item["photo_links"],file+"/"+item["md5"],file) os.system(cmd) return item结语
至此,整个爬虫基本的结构已经写完。。。源码地址
In the end, we will remember not the words of our enemies but the silence of our friends
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/38167.html
摘要:主要元素是身体内容,可以表示为。提取每个元素的文本并最终组成单个文本。我们将使用故意慢的服务器来显示这一点。是表示值的承诺的对象。我们将使用仓库中提供的准备示例作为示例。请注意,其余代码基本上不受影响除了返回函数中的源链接。 showImg(https://segmentfault.com/img/remote/1460000019190698?w=480&h=260); 来源 | ...
摘要:注意爬豆爬一定要加入选项,因为只要解析到网站的有,就会自动进行过滤处理,把处理结果分配到相应的类别,但偏偏豆瓣里面的为空不需要分配,所以一定要关掉这个选项。 本课只针对python3环境下的Scrapy版本(即scrapy1.3+) 选取什么网站来爬取呢? 对于歪果人,上手练scrapy爬虫的网站一般是官方练手网站 http://quotes.toscrape.com 我们中国人,当然...
摘要:负责处理被提取出来的。典型的处理有清理验证及持久化例如存取到数据库知识库项目的设置文件实现自定义爬虫的目录中间件是在引擎及之间的特定钩子,处理的输入和输出及。 【百度云搜索:http://www.bdyss.com】 【搜网盘:http://www.swpan.cn】 Scrapy框架安装 1、首先,终端执行命令升级pip: python -m pip install --upgrad...
摘要:组件引擎负责控制数据流在系统中所有组件中流动,并在相应动作发生时触发事件。下载器下载器负责获取页面数据并提供给引擎,而后提供给。下载器中间件下载器中间件是在引擎及下载器之间的特定钩子,处理传递给引擎的。 Scrapy 是用Python实现一个为爬取网站数据、提取结构性数据而编写的应用框架。 一、Scrapy框架简介 Scrapy是一个为了爬取网站数据,提取结构性数据而编写的应用框架。 ...
阅读 676·2023-04-26 02:03
阅读 1044·2021-11-23 09:51
阅读 1158·2021-10-14 09:42
阅读 1749·2021-09-13 10:23
阅读 972·2021-08-27 13:12
阅读 850·2019-08-30 11:21
阅读 1009·2019-08-30 11:14
阅读 1053·2019-08-30 11:09