LocalStack: 本机 Mock AWS 服务利器

如果代码中需要与 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 --

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,359评论 19 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 175,270评论 25 709
  • https://aws.amazon.com/cn/s3/faqs/#sia_anchor Amazon Simp...
    守望者_1065阅读 12,643评论 0 5
  • 本文是GitChat《Serverless 风格微服务的持续交付(上):架构案例》部分内容已做修改。文章聊天实录请...
    顾宇阅读 8,456评论 1 13
  • 5.整数的同余 一个固定的整数d去除整数(Zi)如果余数相同,我们则称这组整数Zi是模d同余的,例如,2,7,...
    jackjianshu阅读 4,544评论 0 0