CPPTest实战演示(CppTest libssh)

1 概述

  CppTest是一个可移植、功能强大但简单的单元测试框架,用于处理C++中的自动化测试。重点在于可用性和可扩展性。支持多种输出格式,并且可以轻松添加新的输出格式。

CppTest下载地址Sourceforge
Github地址

下面编写实际测试用例,来熟练CppTest库使用。

2 实例

2.1 接口

该接口是对libssh库C++封装,管理shh连接会话,定义如下:

namespace ssh
{
class Session
{
public:
    Session();
    ~Session();

    void set_host(const char* host);
    void set_port(int port);
    void set_port(const char* port);
    void set_user(const char* user);

    bool connect(int timeout = 10);
    void disconnect();
    bool verify();

    AuthMethod auth_method(const char *username);

    bool login(const char* password);
    bool login(const char* user, const char* password);
    bool login_by_interactive(const char* user, const char* password);
    bool login_by_prikey(const char* user, const char* privkeyfile);
    bool login_by_pubkey(const char* user, const char* pubkeyfile);
    const char* error();
};
}

2.2 测试用例

2.2.1 用例定义

class SeesionSuite : public Test::Suite
{
    ssh::Session sessioin_;
    std::string privkeyfile_ = "d:/id_rsa";
    int timeout = 5;
public:
    SeesionSuite()
    {
        TEST_ADD(SeesionSuite::connect_without_host);
        TEST_ADD(SeesionSuite::connect_error_host);
        TEST_ADD(SeesionSuite::connect_not_unconnected);
        TEST_ADD(SeesionSuite::connect_invalid_host);
        TEST_ADD(SeesionSuite::connect_without_port);
        TEST_ADD(SeesionSuite::connect_error_port);
        TEST_ADD(SeesionSuite::connect_correct_port);

        TEST_ADD(SeesionSuite::disconnect);
        TEST_ADD(SeesionSuite::login);
    }
private:
    void connect_without_host();
    void connect_error_host();
    void connect_not_unconnected();
    void connect_invalid_host();
    void connect_without_port();
    void connect_error_port();
    void connect_correct_port();
    void disconnect();
    void login();
    void error();
};

2.2.1 用例实现

#include "SessionTest.h"
#include <iostream>


void SeesionSuite::connect_without_host()
{
    TEST_ASSERT_EQUALS_MSG(false, sessioin_.connect(timeout), "not set host, cannot connected")
    error();
}

void SeesionSuite::connect_error_host()
{
    sessioin_.set_host("127.0.0.1");
    TEST_ASSERT_EQUALS_MSG(false, sessioin_.connect(timeout), "error host, cannot connected")
    error();
}

void SeesionSuite::connect_not_unconnected()
{
    sessioin_.set_host("20.20.20.109");
    TEST_ASSERT_EQUALS_MSG(false, sessioin_.connect(timeout), "not unconnected, cannot connected")
    error();
}

void SeesionSuite::connect_invalid_host()
{
    sessioin_.set_host("x.y.z.d");
    sessioin_.disconnect();
    TEST_ASSERT_EQUALS_MSG(false, sessioin_.connect(timeout), "error host, cannot connected")
    error();
}

void SeesionSuite::connect_without_port()
{
    sessioin_.set_host("20.20.20.109");
    sessioin_.disconnect();
    TEST_ASSERT_EQUALS_MSG(true, sessioin_.connect(timeout), sessioin_.error())
}

void SeesionSuite::connect_error_port()
{
    sessioin_.set_port(23);
    sessioin_.disconnect();
    TEST_ASSERT_EQUALS_MSG(false, sessioin_.connect(timeout), sessioin_.error())
    std::cerr << sessioin_.error() << ":23" << std::endl;
}

void SeesionSuite::connect_correct_port()
{
    sessioin_.set_port(22);
    sessioin_.disconnect();
    TEST_ASSERT_EQUALS_MSG(true, sessioin_.connect(timeout), sessioin_.error())
}

void SeesionSuite::disconnect()
{
    sessioin_.disconnect();
}

