Я пытаюсь понять async, разработанную на Python.Поэтому я написал несколько веб-парсеров, которые используют Asyncio и Aiohttp.Поэтому у меня возникли некоторые проблемы, мой скрипт зависает на методе .join (), хотя асинхронная очередь заполнена.На самом деле я думаю, что главная проблема с aiohttp.ClientSession в методе _consume (), потому что, когда я не использую запрос get в этом методе, у меня не возникает никаких проблем.Так что я не так?
import asyncio
import aiohttp
import random
from bs4 import BeautifulSoup
class OlxParser:
def __init__(self):
self.loop = asyncio.get_event_loop()
self._queue = asyncio.Queue()
self._run_loop = True
self._sess = None
self._url = 'https://www.olx.kz/elektronika/telefony-i-aksesuary/mobilnye-telefony-smartfony/alma-ata/?search%5Bprivate_business%5D=private'
self._headers = {
'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_11_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/50.0.2661.102 Safari/537.36'
}
self._urls = []
self._sleep_range = (1, 3)
self.data = []
async def _sleep(self):
sleep_time = random.randint(*self._sleep_range)
await asyncio.sleep(sleep_time)
async def _consume(self):
while True:
url = await self._queue.get()
html = None
print('Pass')
if url not in self._urls:
self._urls.append(url)
print('Fetching:', url)
async with self._sess.get(url, allow_redirects=False, headers=self._headers) as res:
if res.status == 200:
html = await res.text()
else:
print('Status:', res.status)
print('Error url:', url)
if html is not None:
soup = BeautifulSoup(html, 'html.parser')
price_label = soup.find('div', {'class': 'price-label'})
price = price_label.find('strong')
self.data.append(price.get_text())
# await self._sleep()
self._queue.task_done()
async def _produce(self, page_num):
url = self._url
if page_num > 1:
url += '&page={}'.format(page_num)
print('Fetching:', url)
html = None
async with self._sess.get(url, allow_redirects=False, headers=self._headers) as res:
if res.status == 200:
html = await res.text()
else:
print('Fetching has stopped at page number:', str(page_num))
self._run_loop = False
if html is not None:
soup = BeautifulSoup(html, 'html.parser')
table = soup.find('table', {'id': 'offers_table'})
links = table.find_all('a', {'class': 'detailsLink'})
for link in links:
await self._queue.put(link['href'])
# await self._sleep()
async def run(self):
consumer = asyncio.ensure_future(self._consume())
page_num = 1
async with aiohttp.ClientSession(loop=self.loop) as sess:
self._sess = sess
while self._run_loop:
await self._produce(page_num)
page_num += 1
await self._queue.join()
consumer.cancel()
def __enter__(self):
return self
def __exit__(self, exc_type, exc_val, exc_tb):
self.loop.close()
return exc_type is None
if __name__ == '__main__':
with OlxParser() as obj:
obj.loop.run_until_complete(obj.run())