docker run dynamodb-local 在 Github 上操作工作流程挂起

Mic*_*ert 4 docker github-actions

我目前正在开发一个小型 CICD 项目,每当我更新代码时,该项目都会使用 dynamodb-local 在 Github Actions 上运行一系列测试,如果测试成功则打包和部署。

我有以下工作流程:

name: backend_actions
on:
  workflow_dispatch:
  push:
    paths:
      - 'backend/*'
    branches:
      - master
jobs:
  test-locally:
    runs-on: ubuntu-latest
    outputs:
      test-result: ${{ steps.run-tests.outputs.result }}
    steps:
      - uses: actions/checkout@v2
      - uses: actions/setup-python@v2
        with:
          python-version: '3.9'
      - uses: aws-actions/setup-sam@v1
      - uses: aws-actions/configure-aws-credentials@v1
        with:
          aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
          aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
          aws-region: us-west-2
      - name: Install dependencies
        run: |
          python -m pip install --upgrade pip
          pip install -r requirements.txt
      - name: Setup local DynamoDB
        run: docker run -p 8000:8000 amazon/dynamodb-local
      - name: Create table
        run: aws dynamodb create-table --cli-input-json file://backend/src/test/make_table.json --endpoint-url http://localhost:8000
      - name: start local API Gateway
        run: sam local start-api --env-vars backend/env.json
      - id:   run-tests
        name: Run tests
        run: |
          python backend/src/test_dynamoDB_lambda.py
          echo "::set-output name=result::$?"
  update_backend:
    needs: test-locally
    if: ${{ needs.test-locally.outputs.test-result == '0' }}
    runs-on: ubuntu-latest
    steps:
      - name: Package and deploy
        run: |
          aws cloudformation package --s3-bucket cloud-resume-bucket \
          --template-file backend/template.yaml --output-template-file backend/gen/template-gen.yaml
          aws cloudformation deploy --template-file backend/gen/template-gen.yaml --stack-name cloud-formation-resume \
          --capabilities CAPABILITY_IAM
Run Code Online (Sandbox Code Playgroud)

当我尝试在 Github Actions 中运行工作流程时,它将进入“设置本地 DynamoDB”步骤,输出下面的文本,然后挂起。

Run docker run -p 8000:8000 amazon/dynamodb-local
Unable to find image 'amazon/dynamodb-local:latest' locally
latest: Pulling from amazon/dynamodb-local
2cbe74538cb5: Pulling fs layer
137077f50205: Pulling fs layer
58932e640a40: Pulling fs layer
58932e640a40: Verifying Checksum
58932e640a40: Download complete
2cbe74538cb5: Verifying Checksum
2cbe74538cb5: Download complete
137077f50205: Verifying Checksum
137077f50205: Download complete
2cbe74538cb5: Pull complete
137077f50205: Pull complete
58932e640a40: Pull complete
Digest: sha256:bdd26570dc0e0ae49e1ea9d49ff662a6a1afe9121dd25793dc40d02802e7e806
Status: Downloaded newer image for amazon/dynamodb-local:latest
Initializing DynamoDB Local with the following configuration:
Port:   8000
InMemory:   true
DbPath: null
SharedDb:   false
shouldDelayTransientStatuses:   false
CorsParams: *
Run Code Online (Sandbox Code Playgroud)

似乎它可以找到 docker 映像并正常下载,但在初始化时停止?这是我第一次使用 Github Actions 和 Docker,所以我不太确定为什么它挂在 Github Actions 上,而不是当我在自己的计算机上运行它时挂起,所以任何帮助将不胜感激!

小智 5

当您运行命令时,docker run -p 8000:8000 amazon/dynamodb-local进程永远不会退出,因此 Githubrun块实际上不知道何时继续下一步\xe2\x80\x94,它只是永远挂在那里。

\n

我在我的项目中所做的只是背景它,通过使用&命令后的背景来设置它:

\n
      - name: Setup local DynamoDB\n        run: docker run -d -p 8000:8000 amazon/dynamodb-local\n
Run Code Online (Sandbox Code Playgroud)\n

Github Workflows 将启动 Docker 容器并移至下一个run,当所有步骤完成后,它只会杀死容器作为正常清理的一部分。因此,您无需担心最后将其关闭。

\n

这种方法的困难在于,DynamoDB-local 启动需要几秒钟的时间,但您的下一步依赖于它,并且可能会抛出 ECONNREFUSED 错误。

\n

我在我的项目中所做的就是让下一个run执行一个尝试列出表的脚本,并在短暂延迟后重试,直到收到响应。

\n

bash 命令很简单(您需要将其放入 while+try/catch 循环中):

\n
aws dynamodb list-tables --endpoint-url http://localhost:8000\n
Run Code Online (Sandbox Code Playgroud)\n

作为指导,这(大致)是我使用aws-sdk在 JavaScript 中所做的事情和 NodeJS@16 在 JavaScript 中所做的事情:

\n
// wait-for-dynamodb.js\nimport timers from \'timers/promises\'\nimport AWS from \'aws-sdk\'\n\nconst dynamodb = new AWS.DynamoDB()\n\nconst waitForDynamoDbToStart = async () => {\n    try {\n        await dynamodb.listTables().promise()\n    } catch (error) {\n        console.log(\'Waiting for Docker container to start...\')\n        await timers.setTimeout(500)\n        return waitForDynamoDbToStart()\n    }\n}\n\nconst start = Date.now()\nwaitForDynamoDbToStart()\n    .then(() => {\n        console.log(`DynamoDB-local started after ${Date.now() - start}ms.`)\n        process.exit(0)\n    })\n    .catch(error => {\n        console.log(\'Error starting DynamoDB-local!\', error)\n        process.exit(1)\n    })\n
Run Code Online (Sandbox Code Playgroud)\n

然后我就把它放在run步骤操作即可:

\n
      - name: Setup local DynamoDB\n        run: docker run -p 8000:8000 amazon/dynamodb-local &\n      - name: Wait for it to boot up\n        run: node ./wait-for-dynamodb.js\n      # now you\'re guaranteed to have DynamoDB-local running\n
Run Code Online (Sandbox Code Playgroud)\n