python - 如何模拟 googleapiclient.discovery.build

标签 python unit-testing google-cloud-platform mocking python-3.7

我正在尝试模拟对计算引擎进行 API 调用以列出 VM 的结果。但不幸的是无法模拟一个确切的函数。

我尝试使用 PATCH 和 MOCK 方法来模拟特定的调用,但仍然不成功

code.py 文件看起来像这样

import googleapiclient.discovery
import logging

class Service:
    def __init__(self, project, event):
        self.project_id = project
        self.compute = googleapiclient.discovery.build('compute', 'v1',
                                                       cache_discovery=False)
        self.event = event
        self.zones = self._validate_event()

    def _validate_event(self):
        if "jsonPayload" not in self.event:
            zones = self.compute.zones().list(
                project=self.project_id).execute()['items']

        else:
            zones = self.compute.zones().get(project=self.project_id,
                                             zone=self.event["jsonPayload"]
                                             ["resource"]["zone"]).execute()

        logging.debug(f"Identified Zones are {zones}")
        return [zone["name"] for zone in zones]

我的测试文件看起来像这样

# in-built
from unittest import TestCase
from unittest.mock import patch

# custom
import code


class TestServiceModule(TestCase):
    def setUp(self):
        self.project_id = "sample-project-id"

    @patch('code.googleapiclient.discovery')
    def test__validate_event_with_empty_inputs(self, mock_discovery):
        mock_discovery.build.zones.list.execute.return_value = {"items": [
            {
                "name": "eu-west-1"
            }
        ]}

        obj = code.Service(event={}, project=self.project_id)

        print(obj.zones)



在上面的测试用例中,我希望在打印 obj.zones 时看到“eu-west-1”作为值

最佳答案

你没有 mock googleapiclient.discovery.build方法正确。这是单元测试解决方案:
例如。code.py :

import googleapiclient.discovery
import logging


class Service:
    def __init__(self, project, event):
        self.project_id = project
        self.compute = googleapiclient.discovery.build('compute', 'v1', cache_discovery=False)
        self.event = event
        self.zones = self._validate_event()

    def _validate_event(self):
        if "jsonPayload" not in self.event:
            zones = self.compute.zones().list(project=self.project_id).execute()['items']
        else:
            zones = self.compute.zones().get(project=self.project_id,
                                             zone=self.event["jsonPayload"]["resource"]["zone"]).execute()

        logging.debug(f"Identified Zones are {zones}")
        return [zone["name"] for zone in zones]
test_code.py :
from unittest import TestCase, main
from unittest.mock import patch
import code


class TestService(TestCase):
    def setUp(self):
        self.project_id = "sample-project-id"

    @patch('code.googleapiclient.discovery')
    def test__validate_event_with_empty_inputs(self, mock_discovery):
        # Arrange
        mock_discovery.build.return_value.zones.return_value.list.return_value.execute.return_value = {
            "items": [{"name": "eu-west-1"}]}

        # Act
        obj = code.Service(event={}, project=self.project_id)

        # Assert
        mock_discovery.build.assert_called_once_with('compute', 'v1', cache_discovery=False)
        mock_discovery.build.return_value.zones.assert_called_once()
        mock_discovery.build.return_value.zones.return_value.list.assert_called_once_with(project='sample-project-id')
        mock_discovery.build.return_value.zones.return_value.list.return_value.execute.assert_called_once()
        self.assertEqual(obj.zones, ["eu-west-1"])


if __name__ == '__main__':
    main()
带有覆盖率报告的单元测试结果:
.
----------------------------------------------------------------------
Ran 1 test in 0.002s

OK
Name                                      Stmts   Miss  Cover   Missing
-----------------------------------------------------------------------
src/stackoverflow/56794377/code.py           14      1    93%   16
src/stackoverflow/56794377/test_code.py      16      0   100%
-----------------------------------------------------------------------
TOTAL                                        30      1    97%
版本:
  • google-api-python-client==1.12.3
  • Python 3.7.5
  • 关于python - 如何模拟 googleapiclient.discovery.build,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56794377/

    相关文章:

    kubernetes - 谷歌云平台的 "Managed Infrastructure Mixer Client"是什么?

    Python回合制游戏,找不到跳到下一个玩家的方法

    python - 文本信息无法正确抓取-Python

    java - 无法设置App Engine测试环境

    angularjs - 模拟服务注入(inject) Angular 模块运行 block

    mysql - 将我们的 Google App Engine Python API 连接到 Digital Ocean 上的外部 MySQL 数据库

    python - 在 PySpark 中将十进制解码为二进制信息

    python - Tensor.name 在 eager execution 中没有意义

    asp.net-mvc - 使用数据对 ASP.NET MVC 进行单元测试

    deployment - 如何强制 Kubernetes 在每个节点中使用 pod 更新部署