Tornado应用笔记05-异步客户端

索引

本节内容将分析Tornado内置的异步HTTP客户端, 包括源码和Tornado官方的爬虫demo. AsyncHTTPClientSimpleAsyncHTTPClient两个类是异步客户端的应用层代码, 下面就来分析其中的源码, 更底层的代码分析留待日后填坑.

简介与说明

Tornado内置了一个没有外部依赖的基于tornado iostreams的非阻塞 HTTP 1.1 客户端,

默认情况下, 为了限制挂起连接的数量, 单个IOLoop只有一个AsyncHTTPClient实例. 当然也可以通过设置force_instance=True来取消这个限制, 然而并不建议这么做. 默认情况下, 只有第一次调用下面这种方式初始化客户端时, 配置才会生效.

AsyncHTTPClient(max_clients=20, defaults=dict(user_agent="WtfAgent"))

原因在上面也提到了, AsyncHTTPClient 是"隐式复用"的, 除非你设置了force_instance取消"单实例"限制, 所以更推荐下面这种配置方式

tornado.httpclient.AsyncHTTPClient.configure(None,
    defaults=dict(
        user_agent="MyUserAgent"
     ), 
    max_clients=20,
)

max_clients是实例的最大同时请求数, 超过限制的请求将会被放入队列中, 需要注意的是, 在队列中等待的时间也是计入超时计时中的, 比方说设定的超时时间是2秒, 如果排队时间超过2秒 , 即便这个请求实际上并没有发出, 那么也会被认定为超时

源码分析
AsyncHTTPClient

主要看fetch, 其内部的操作如下:

  • 新建一个future
  • 统一请求格式, 如果输入的是url那么需要封装成HTTPRequest
  • 为请求添加回调(具体操作是, 先为future添加回调, future完成后执行自身回调, future的回调再在ioloop注册用户定义的回调)
  • 执行fetch_impl, 这个函数实际上调用的是AsyncHTTPClient的子类SimpleAsyncHTTPClient 的方法, fetch_impl完成后会执行future.set_result的回调
class AsyncHTTPClient(Configurable):

    def fetch(self, request, callback=None, raise_error=True, **kwargs):
        # 实际上并不需要显式close, 除非用多实例, 一般情况不会关异步客户端
        if self._closed:
            raise RuntimeError("fetch() called on closed AsyncHTTPClient")
        # 统一请求格式
        if not isinstance(request, HTTPRequest):
            request = HTTPRequest(url=request, **kwargs)
        request.headers = httputil.HTTPHeaders(request.headers)
        request = _RequestProxy(request, self.defaults)
        future = TracebackFuture()
        # 注册回调
        if callback is not None:
            callback = stack_context.wrap(callback)

            def handle_future(future):
                exc = future.exception()
                if isinstance(exc, HTTPError) and exc.response is not None:
                    response = exc.response
                elif exc is not None:
                    response = HTTPResponse(
                        request, 599, error=exc,
                        request_time=time.time() - request.start_time)
                else:
                    response = future.result()
                # 注册HTTP请求完成的回调
                self.io_loop.add_callback(callback, response)

            # 注册future的回调
            future.add_done_callback(handle_future)

        # fetch_impl完成的回调
        def handle_response(response):
            if raise_error and response.error:
                future.set_exception(response.error)
            else:
                future.set_result(response)

        # "发送"请求, 返回`future`
        self.fetch_impl(request, handle_response)
        return future
SimpleAsyncHTTPClient

客户端是以队列的形式管理HTTP请求的, 单个实例允许同时发起和处理的最大请求数(即max_clients)默认是10, 其内部工作流程如下:

  • 将新的HTTP请求加入队列
  • 将请求放入"等待区", 判断当前"工作区"的请求数, 如果超出可处理的请求数则为其在ioloop注册超时事件(超时事件是将请求移出等待区, 并通过在ioloop注册回调的方式抛出异常)
  • 处理"等待区"的请求, 先注销ioloop中相应的超时事件(如果有的话), 然后将请求放入"工作区", 接着发出异步HTTP请求, 请求包含了释放资源的回调, 回调会将请求移出"工作区", 并重复这一步骤, 继续处理"等待区"的请求, 直到"等待区"被清空.
