3

我有一个通过 REST 服务从 API 检索一些数据的子程序。代码比较简单,但是我需要将参数发布到 API 并且我需要使用 SSL,所以我必须通过LWP::UserAgent并且不能使用LWP::Simple。这是它的简化版本。

sub _request {
  my ( $action, $params ) = @_;

  # User Agent fuer Requests
  my $ua = LWP::UserAgent->new;
  $ua->ssl_opts( SSL_version => 'SSLv3' );

  my $res = $ua->post( 
    $url{$params->{'_live'} ? 'live' : 'test'}, { action => $action, %$params } 
  );
  if ( $res->is_success ) {
    my $json = JSON->new;

    return $json->decode( $res->decoded_content );
  } else {
    cluck $res->status_line;
    return;
  }
}

这是我的模块(不是 OOp)中唯一需要$ua.

现在我想为此编写一个测试,经过一些研究决定最好使用Test::LWP::UserAgent,这听起来很有希望。不幸的是,有一个问题。在文档中,它说:

请注意,LWP::UserAgent 本身没有猴子补丁 - 您必须使用此模块(或子类)来发送您的请求,否则无法捕获和处理它。

换出用户代理实现的一种常见机制是通过延迟构建的 Moose 属性。如果在构建时没有提供覆盖,默认为 LWP::UserAgent->new(%options)。

哎呀。显然我不能做 Moose 的事情。我也不能只将 a 传递$ua给 sub。我当然可以为 sub 添加一个可选的第三个参数$ua,但我不喜欢这样做的想法。我觉得为了使其可测试而如此彻底地改变这种简单代码的行为是不合适的。

我基本上想做的是像这样运行我的测试:

use strict;
use warnings;
use Test::LWP::UserAgent;
use Test::More;

require Foo;

Test::LWP::UserAgent->map_response( 'www.example.com',
  HTTP::Response->new( 200, 'OK', 
    [ 'Content-Type' => 'text/plain' ], 
    '[ "Hello World" ]' ) );

is_deeply(
  Foo::_request('https://www.example.com', { foo => 'bar' }),
  [ 'Hello World' ],
  'Test foo'
);

有没有办法将 Test::LWP::UserAgent 功能猴子修补到 LWP::UserAgent 中,以便我的代码只使用 Test:: one?

4

3 回答 3

4

更改您的代码,以便在_request()您调用_ua()以收集您的用户代理并在您的测试脚本中覆盖此方法。像这样:

在您的模块中:

sub _request {
...
 my $ua = _ua();
...
}

sub _ua { 
 return LWP::UserAgent->new();
}

在您的测试脚本中:

...
Test::More::use_ok('Foo');

no warnings 'redefine';
*Foo::_ua = sub { 
    # return your fake user agent here
};
use warnings 'redefine';
... etc etc
于 2013-02-19T19:21:21.787 回答
2

我当然可以为 sub 添加一个可选的第三个参数 $ua,但我不喜欢这样做的想法。我觉得为了使其可测试而如此彻底地改变这种简单代码的行为是不合适的。

这被称为依赖注入,它是完全有效的。对于测试,您需要能够覆盖您的类将用来模拟各种结果的对象。

如果您更喜欢覆盖对象的更隐式方式,请考虑Test::MockObjectTest::MockModule。您可以模拟 LWP::UserAgent 的构造函数以返回测试对象,或者模拟您正在测试的代码的更广泛部分,这样根本不需要 Test::LWP::UserAgent。

另一种方法是重构您的生产代码,以便组件可以(单元)单独测试。从响应处理中拆分 HTTP 获取。然后通过创建自己的响应对象并将其传入来测试第二部分非常简单。

最终,程序员使用上述所有工具。有些适用于单元测试,有些适用于更广泛的集成测试。

于 2013-02-19T16:27:05.250 回答
0

截至今天,我将采用以下方法来解决这个问题。想象一下这段遗留代码1,它不是面向对象的,不能重构,因此它使依赖注入变得容易。

package Foo;
use LWP::UserAgent;

sub frobnicate {
    return LWP::UserAgent->new->get('http://example.org')->decoded_content;
}

这确实很难测试,rjh 的答案很准确。但是在 2016 年,我们有比 2013 年更多的可用模块。我特别喜欢Sub::Override,它替换了给定命名空间中的 sub,但只保留在当前范围内。这使得它非常适合单元测试,因为您不需要在完成后关心恢复所有内容。

package Test::Foo;
use strict;
use warnings 'all';
use HTTP::Response;
use Sub::Override;
use Test::LWP::UserAgent;
use Test::More;

# create a rigged UA
my $rigged_ua = Test::LWP::UserAgent->new;
$rigged_ua->map_response( 
    qr/\Qexample\E/ => HTTP::Response->new( 
        '200', 
        'OK', 
        [ 'Content-Type' => 'text/plain' ], 
        'foo',
    ), 
);

# small scope for our override
{
    # make LWP return it inside our code
    my $sub = Sub::Override->new( 
        'LWP::UserAgent::new'=> sub { return $rigged_ua } 
    );
    is Foo::frobnicate(), 'foo', 'returns foo';
}

我们基本上创建了一个Test::LWP::UserAgent对象,我们提供所有的测试用例。如果需要,我们还可以给它一个代码引用,它将在请求上运行测试(此处未显示)。然后我们使用 Sub::Override 使 LWP::UserAgent 的构造函数不返回实际的 LWP::UA,而是返回已经准备好的$rigged_ua. 然后我们运行我们的测试。一旦$sub超出范围,LWP::UserAgent::new就会恢复,我们不会干涉其他任何事情。

始终以尽可能小的范围进行这些测试很重要(就像 Perl 中的大多数事情一样)。

如果有很多这样的测试用例,那么为每个请求构建某种配置散列是一个很好的策略,并使用构建辅助函数来创建操纵的用户代理,并使用另一个来创建 Sub: :覆盖对象。在词法作用域中使用,这种方法非常强大,同时也非常简洁。


use strict1) 这里用和的缺失来表示use warnings

于 2016-08-29T10:50:09.487 回答