2

我正在尝试为我的微服务创建集成测试,类似于Spotify 的方法

我仍在研究如何启动和播种数据库。目前我有一个带有FluentDocker v2.2.15 和DbUp 4.1.0 的 .NET Core 2.0 项目。

我使用 FluentDocker 调用 DockerCompose 并启动我的服务,包括SQL Server 容器

var hosts = new Hosts().Discover();
var dockerHost = hosts.FirstOrDefault(x => x.IsNative) ?? hosts.FirstOrDefault(x => x.Name == "default");

if (dockerHost == null)
{
    return;
}

var composeFile = Args["composeFile"];

var result = dockerHost.Host.ComposeUp(composeFile: composeFile);

然后我使用 DbUp 运行我的脚本并为数据库播种。

var connectionString = Args["connectionString"];
var scriptsPath = Args["scriptsPath"];

EnsureDatabase.For.SqlDatabase(connectionString);

var upgradeEngine = DeployChanges.To.SqlDatabase(connectionString).WithScriptsFromFileSystem(scriptsPath).Build();

var result = upgradeEngine.PerformUpgrade();

当我给 SQL Server 足够的时间来启动它时,例如在调试时,我可以成功运行它。但是,如果我全速运行它,那么 DbUp 会在它尚未准备好时尝试连接到 SQL Server。FluentDocker 有一个WaitForPort方法,但它似乎不适用于 DockerCompose API。

我想知道是否有办法在运行脚本之前等待 SQL Server 的端口 1433 响应(不包括非确定性策略,例如await Task.Delay),或者是否有其他库允许我进行这种控制。

谢谢

4

1 回答 1

2

您可以在FluentDocker v2.6.2中的 compose 上使用WaitForPortWaitForProcessWaitForHttp或自定义 lambdaWait函数。例如:

给定 docker-compose 文件:

version: '3.3'
services:
  db:
    image: mysql:5.7
    volumes:
    - db_data:/var/lib/mysql
    restart: always
    environment:
      MYSQL_ROOT_PASSWORD: somewordpress
      MYSQL_DATABASE: wordpress
      MYSQL_USER: wordpress
      MYSQL_PASSWORD: wordpress

  wordpress:
    depends_on:
    - db
    image: wordpress:latest
    ports:
    - "8000:80"
    restart: always
    environment:
      WORDPRESS_DB_HOST: db:3306
      WORDPRESS_DB_USER: wordpress
      WORDPRESS_DB_PASSWORD: wordpress
volumes:
  db_data:

该文件指定 wordpress 依赖于 db,因此首先启动 db,然后实例化 wordpress 容器。但是,要确保在wordpressusing web 正在运行的子句中,您需要使用HTTP来确定这一点。您可以使用Fluent API来实例化并等待服务像这样启动。

  var file = Path.Combine(Directory.GetCurrentDirectory(),
    (TemplateString) "Resources/ComposeTests/WordPress/docker-compose.yml");

using (new Builder()
            .UseContainer()
            .UseCompose()
            .FromFile(file)
            .RemoveOrphans()
            .Wait("wordpress", (service, cnt) => {
                if (cnt > 60) throw new FluentDockerException("Failed to wait for wordpress service");

                var res = HttpExtensions.DoRequest("http://localhost:8000/wp-admin/install.php").Result;            
                return (res.Code == HttpStatusCode.OK && 
                        res.Body.IndexOf("https://wordpress.org/", StringComparison.Ordinal) != -1) ? 0 : 500;
              })
            .Build().Start())
  {
    // Since we have waited - this shall now always work.       
    var installPage = await "http://localhost:8000/wp-admin/install.php".Wget();
    Assert.IsTrue(installPage.IndexOf("https://wordpress.org/", StringComparison.Ordinal) != -1);
  }

(上面的示例比较麻烦WaitForHttp,但使用自定义 lambda 来说明这一点)。

通过这种方式,您甚至可以使用数据库连接并在继续之前查询表。高于零的返回值是等待下一次测试的时间。零及以下将成功结束等待。异常将终止等待(并失败)。

上面的示例使用FluentAPI语法,但您可以手动在 compose 容器上添加挂钩并自行使用扩展。

干杯,马里奥

于 2018-10-03T16:24:23.053 回答