class SimpleAsyncHTTPClient(AsyncHTTPClient):
    def initialize(self, io_loop, max_clients=10,
                   hostname_mapping=None, max_buffer_size=104857600,
                   resolver=None, defaults=None, max_header_size=None,
                   max_body_size=None):

        super(SimpleAsyncHTTPClient, self).initialize(io_loop,
                                                      defaults=defaults)
        self.max_clients = max_clients
        self.queue = collections.deque()
        self.active = {}
        self.waiting = {}
        self.max_buffer_size = max_buffer_size
        self.max_header_size = max_header_size
        self.max_body_size = max_body_size
        # TCPClient could create a Resolver for us, but we have to do it
        # ourselves to support hostname_mapping.
        if resolver:
            self.resolver = resolver
            self.own_resolver = False
        else:
            self.resolver = Resolver(io_loop=io_loop)
            self.own_resolver = True
        if hostname_mapping is not None:
            self.resolver = OverrideResolver(resolver=self.resolver,
                                             mapping=hostname_mapping)
        self.tcp_client = TCPClient(resolver=self.resolver, io_loop=io_loop)

    def fetch_impl(self, request, callback):

        # 将请求加入`collections.deque()`队列中
        key = object()
        self.queue.append((key, request, callback))

        # 对于超出最大同时请求量的请求, 为其在ioloop注册超时事件,
        if not len(self.active) < self.max_clients:
            timeout_handle = self.io_loop.add_timeout(
                self.io_loop.time() + min(request.connect_timeout,
                                          request.request_timeout),
                functools.partial(self._on_timeout, key))
        else:
            timeout_handle = None

        # 在`waiting`(等待发出的请求)中添加这`整个请求事件`(包括请求本身, 请求回调, 超时处理)
        self.waiting[key] = (request, callback, timeout_handle)

        # 处理请求队列
        self._process_queue()
        if self.queue:
            gen_log.debug("max_clients limit reached, request queued. "
                          "%d active, %d queued requests." % (
                              len(self.active), len(self.queue)))

    def _process_queue(self):
        with stack_context.NullContext():
            # 只能同时处理不超过 `max_clients` 的请求数
            while self.queue and len(self.active) < self.max_clients:
                # 从队列中获取在`等待`的请求,
                key, request, callback = self.queue.popleft()
                if key not in self.waiting:
                    continue
                # 注销超时事件(如果有)并退出`waiting`, 在`active`(正在处理的请求)中添加请求
                self._remove_timeout(key)
                self.active[key] = (request, callback)
                # 构建释放资源的回调函数
                release_callback = functools.partial(self._release_fetch, key)
                # `真正`发送HTTP请求的操作
                self._handle_request(request, release_callback, callback)

    def _handle_request(self, request, release_callback, final_callback):
        self._connection_class()(
            self.io_loop, self, request, release_callback,
            final_callback, self.max_buffer_size, self.tcp_client,
            self.max_header_size, self.max_body_size)

    def _release_fetch(self, key):
        # 完成请求后释放资源, 退出`active`
        del self.active[key]
        self._process_queue()

    def _remove_timeout(self, key):
        # 注销超时事件(如果有)并退出`waiting`
        if key in self.waiting:
            request, callback, timeout_handle = self.waiting[key]
            if timeout_handle is not None:
                self.io_loop.remove_timeout(timeout_handle)
            del self.waiting[key]

    def _on_timeout(self, key):
        # 退出`waiting`并注册超时响应回调
        request, callback, timeout_handle = self.waiting[key]
        self.queue.remove((key, request, callback))
        timeout_response = HTTPResponse(
            request, 599, error=HTTPError(599, "Timeout"),
            request_time=self.io_loop.time() - request.start_time)
        self.io_loop.add_callback(callback, timeout_response)
        del self.waiting[key]
官方爬虫demo(源码地址)

这个爬虫来自Tornado源码附带的demo, 使用其内置的队列, 协程和异步HTTP客户端实现一个"全站"爬虫. 理解这个demo对于理解协程和异步客户端都有一定帮助.

