Отправка почтовых запросов с помощью Scrapy - PullRequest
0 голосов
/ 16 июня 2020

Я учусь выполнять парсинг веб-страниц с помощью Scrapy, и у меня возникли проблемы с парсингом динамически загружаемого контента. Я пытаюсь очистить номер телефона с веб-сайта , который отправляет запрос POST, чтобы получить номер: Это заголовок почтового запроса , который он отправляет:

Host: www.mymarket.ge
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:77.0) Gecko/20100101 Firefox/77.0
Accept: */*
Accept-Language: en-US,en;q=0.5
Accept-Encoding: gzip, deflate, br
Referer: https://www.mymarket.ge/en/pr/16399126/savaWro-inventari/fulis-yuTi
Content-Type: application/x-www-form-urlencoded; charset=UTF-8
X-Requested-With: XMLHttpRequest
Content-Length: 13
Origin: https://www.mymarket.ge
Connection: keep-alive
Cookie: Lang=en; split_test_version=v1; CookieID=eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJEYXRhIjp7IklEIjozOTUwMDY2MzUsImN0IjoxNTkyMzA2NDMxfSwiVG9rZW5JRCI6Ik55empxVStDa21QT1hKaU9lWE56emRzNHNSNWtcL1wvaVVUYjh2dExCT3ZKWT0iLCJJc3N1ZWRBdCI6MTU5MjMyMTc1MiwiRXhwaXJlc0F0IjoxNTkyMzIyMDUyfQ.mYR-I_51WLQbzWi-EH35s30soqoSDNIoOyXgGQ4Eu84; ka=da; SHOW_BETA_POPUP=B; APP_VERSION=B; LastSearch=%7B%22CatID%22%3A%22515%22%7D; PHPSESSID=eihhfcv85liiu3kt55nr9fhu5b; PopUpLog=%7B%22%2A%22%3A%222020-05-07+15%3A13%3A29%22%7D

и это тело :

PrID=16399126

Мне удалось реплицировать почтовый запрос на reqbin.com , но не могу выяснить, как это сделать с помощью Scrapy. Вот как выглядит мой код:

class MymarketcrawlerSpider(CrawlSpider):
    name = "mymarketcrawler"
    allowed_domains = ["mymarket.ge"]
    start_urls = ["http://mymarket.ge/"]

    rules = (
        Rule(
            LinkExtractor(allow=r".*mymarket.ge/ka/*", restrict_css=".product-card"),
            callback="parse_item",
            follow=True,
        ),
    )

    def parse_item(self, response):
        item_loader = ItemLoader(item=MymarketItem(), response=response)

        def parse_num(response):
            try:
                response_text = response.text
                response_dict = ast.literal_eval(response_text)
                number = response_dict['Data']['Data']['numberToShow']
                nonlocal item_loader
                item_loader.add_value("number", number)
                yield item_loader.load_item()
            except Exception as e:
                raise CloseSpider(e)


        yield FormRequest.from_response(
            response,
            url=r"https://www.mymarket.ge/ka/pr/ShowFullNumber/",
            headers={
                "Host": "www.mymarket.ge",
                "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:77.0) Gecko/20100101 Firefox/77.0",
                "Accept": "*/*",
                "Accept-Language": "en-US,en;q=0.5",
                "Accept-Encoding": "gzip, deflate, br",
                "Referer": "https://www.mymarket.ge/ka/pr/16399126/savaWro-inventari/fulis-yuTi",
                "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
                "X-Requested-With": "XMLHttpRequest",
            },
            formdata={"PrID": "16399126"},
            method="POST",
            dont_filter=True,
            callback=parse_num
        )
        item_loader.add_xpath(
            "seller", "//div[@class='d-flex user-profile']/div/span/text()"
        )
        item_loader.add_xpath(
            "product",
            "//div[contains(@class, 'container product')]//h1[contains(@class, 'product-title')]/text()",
        )
        item_loader.add_xpath(
            "price",
            "//div[contains(@class, 'container product')]//span[contains(@class, 'product-price')][1]/text()",
            TakeFirst(),
        )
        item_loader.add_xpath(
            "images",
            "//div[@class='position-sticky']/ul[@id='imageGallery']/li/@data-src",
        )
        item_loader.add_xpath(
            "condition", "//div[contains(@class, 'condition-label')]/text()"
        )
        item_loader.add_xpath(
            "city",
            "//div[@class='d-flex font-14 font-weight-medium location-views']/span[contains(@class, 'location')]/text()",
        )
        item_loader.add_xpath(
            "number_of_views",
            "//div[@class='d-flex font-14 font-weight-medium location-views']/span[contains(@class, 'svg-18')]/span/text()",
        )
        item_loader.add_xpath(
            "publish_date",
            "//div[@class='d-flex left-side']//div[contains(@class, 'font-12')]/span[2]/text()",
        )
        item_loader.add_xpath(
            "total_products_amount",
            "//div[contains(@class, 'user-profile')]/div/a/text()",
            re=r"\d+",
        )
        item_loader.add_xpath(
            "description", "//div[contains(@class, 'texts full')]/p/text()"
        )
        item_loader.add_value("url", response.url)
        yield item_loader.load_item()

