通过scrapy和Django登录、爬取和持久化数据

使用 ScrapyDjango 实现登录、爬取和持久化数据的完整流程,可以通过以下步骤完成:

  1. 创建 Django 项目和数据库模型:定义一个存储爬取数据的数据库模型。
  2. 创建 Scrapy 项目:实现登录并抓取目标页面的数据。
  3. 整合 Scrapy 和 Django:在 Scrapy 中使用 Django 的模型保存爬取的数据到数据库。

问题背景

在将 Django 和 Scrapy 成功集成后,用户想要持久化爬取到的数据到数据库中。但是,存储后发现,部分元素丢失了。用户猜测自己可能遗漏了一些东西,但无法解决。

以下是用户的爬虫代码:

python 复制代码
from scrapy.http import FormRequest, Request
from scrapy.spider import BaseSpider
from scrapy.selector import HtmlXPathSelector
from scrapy import log
from scrapy.contrib.loader import XPathItemLoader
from datacrowdscrapy.items import DatacrowdItem

class DatacrowdSpider(BaseSpider):
    name = 'datacrowd'
    start_urls = ['https://www.exemple.com/login']

    def parse(self, response):
        parsed = [FormRequest.from_response(
            response,
            formdata={
                'login': 'email@gmail.com',
                'password': 'password'
            },
            callback=self.after_login)]

        return parsed

    def after_login(self, response):
        # check login succeed before going on
        if "authentication failed" in response.body:
            self.log("Login failed", level=log.ERROR)
            return

        selector = HtmlXPathSelector(response)
        investmentsLinks = selector.select('//a[contains(@class, "myClass")]/@href').extract()
        for link in investmentsLinks:
            curDatacrowdItem = XPathItemLoader(item=DatacrowdItem(), response=response)
            curDatacrowdItem.add_value('url', link)
            curRequest = Request(url=link, callback=self.parse_investments, meta={'item': curDatacrowdItem})
            yield curRequest


    def parse_investments(self, response):
        selector = HtmlXPathSelector(response)
        curDatacrowdItem = response.meta['item']

        # Details
        details = selector.select('//td/div[contains(@class, "myClass")]/text()').extract()
        curDatacrowdItem.add_value('someVal', details[0].strip())
        /* ... */

        # Get nbInvestors
        investorLink = selector.select('//ul[contains(@id, "myId")]/li/@onclick').re(r'window.location.href=\'(http.+/data.+)\'')
        curRequest = Request(url=investorLink[0], callback=self.parse_investors, meta={'item': curDatacrowdItem})
        yield curRequest


        # Get last company details
        detailsLink = selector.select('//ul[contains(@id, "myData")]/li/@onclick').re(r'window.location.href=\'(http.+/company-details.+)\'')
        curRequest = Request(url=detailsLink[0], callback=self.parse_details, meta={'item': curDatacrowdItem})
        yield curRequest

    def parse_investors(self, response):
        selector = HtmlXPathSelector(response)
        curDatacrowdItem = response.meta['item']
        nbInvestors = len(selector.select('//ul/li[contains(@class, "myClass")]'))
        curDatacrowdItem.add_value('nbInvestors', nbInvestors)
        return curDatacrowdItem

    def parse_details(self, response):
        selector = HtmlXPathSelector(response)
        curDatacrowdItem = response.meta['item']

        # Company name
        name = selector.select('//div[contains(@class, "myClass")]/h2/text()').extract()
        curDatacrowdItem.add_value('name', name[0].strip())
        item = curDatacrowdItem.load_item()
        item.save() # Here I'm persisiting datas
        return item

用户收到的错误日志如下:

复制代码
[datacrowd] ERROR: Spider must return Request, BaseItem or None, got 'XPathItemLoader' in <GET http://www.exemple.com/url/slug>

解决方案

用户犯的错误是,他正在返回一个 XPathItemLoader 对象,而不是一个 Item 对象。在 "after_login" 方法中,用户将一个 XPathItemLoader 对象添加到 meta 中,然后尝试在稍后返回它。正确的做法是使用 load_item 方法来返回 Item 对象。

要解决这个问题,用户可以将以下代码添加到 "after_login" 方法中:

python 复制代码
curRequest = Request(url=link, callback=self.parse_investments, meta={'item': curDatacrowdItem.load_item()})

另外,建议用户重命名变量,以避免类似的错误。

总结

这段代码展示了如何结合 ScrapyDjango 登录、抓取和持久化数据的基本流程。这个组合适用于需要在 Web 项目中自动抓取并存储数据的需求,如商品数据爬取。

相关推荐
≮傷£≯√3 分钟前
动态创建combobox
数据库
skywalk81635 分钟前
段言的设计文档:中文编程赛道的竞争格局,谁在牌桌上?
开发语言·学习·编程
就叫_这个吧6 分钟前
HTML或JSP页面链接CSS,link标签没问题,但不显示样式问题解决
java·前端·css·html·intellij-idea·jsp
阿正的梦工坊6 分钟前
【Rust】03-所有权、移动与复制
开发语言·算法·rust
yi念zhi间8 分钟前
C#实现控制台多区域输出
开发语言·c#
阿坤带你走近大数据10 分钟前
分别介绍下java主流的开发框架、设计模式与对应编程语言的高级特性
java·开发语言·设计模式
小小龙学IT11 分钟前
Go 后端开发中的并发模式:从 Goroutine 到 Pipeline 实战
开发语言·后端·golang
摇滚侠12 分钟前
Spring 零基础入门到进阶 基于 XML 的声明式事务 71
xml·数据库·spring
小短腿的代码世界12 分钟前
Qt文本布局引擎深度解析:从QTextDocument排版到渲染的完整架构
开发语言·qt·架构
Leweslyh15 分钟前
《3GPP TS 28.312 面向移动网络的意图驱动管理服务》完整自学教程
开发语言·网络·php