先简单说明这个爬虫的工作原理:

  • 创建一个队列, 用于存放等待爬取的网页url, 并将网站的"根url"放入队列中
  • 开启10个worker(可以理解为10个异步客户端)爬网站, 并为这个爬虫任务设置一个超时时间, 如果超时将抛出异常终止任务
  • 每个worker的工作流程大致为:
    • 从队列获取一个新的url加入到"工作区",
    • 爬取新url对应的页面
    • 获取新页面中的所有url
    • 将本次已爬取的url加入"完成区", 移出"工作区"
    • 按照规定筛选前面获取到的新url, 然后将筛选的结果加入队列
    • 重复第一步, 直到队列清空
#!/usr/bin/env python

import time
from datetime import timedelta

try:
    from HTMLParser import HTMLParser
    from urlparse import urljoin, urldefrag
except ImportError:
    from html.parser import HTMLParser
    from urllib.parse import urljoin, urldefrag

from tornado import httpclient, gen, ioloop, queues

# 设定根url和最大worker数
base_url = 'http://www.tornadoweb.org/en/stable/'
concurrency = 10


@gen.coroutine
def get_links_from_url(url):
    # 爬取网页内容并获取网页中所有去尾的(去除锚点"#"及其后续内容)url
    try:
        response = yield httpclient.AsyncHTTPClient().fetch(url)
        print('fetched %s' % url)

        html = response.body if isinstance(response.body, str) \
            else response.body.decode()
        urls = [urljoin(url, remove_fragment(new_url))
                for new_url in get_links(html)]
    except Exception as e:
        print('Exception: %s %s' % (e, url))
        raise gen.Return([])

    raise gen.Return(urls)


def remove_fragment(url):
    # 去除url中锚点"#"及其后续内容
    pure_url, frag = urldefrag(url)
    return pure_url


def get_links(html):
    # 获取网页中所有<a>标签url
    class URLSeeker(HTMLParser):
        def __init__(self):
            HTMLParser.__init__(self)
            self.urls = []

        def handle_starttag(self, tag, attrs):
            href = dict(attrs).get('href')
            if href and tag == 'a':
                self.urls.append(href)

    url_seeker = URLSeeker()
    url_seeker.feed(html)
    return url_seeker.urls


@gen.coroutine
def main():
    # 创建url队列, "工作区" 和 "完成区"
    q = queues.Queue()
    start = time.time()
    fetching, fetched = set(), set()

    @gen.coroutine
    def fetch_url():
        # 从队列中获取新的url
        current_url = yield q.get()
        try:
            # 当前url已经在爬取(即url在"工作区中"), 就退出
            if current_url in fetching:
                return

            print('fetching %s' % current_url)
            
            # 将url加入"工作区", 爬取内容, 完成后加入"完成区"
            fetching.add(current_url)
            urls = yield get_links_from_url(current_url)
            fetched.add(current_url)

            # 筛选新的url并将其加入队列
            for new_url in urls:
                # Only follow links beneath the base URL
                if new_url.startswith(base_url):
                    yield q.put(new_url)

        # 完成后调用`q.task_done`, 队列中的未完成任务计数将减1, 与`q.get`一一对应
        # 为的是配合`q.join`, 当计数为0时, `q.join`会结束"等待"
        finally:
            q.task_done()

    # worker
    @gen.coroutine
    def worker():
        while True:
            yield fetch_url()

    # 为队列添加第一个url("任务")
    q.put(base_url)

    # 启动workers, 然后"等待"队列清空, 时间为300秒, 超时会抛出异常
    # 完成后检查"url队列"和"完成区"是否一致, 最后输出任务耗时
    for _ in range(concurrency):
        worker()

    # 这里使用了`.join`阻塞等待, 直到队列被清空才会恢复, 继续往下走
    yield q.join(timeout=timedelta(seconds=300))
    assert fetching == fetched
    print('Done in %d seconds, fetched %s URLs.' % (
        time.time() - start, len(fetched)))


if __name__ == '__main__':
    import logging
    logging.basicConfig()
    # 启动任务
    io_loop = ioloop.IOLoop.current()
    io_loop.run_sync(main)

本节内容就是这些, 下节内容将通过两种方式实现的聊天室demo, 介绍WebSocket与长轮询在Tornado中的实现方法.

NEXT ===> Tornado应用笔记06-WebSocket与长轮询

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容