void SeesionSuite::login()
{
    sessioin_.set_host("20.20.20.109");
    sessioin_.set_port(22);
    TEST_ASSERT_EQUALS_MSG(true, sessioin_.connect(timeout), sessioin_.error())
    TEST_ASSERT_EQUALS_MSG(true, sessioin_.verify(), "verify is failed")
    ssh::AuthMethod method = sessioin_.auth_method("james");
    TEST_ASSERT_EQUALS_MSG(false, ssh::isUnknowOfAuthMethod(method), "Unknow auth method!")
    if(ssh::isPubKeyOfAuthMethod(method) && !privkeyfile_.empty())
        TEST_ASSERT_EQUALS_MSG(true, sessioin_.login_by_prikey("james", privkeyfile_.c_str()), "Pubkey's login is failed!")
    else if(ssh::isPasswordOfAuthMethod(method))
        TEST_ASSERT_EQUALS_MSG(true, sessioin_.login("james", "james010"), "Password's login is failed!")
    else if(ssh::isInteractiveOfAuthMethod(method))
        TEST_ASSERT_EQUALS_MSG(true, sessioin_.login_by_interactive("james", "james010"), "Interactive's login is failed!")
}

void SeesionSuite::error()
{
    std::cerr << sessioin_.error() << std::endl;
}

说明:

  • TEST_ASSERT_EQUALS_MSG(expected, got, msg)是CppTest库定义的宏,如果预期值expected与实际值got不同则报错,错误信息为msg

3 运行用例

3.1 主函数

int main(int argc, char *argv[])
{
    Test::Suite mainSuite;
    Test::TextOutput output(Test::TextOutput::Verbose);

    mainSuite.add(std::unique_ptr<Test::Suite>(new SeesionSuite));
    mainSuite.run(output, true);

    return 0;
}

3.1 运行结果

Hostname required
Timeout connecting to 127.0.0.1
ssh_socket_connect called on socket not unconnected
Failed to resolve hostname x.y.z.d ()
Timeout connecting to 20.20.20.109:23

SeesionSuite: 9/9, 100% correct in 10.094000 seconds
Total: 9 tests, 100% correct in 10.094000 seconds

说明:

  • 前5行是测试程序的错误打印
  • 后2行是测试结果,SeesionSuite的9个测试用例都通过了。
  • 5
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
C++ 的测试框架通常会使用桩函数(Stub)来模拟被测对象的行为,以便在测试中进行隔离和控制。对于一些常见的函数,cpptest 可以自动生成桩函数,从而简化测试的编写过程。下面是 cpptest 如何自动生成桩函数的示例: 假设我们有一个计算器类 Calculator,其中包含一个 `add()` 方法用于计算两个整数的和。我们想要测试这个方法,但是它依赖于另外一个类的方法,因此我们需要使用桩函数来模拟这个方法的行为。 ```c++ class Calculator { public: Calculator(Adder* adder) : adder_(adder) {} int add(int a, int b) { return adder_->add(a, b); } private: Adder* adder_; }; class Adder { public: virtual ~Adder() {} virtual int add(int a, int b) = 0; }; class RealAdder : public Adder { public: int add(int a, int b) override { return a + b; } }; ``` 在使用 cpptest 进行测试时,我们可以使用 `CPPTEST_AUTO_STUB` 宏来自动生成桩函数。这个宏会根据被测函数的参数类型和返回值类型自动生成一个桩函数,并且将其添加到测试代码中。 ```c++ TEST(CalculatorTest, AddTest) { // 自动创建桩函数 CPPTEST_AUTO_STUB(int, Adder::add, (int a, int b), (a, b), (0)); // 模拟 Adder::add() 方法 Adder* adder = new AdderStub(); EXPECT_CALL(*adder, add(2, 3)).WillOnce(Return(5)); // 测试 Calculator::add() 方法 Calculator calculator(adder); EXPECT_EQ(calculator.add(2, 3), 5); } ``` 在这个示例中,我们使用 `CPPTEST_AUTO_STUB` 宏来自动生成 Adder::add() 方法的桩函数,这个桩函数的返回值类型为 int,参数类型为两个 int 类型的参数。在测试中,我们模拟了 Adder::add() 方法的行为,并且测试了 Calculator::add() 方法的正确性。 总的来说,使用 cpptest 自动生成桩函数可以简化测试代码的编写过程,提高测试的效率和可靠性。但是需要注意的是,自动生成的桩函数可能无法满足所有的测试需求,需要在实际使用中加以验证和修改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

flysnow010

你的鼓励就是我最大的创作动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值