3

我们现在正在公司中引入单元测试,并考虑模拟系统调用的最佳方式。

考虑以下代码

    int fd = open(path, O_RDONLY);
    if (fd < 0) {
        LOG.error() << "cannot load plugin " << path << std::endl;
        return ERROR(ERROR_OPENING_PLUGING);
    }
    // do other stuff

显然,我们需要mock系统调用才能打开

我找到了以下方法:

  1. 正确 - 在设计方面但丑陋的方式。创建接口和实现

    class ISystem
    {
    public:
        typedef std::auto_ptr<ISystem> Ptr;
    
        ISystem() {};
        virtual ~ISystem() {};
    
        virtual int open(const char* file, int path) = 0;
    };
    
    class System : public ISystem
    {
    public:
        System() {};
        virtual ~System() {};
    
        virtual int open(const char* file, int path);
    
        static ISystem::Ptr Get();
    };
    

并使用它

    Common::Error dlopen_signed(ISystem::Ptr& system, const char* path, int flags, void*& ret)
    {
    int fd = system->open(path, O_RDONLY);
    if (fd < 0) {
        LOG.error() << "cannot load plugin " << path << std::endl;
        return ERROR(ERROR_OPENING_PLUGING);
    }
    char fd_path[32];

我不喜欢它,因为每个函数都需要多一个参数 - ISystem::Ptr& 系统,这对于所有生产代码都是相同的。

此外,不确定速度(这涉及必须非常快的基本系统调用的额外层)

2) 使用链接接缝 链接器的设计使其更喜欢您的功能版本而不是系统版本。

但这不适用于某些系统调用,例如 open (不确定原因),而且这个解决方案有点 hackish。

3) 使用 --wrap 编译器功能

--wrap symbol 使用符号的包装函数。任何未定义的符号引用都将解析为 __wrap_symbol。对 __real_symbol 的任何未定义引用都将被解析为符号。这可用于为系统函数提供包装器。包装函数应称为 __wrap_symbol。如果它想调用系统函数,它应该调用__real_symbol。这是一个简单的例子:

void *
__wrap_malloc (int c)
{
  printf ("malloc called with %ld\n", c);
  return __real_malloc (c);
} 

这个解决方案很好,但不适用于我猜的所有编译器。

问题是 -您在项目中使用的是哪一个?

4

3 回答 3

2

您必须对要模拟的内容和要进行单元测试的内容划清界限。100% 的单元测试覆盖率并不是最终目标。

如果您真的想模拟系统调用,最好将它们放在包装器中(问题中的第一个选项)。不是一个巨大的包装器,但您应该按功能将它们分成几个包装器。

于 2013-11-05T09:23:59.300 回答
0

系统调用是基础设施
我们使用包装器包装基础逻辑
还设计包装器,以便可以有许多基于上下文的包装器。
我也同意“BЈовић”100% 的覆盖率不是最终目标——平衡它。

于 2013-11-05T09:52:53.787 回答
-1

(1)示例关于模拟不正确,可能是因为您不想澄清自己在做什么,而您的示例没有向我展示什么模拟了什么。模拟实际上是多态性的实际示例使用。人们倾向于将所有东西都包装在包装函数中,然后甚至包装在包装类中

IParam param=new ParamMock(xxx);
File file(param); // mocked
....
file.open(xxx)

(2) 我不知道

(3) 这绝对表明你知道太多并且在单元测试上花费了太多精力。我始终牢记单元测试毕竟会被丢弃。您遵循什么开发流程?

于 2013-11-05T09:47:42.347 回答