A RetroSearch Logo

Home - News ( United States | United Kingdom | Italy | Germany ) - Football scores

Search Query:

Showing content from https://docs.apify.com/sdk/python/docs/guides/scrapy below:

Using Scrapy | SDK for Python

Using Scrapy

Scrapy is an open-source web scraping framework for Python. It provides tools for defining scrapers, extracting data from web pages, following links, and handling pagination. With the Apify SDK, Scrapy projects can be converted into Apify Actors, integrated with Apify storages, and executed on the Apify platform.

Integrating Scrapy with the Apify platform

The Apify SDK provides an Apify-Scrapy integration. The main challenge of this is to combine two asynchronous frameworks that use different event loop implementations. Scrapy uses Twisted for asynchronous execution, while the Apify SDK is based on asyncio. The key thing is to install the Twisted's asyncioreactor to run Twisted's asyncio compatible event loop. This allows both Twisted and asyncio to run on a single event loop, enabling a Scrapy spider to run as an Apify Actor with minimal modifications.

__main.py__: The Actor entry point

from __future__ import annotations

from scrapy.utils.reactor import install_reactor



install_reactor('twisted.internet.asyncioreactor.AsyncioSelectorReactor')

import os

from apify.scrapy import initialize_logging, run_scrapy_actor


from .main import main


os.environ['SCRAPY_SETTINGS_MODULE'] = 'src.settings'


if __name__ == '__main__':
initialize_logging()
run_scrapy_actor(main())

In this setup, apify.scrapy.initialize_logging configures an Apify log formatter and reconfigures loggers to ensure consistent logging across Scrapy, the Apify SDK, and other libraries. The apify.scrapy.run_scrapy_actor bridges asyncio coroutines with Twisted's reactor, enabling the Actor's main coroutine, which contains the Scrapy spider, to be executed.

Make sure the SCRAPY_SETTINGS_MODULE environment variable is set to the path of the Scrapy settings module. This variable is also used by the Actor class to detect that the project is a Scrapy project, triggering additional actions.

main.py: The Actor main coroutine

from __future__ import annotations

from scrapy.crawler import CrawlerRunner
from scrapy.utils.defer import deferred_to_future

from apify import Actor
from apify.scrapy import apply_apify_settings


from .spiders import TitleSpider as Spider


async def main() -> None:
"""Apify Actor main coroutine for executing the Scrapy spider."""
async with Actor:

actor_input = await Actor.get_input() or {}
start_urls = [url['url'] for url in actor_input.get('startUrls', [])]
allowed_domains = actor_input.get('allowedDomains')
proxy_config = actor_input.get('proxyConfiguration')


settings = apply_apify_settings(proxy_config=proxy_config)


crawler_runner = CrawlerRunner(settings)
crawl_deferred = crawler_runner.crawl(
Spider,
start_urls=start_urls,
allowed_domains=allowed_domains,
)
await deferred_to_future(crawl_deferred)

Within the Actor's main coroutine, the Actor's input is processed as usual. The function apify.scrapy.apply_apify_settings is then used to configure Scrapy settings with Apify-specific components before the spider is executed. The key components and other helper functions are described in the next section.

Key integration components

The Apify SDK provides several custom components to support integration with the Apify platform:

Additional helper functions in the apify.scrapy subpackage include:

Create a new Apify-Scrapy project

The simplest way to start using Scrapy in Apify Actors is to use the Scrapy Actor template. The template provides a pre-configured project structure and setup that includes all necessary components to run Scrapy spiders as Actors and store their output in Apify datasets. If you prefer manual setup, refer to the example Actor section below for configuration details.

Wrapping an existing Scrapy project

The Apify CLI supports converting an existing Scrapy project into an Apify Actor with a single command. The CLI expects the project to follow the standard Scrapy layout (including a scrapy.cfg file in the project root). During the wrapping process, the CLI:

For further details, see the Scrapy migration guide.

Example Actor

The following example demonstrates a Scrapy Actor that scrapes page titles and enqueues links found on each page. This example aligns with the structure provided in the Apify Actor templates.

from __future__ import annotations

from scrapy.utils.reactor import install_reactor



install_reactor('twisted.internet.asyncioreactor.AsyncioSelectorReactor')

import os

from apify.scrapy import initialize_logging, run_scrapy_actor


from .main import main


os.environ['SCRAPY_SETTINGS_MODULE'] = 'src.settings'


