美文网首页
LocalStack: 本机 Mock AWS 服务利器

LocalStack: 本机 Mock AWS 服务利器

作者: haitaoyao | 来源:发表于2017-07-19 09:32 被阅读191次

    如果代码中需要与 AWS 服务(比如 S3)交互, 如何写单元测试?

    0x00 mock API

    既然是调用 AWS 的 API, 那么可以从 AWS SDK 入手, 通过开发语言级别的 Mock framework, 拦截 API 调用. 以 Java 访问 S3 为例, 使用 Unitils 框架, Mock AmazonS3Client 类.

    // 通过 mock AmazonS3Client 
    private Mock<AmazonS3Client> mockS3Client = null;
    
    // 约定 API 返回内容
    this.mockS3Client.returns(objectListingResult).listObjects(request);
    

    这种做法有几个缺点:

    • 需要开发语言支持, 如果选择了 Go, 我也不知道怎么动态 Mock
    • 需要自己写的代码容易传入 Mock 过的 client 代码, 比如一个静态变量的 AmazonS3Client 怎办, 咳咳
    • 没有真正通过 API 请求服务, 如果 mock 逻辑错误, 没有达到测试的目的

    0x01 LocalStack:

    LocalStack 是开发 JIRA 的公司 Atlassian 开发的, 用 Python "山寨"了 AWS 的 API, 通过 REST API 提供跟 AWS 一模一样的服务. 使用起来也非常简单, 直接 docker pull atlassianlabs/localstack 就完成了安装. 启动也足够简单,

    # 8080 端口是 web 使用
    # SERVICES 环境变量用于指定启动的服务
    # 4560-4582 是各个服务使用的端口
    docker run -p 8080:8080 -p 4560-4582:4560-4582 --name localstack -e SERVICES='s3,web' atlassianlabs/localstack
    

    各个服务使用的端口如下:

    AWS 的 cli/SDK 都提供一个 endpoint-url(也就是 AWS API server 的 url) 的 hook, 方便的让我们使用 Localstack. 例如, aws cli 中使用本地的 S3:

    # 创建 bucket
    aws s3api --endpoint http://localhost:4572  create-bucket --bucket test-bucket
    
    # 执行 ls 操作
    aws s3 --endpoint http://localhost:4572 ls s3://test-bucket/
    

    为了与 Java/JUnit 集成, LocalStack 还提供了 LocalstackTestRunner, 参见官方示例:

    @RunWith(LocalstackTestRunner.class)
    public class MyCloudAppTest {
    
      @Test
      public void testLocalS3API() {
        AmazonS3 s3 = new AmazonS3Client(...);
        s3.setEndpoint(LocalstackTestRunner.getEndpointS3());
        List<Bucket> buckets = s3.listBuckets();
        ...
      }
    }
    

    不过值得提醒的是, 这个 LocalstackTestRunner 从 github 上下载最新的 localstack 并且在本机安装, 作为天朝码农你懂的, 因此还是建议使用 docker 方式运行 localstack, 非常便捷.

    LocalStack 与 gitlab CI 的集成也非常简单, 仅需要在前置的 stage 的 services 中定义 localstack 的 image 即可.

    LocalStack 的好处也非常明显:

    • 真正的 REST API 调用, 不用代码级别 Mock SDK, 因此也做到了所有语言通吃
    • 错误注入, 比如通过 KINESIS_ERROR_PROBABILITY 环境变量的值指定多大概率扔出 ProvisionedThroughputExceededException 异常. 不过这种设计在 gitlab CI 中就不是很容易集成进来, 毕竟这个环境变量是 Localstack 进程全局的, 如果想同时测试正常情况和异常情况, 还是要动动脑筋

    总结

    再也没有借口不写 AWS 服务相关的测试代码了, 不是么.....

    -- EOF --

    相关文章

      网友评论

          本文标题:LocalStack: 本机 Mock AWS 服务利器

          本文链接:https://www.haomeiwen.com/subject/ktokkxtx.html