Код выше не работает; Поле номер не заполняется. Я могу распечатать номер на экране, но не могу сохранить его в файле csv. Столбец число в CSV-файле пуст и не содержит значений.

1 Ответ

1 голос
/ 17 июня 2020

Scrapy работает асинхронно, и каждая ссылка для сканирования, каждый элемент для обработки и т. Д. c. помещается в очередь. Вот почему вы отправляете запрос и ждете SpiderDownloader, ItemPipeline и т. Д. c. для обработки вашего запроса.

Происходит то, что у вас есть запросы, которые обрабатываются отдельно , и поэтому вы не видите свои результаты. Лично я бы проанализировал результаты первого запроса, сохранил их в «метаданных» и передал их следующему запросу, чтобы данные были доступны впоследствии.

Например

class MymarketcrawlerSpider(CrawlSpider):
    name = "mymarketcrawler"
    allowed_domains = ["mymarket.ge"]
    start_urls = ["http://mymarket.ge/"]

    rules = (
        Rule(
            LinkExtractor(allow=r".*mymarket.ge/ka/*", restrict_css=".product-card"),
            callback="parse_item",
            follow=True,
        ),
    )

    def parse_item(self, response):

        def parse_num(response):
            item_loader = ItemLoader(item=MymarketItem(), response=response)
            try:
                response_text = response.text
                response_dict = ast.literal_eval(response_text)
                number = response_dict['Data']['Data']['numberToShow']
                # New part: 
                product = response.meta['product']             

                # You won't need this now: nonlocal item_loader
                # Also new: 
                item_loader.add_value("number", number)

                item_loader.add_value("product", product)
                yield item_loader.load_item()
            except Exception as e:
                raise CloseSpider(e)
        # Rewrite your parsers like this: 
        product = response.xpath(
            "//div[contains(@class, 'container product')]//h1[contains(@class, 'product-title')]/text()"
        ).get()

        yield FormRequest.from_response(
            response,
            url=r"https://www.mymarket.ge/ka/pr/ShowFullNumber/",
            headers={
                "Host": "www.mymarket.ge",
                "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:77.0) Gecko/20100101 Firefox/77.0",
                "Accept": "*/*",
                "Accept-Language": "en-US,en;q=0.5",
                "Accept-Encoding": "gzip, deflate, br",
                "Referer": "https://www.mymarket.ge/ka/pr/16399126/savaWro-inventari/fulis-yuTi",
                "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
                "X-Requested-With": "XMLHttpRequest",
            },
            formdata={"PrID": "16399126"},
            method="POST",
            dont_filter=True,
            callback=parse_num,
            meta={"product": product}
        )
...