if __name__ == '__main__':
initialize_logging()
run_scrapy_actor(main())
from __future__ import annotations

from scrapy.crawler import CrawlerRunner
from scrapy.utils.defer import deferred_to_future

from apify import Actor
from apify.scrapy import apply_apify_settings


from .spiders import TitleSpider as Spider


async def main() -> None:
"""Apify Actor main coroutine for executing the Scrapy spider."""
async with Actor:

actor_input = await Actor.get_input() or {}
start_urls = [url['url'] for url in actor_input.get('startUrls', [])]
allowed_domains = actor_input.get('allowedDomains')
proxy_config = actor_input.get('proxyConfiguration')


settings = apply_apify_settings(proxy_config=proxy_config)


crawler_runner = CrawlerRunner(settings)
crawl_deferred = crawler_runner.crawl(
Spider,
start_urls=start_urls,
allowed_domains=allowed_domains,
)
await deferred_to_future(crawl_deferred)
BOT_NAME = 'titlebot'
DEPTH_LIMIT = 1
LOG_LEVEL = 'INFO'
NEWSPIDER_MODULE = 'src.spiders'
ROBOTSTXT_OBEY = True
SPIDER_MODULES = ['src.spiders']
TELNETCONSOLE_ENABLED = False

TWISTED_REACTOR = 'twisted.internet.asyncioreactor.AsyncioSelectorReactor'
HTTPCACHE_ENABLED = True
HTTPCACHE_EXPIRATION_SECS = 7200
from __future__ import annotations

from scrapy import Field, Item


class TitleItem(Item):
"""Represents a title item scraped from a web page."""

url = Field()
title = Field()
from __future__ import annotations

from typing import TYPE_CHECKING, Any
from urllib.parse import urljoin

from scrapy import Request, Spider

from ..items import TitleItem

if TYPE_CHECKING:
from collections.abc import Generator

from scrapy.http.response import Response


class TitleSpider(Spider):
"""A spider that scrapes web pages to extract titles and discover new links.

This spider retrieves the content of the <title> element from each page and queues
any valid hyperlinks for further crawling.
"""

name = 'title_spider'


custom_settings = {'CLOSESPIDER_PAGECOUNT': 10}

def __init__(
self,
start_urls: list[str],
allowed_domains: list[str],
*args: Any,
**kwargs: Any,
) -> None:
"""A default costructor.

Args:
start_urls: URLs to start the scraping from.
allowed_domains: Domains that the scraper is allowed to crawl.
*args: Additional positional arguments.
**kwargs: Additional keyword arguments.
"""
super().__init__(*args, **kwargs)
self.start_urls = start_urls
self.allowed_domains = allowed_domains

def parse(self, response: Response) -> Generator[TitleItem | Request, None, None]:
"""Parse the web page response.

Args:
response: The web page response.

Yields:
Yields scraped `TitleItem` and new `Request` objects for links.
"""
self.logger.info('TitleSpider is parsing %s...', response)


url = response.url
title = response.css('title::text').extract_first()
yield TitleItem(url=url, title=title)



for link_href in response.css('a::attr("href")'):
link_url = urljoin(response.url, link_href.get())
if link_url.startswith(('http://', 'https://')):
yield Request(link_url)
Dealing with ‘imminent migration to another host’

Under some circumstances, the platform may decide to migrate your Actor from one piece of infrastructure to another while it's in progress. While Crawlee-based projects can pause and resume their work after a restart, achieving the same with a Scrapy-based project can be challenging.

As a workaround for this issue (tracked as apify/actor-templates#303), turn on caching with HTTPCACHE_ENABLED and set HTTPCACHE_EXPIRATION_SECS to at least a few minutes—the exact value depends on your use case. If your Actor gets migrated and restarted, the subsequent run will hit the cache, making it fast and avoiding unnecessary resource consumption.

Conclusion

In this guide you learned how to use Scrapy in Apify Actors. You can now start building your own web scraping projects using Scrapy, the Apify SDK and host them on the Apify platform. See the Actor templates to get started with your own scraping tasks. If you have questions or need assistance, feel free to reach out on our GitHub or join our Discord community. Happy scraping!

Additional resources

RetroSearch is an open source project built by @garambo | Open a GitHub Issue

Search and Browse the WWW like it's 1997 | Search results from DuckDuckGo

HTML: 3.2 | Encoding: UTF-8 | Version: 0.7.4