如何使用返回 IP 的函数进行 PHPUnit 测试?

问题描述

我刚开始接触 TDD,并且已经安装了 PHPUnit 和 PHPStorm。

我有这个类和函数,我想测试 IP 地址匹配。

class ip_request
{

    function getRealIpAddr()
    {
        if (!empty($_SERVER['HTTP_CLIENT_IP']))   //check ip from share internet
        {
            $ip=$_SERVER['HTTP_CLIENT_IP'];
        }
        elseif (!empty($_SERVER['HTTP_X_FORWARDED_FOR']))   //to check ip is pass from proxy
        {
            $ip=$_SERVER['HTTP_X_FORWARDED_FOR'];
        }
        else
        {
            $ip=$_SERVER['REMOTE_ADDR'];
        }
        return $ip;
    }
}

我正在尝试为测试创建代码,但我不确定要输入什么。

我试过了

public function testGetRealIpAddr()
    {
        
        $expected = '127.0.0.1';
        $this->assertEquals($expected,$test);
    }

但是 $test 显然是未定义的。

接下来我该怎么办?

尝试给出的建议

public function testGetRealIpAddr()
    {
        $UserIpAddress = $this
            ->getMockBuilder('ip_request')
            ->getMock();

        $UserIpAddress->expects($this->once())
            ->method('getRealIpAddr')
            ->willReturn('127.0.0.1'); // Set ip address whatever you want to use
    }

但我现在得到的错误

Trying to configure method "getRealIpAddr()" which cannot be configured because it does not exist,has not been specified,is final,or is static

解决方法

您要测试的方法/函数具有隐藏依赖$_SERVER

发现这一点还可以找到一种解决方案,使代码更加模块化且更易于测试。

这是通过使用可选参数公开先前隐藏的依赖项来工作的:

function getRealIpAddr(array $server = null)
{
    $server ??= $_SERVER;
    
    if (!empty($server['HTTP_CLIENT_IP']))   //check ip from share internet
    {
        $ip = $server['HTTP_CLIENT_IP'];
    } elseif (!empty($server['HTTP_X_FORWARDED_FOR']))   //to check ip is pass from proxy
    {
        $ip = $server['HTTP_X_FORWARDED_FOR'];
    } else {
        $ip = $server['REMOTE_ADDR'];
    }

    return $ip;
}

在单元测试中,您然后根据输入测试函数结果。

然而,单元测试无法突出功能所具有的大量安全问题,这可能超出了您的问题的范围,所以我只留下评论。

,

我按照@hakre 的建议修复了这个问题,

$ip = $_SERVER['REMOTE_ADDR'] ?? '127.0.0.1';

我在该语句的末尾添加了 ?? '127.0.0.1',并修复了它。

也刚刚更新了我的测试功能以仅显示

 $local = new ip_request();
 $this->assertEquals('127.0.0.1',$local->getRealIpAddr());