From 0c265ec4da492286e6960bfeeaf1b7cfd1914dfd Mon Sep 17 00:00:00 2001 From: Aleksander Zaruczewski Date: Tue, 27 Feb 2024 21:04:10 +0200 Subject: [PATCH] test(acctest): incremental run support --- .../incremental-acceptance-tests.yml | 421 ++++++++++++++++++ 1 file changed, 421 insertions(+) create mode 100644 .github/workflows/incremental-acceptance-tests.yml diff --git a/.github/workflows/incremental-acceptance-tests.yml b/.github/workflows/incremental-acceptance-tests.yml new file mode 100644 index 000000000..d488301cc --- /dev/null +++ b/.github/workflows/incremental-acceptance-tests.yml @@ -0,0 +1,421 @@ +name: Incremental Acceptance Tests + +on: + pull_request: + branches: + - main + - v* + types: + - opened + - synchronize + - reopened + - labeled + - unlabeled + push: + branches: + - main + - v* + workflow_dispatch: {} + +permissions: read-all + +concurrency: + group: ci-${{ github.head_ref || github.ref }} + cancel-in-progress: true + +jobs: + selproj: + name: Run selproj + runs-on: ubuntu-latest + if: > + (github.event_name == 'pull_request' && !contains(github.event.pull_request.labels.*.name, 'skip workflows')) || + github.event_name == 'push' + + outputs: + suffix: ${{ steps.selproj.outputs.suffix }} + + steps: + - uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - uses: actions/setup-go@v5 + with: + go-version-file: go.mod + + - id: selproj + run: echo "suffix=$(make -s ci-selproj | tr -d '\n')" >> $GITHUB_OUTPUT + env: + AIVEN_TOKEN: ${{ secrets.AIVEN_TOKEN }} + AIVEN_PROJECT_NAME_PREFIX: ${{ secrets.AIVEN_PROJECT_NAME_PREFIX }} + + find_sdkv2_entities: + name: Find SDKv2 entities + runs-on: ubuntu-latest + if: > + (github.event_name == 'pull_request' && !contains(github.event.pull_request.labels.*.name, 'skip workflows')) || + github.event_name == 'push' + + env: + SDKV2_IMPL_PATH: internal/sdkprovider + + outputs: + entities: ${{ steps.find_entities.outputs.entities }} + + steps: + - name: Checkout + uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: Find provider.go file + id: find_provider_go_file + run: | + echo "path=$(find "$SDKV2_IMPL_PATH" -type f -name 'provider.go')" >> $GITHUB_OUTPUT + + - name: Extract imports + id: extract_imports + run: | + # Retrieve the path of the Go file for the provider from the previous step's outputs + provider_go_file=${{ steps.find_provider_go_file.outputs.path }} + + # Extract the import block from the Go file, starting from 'import (' until the closing parenthesis ')' + imports=$(awk '/^import \(/{flag=1;next}/^\)/{flag=0}flag' "$provider_go_file") + + # Format the extracted imports, removing new lines, spaces, and quotes, then convert to a JSON array + imports=$(echo "$imports" | jq -R -s -c 'split("\n") | map(select(length > 0) | gsub("[[:space:]\"]";""))') + + # Append the formatted imports to the GitHub Actions output variable for use in subsequent steps + echo "imports=$imports" >> $GITHUB_OUTPUT + + - name: Build import map + id: build_import_map + run: | + # Extract import lines from the given output variable and parse as JSON + import_lines=$(echo '${{ steps.extract_imports.outputs.imports }}' | jq -r '.[]') + import_map="{}" # Initialize an empty JSON object for the import map + + # Read each line from the extracted import lines + while IFS= read -r line; do + # Check if the line matches the expected pattern with SDKV2_IMPL_PATH + if [[ $line =~ ^(.*)"$SDKV2_IMPL_PATH"/(.*)$ ]]; then + # Extract the package path and name from the matched line + package_path="${BASH_REMATCH[2]}" + package_name=$(basename "$package_path") + + # Log the found package name and path for debugging + echo "Found package $package_name in $SDKV2_IMPL_PATH/$package_path" + + # Update the import map JSON object with the new package name and path + import_map=$( + echo "$import_map" | jq -c --arg key "$package_name" --arg value "$package_path" \ + '. + {($key): $value}' + ) + fi + done <<< "$import_lines" + + # Append the final import map JSON object to the GitHub action's output variable + echo "import_map=$import_map" >> $GITHUB_OUTPUT + + - name: Find entities + id: find_entities + run: | + # Define the path to the provider Go file from previous steps + provider_go_file=${{ steps.find_provider_go_file.outputs.path }} + # Define the import map from previous steps + import_map='${{ steps.build_import_map.outputs.import_map }}' + # Initialize entities as an empty JSON array + entities="[]" + + # Read each line containing name, type, and function with parentheses + while read -r name type func_with_parentheses; do + # Skip the iteration if name or function with parentheses is missing + if [[ -z "$name" || -z "$func_with_parentheses" ]]; then + continue + fi + + # Extract function name from func_with_parentheses + func="${func_with_parentheses%)*}" + func="${func##*.}" + # Extract package name from func_with_parentheses + package_name="${func_with_parentheses%%.*}" + # Get the path from the import map using the package name + path=$(echo $import_map | jq -r --arg package_name "$package_name" '.[$package_name] // empty') + + # Check if path is not empty + if [[ -n "$path" ]]; then + # Prepend the SDKV2 implementation path to the found path + path="$SDKV2_IMPL_PATH/$path" + # Find the Go file containing the function within the path + file=$(find "$path" -type f -name '*.go' -exec grep -l "$func" {} +) + + # If a file is found, print the information and add to entities + if [[ -n "$file" ]]; then + echo "Found $type $name in $file" + entities=$( + echo "$entities" | jq -c --arg name "$name" --arg type "$type" --arg file "$file" \ + '. + [{($name): {("type"): $type, ("file"): $file}}]' + ) + fi + fi + done < <(awk -v file="$provider_go_file" ' + # Start capturing after ResourcesMap declaration + /ResourcesMap: map\[string\]\*schema\.Resource{/ { type="resource"; capture = 1; next } + # Start capturing after DataSourcesMap declaration + /DataSourcesMap: map\[string\]\*schema\.Resource{/ { type="datasource"; capture = 1; next } + # Stop capturing after encountering a closing brace + capture && /\}/ { capture = 0 } + # Process each captured line + capture && NF { + # Remove tabs, spaces, and quotes + gsub(/[\t "]+/, "") + # Split the line by ':' and print name, type, and function + split($0, a, ":") + print a[1], type, a[2] + } + ' "$provider_go_file") + + # Append entities to the GitHub output + echo "entities=$entities" >> $GITHUB_OUTPUT + + find_plugin_framework_entities: + name: Find Plugin Framework entities + runs-on: ubuntu-latest + if: > + (github.event_name == 'pull_request' && !contains(github.event.pull_request.labels.*.name, 'skip workflows')) || + github.event_name == 'push' + + env: + PLUGIN_FRAMEWORK_IMPL_PATH: internal/plugin + + outputs: + entities: ${{ steps.find_entities.outputs.entities }} + + steps: + - name: Checkout + uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: Find entities + id: find_entities + run: | + entities="[]" + + while IFS= read -r file; do + while IFS=, read -r name type file; do + echo "Found $type $name in $file" + entities=$( + echo "$entities" | jq -c --arg name "$name" --arg type "$type" --arg file "$file" \ + '. + [{($name): {("type"): $type, ("file"): $file}}]' + ) + done < <(awk -v file="$file" ' + /resp\.TypeName\s*=\s*req\.ProviderTypeName\s*\+\s*"/ { + match($0, /"(_[^"]+)"/, arr) + if (length(arr[1]) > 0) { + type = "datasource" + while((getline line < file) > 0) { + if (line ~ /_ resource\.Resource/) { + type = "resource" + break + } + } + close(file) + print "aiven" arr[1] "," type "," file + } + } + ' "$file") + done < <(find "$PLUGIN_FRAMEWORK_IMPL_PATH" -type f -name '*.go') + + echo "entities=$entities" >> $GITHUB_OUTPUT + + combine_entities: + name: Combine entities + runs-on: ubuntu-latest + + needs: + - find_sdkv2_entities + - find_plugin_framework_entities + + outputs: + entities: ${{ steps.combine_entities.outputs.entities }} + + steps: + - name: Combine entities + id: combine_entities + run: | + # Initialize an empty JSON array to hold the entity details + entities="[]" + + # Find all '.go' files within the specified path and process each file + while IFS= read -r file; do + # Search for specific patterns within the file and extract relevant information + while IFS=, read -r name type file; do + # Log the found entities + echo "Found $type $name in $file" + + # Update the JSON array with the found entity details + entities=$( + echo "$entities" | jq -c --arg name "$name" --arg type "$type" --arg file "$file" \ + '. + [{($name): {("type"): $type, ("file"): $file}}]' + ) + done < <(awk -v file="$file" ' + /resp\.TypeName\s*=\s*req\.ProviderTypeName\s*\+\s*"/ { + match($0, /"(_[^"]+)"/, arr) + if (length(arr[1]) > 0) { + type = "datasource" + while((getline line < file) > 0) { + if (line ~ /_ resource\.Resource/) { + type = "resource" + break + } + } + close(file) + print "aiven" arr[1] "," type "," file + } + } + ' "$file") + done < <(find "$PLUGIN_FRAMEWORK_IMPL_PATH" -type f -name '*.go') + + # Output the final entities JSON array to the GitHub action's output + echo "entities=$entities" >> $GITHUB_OUTPUT + + find_tests: + name: Find tests + runs-on: ubuntu-latest + + needs: + - combine_entities + + outputs: + tests: ${{ steps.find_tests.outputs.tests }} + + steps: + - name: Checkout + uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: Set up Go + uses: actions/setup-go@v5 + with: + go-version-file: go.mod + + - name: Find tests + id: find_tests + run: | + # Get a list of changed test files that match the pattern '_test.go' and store in 'changed_test_files' + git diff "origin/${{ github.base_ref || 'main' }}" --name-only | grep '_test\.go$' > changed_test_files + + # Read the combined entities from previous job outputs, decode JSON into bash array + mapfile -t entities < <(echo '${{ needs.combine_entities.outputs.entities }}' | jq -c '.[]') + + # Declare associative array to hold entity tests + declare -A entity_tests + + # Loop through each changed test file + while read -r test_file; do + # Read each line of the test file + while read -r line; do + # Check if the line declares a resource or data entity + if [[ $line =~ resource\ \"([^\"]+)\" || $line =~ data\ \"([^\"]+)\" ]]; then + entity="${BASH_REMATCH[1]}" + echo "Found entity $entity in changed test file $test_file" + # Add the test file to the entity in the associative array + entity_tests[$entity]+="$test_file " + fi + done < "$test_file" + done < changed_test_files + + # Loop through each entity from the previous job's output + for entity in "${entities[@]}"; do + # Extract key information about the entity + name=$(echo "$entity" | jq -r 'keys[0]') + type=$(echo "$entity" | jq -r --arg name "$name" '.[$name].type') + file=$(echo "$entity" | jq -r --arg name "$name" '.[$name].file') + dir=$(dirname "$file") + + # Check if the entity's file has changed + file_changed=$(git diff "origin/${{ github.base_ref || 'main' }}" --name-only | grep -q "^$file$" && echo true || echo false) + dep_changed=false + + # Check if there are changed test files associated with the entity + if [[ ${entity_tests[$name]+_} ]]; then + echo "Test involving $name $type was changed" + elif [[ $file_changed == true ]]; then + echo "Definition of $name $type was changed" + else + # Resolve repository URL for dependency check + repo_url="${{ github.server_url }}/${{ github.repository }}" + repo_url="${repo_url/https:\/\/}" + + # List all dependencies for the entity's directory and check for changes + deps=$(go list -json "./$dir" | jq -r '.Deps[]' | grep -oP "^$repo_url/\K.*") + for dep in $deps; do + if git diff "origin/${{ github.base_ref || 'main' }}" --name-only | grep -q "$dep"; then + echo "Dependency $dep of $name $type changed" + dep_changed=true + break + fi + done + fi + + # If there are any changes related to the entity, find and list relevant test files + if [[ $file_changed == true || $dep_changed == true || ${entity_tests[$name]+_} ]]; then + pattern=$([[ "$type" == "resource" ]] && echo "resource \"$name\"" || echo "data \"$name\"") + while IFS= read -r test_file; do + echo "Found test for $name $type in $test_file" + entity_tests[$name]+="$test_file " + done < <(find internal -type f -name '*_test.go' -exec grep -l "$pattern" {} +) + fi + done + + # Compile a unique list of all test files that need to be run + tests=() + for test_files in "${entity_tests[@]}"; do + for test_file in $test_files; do + [[ ! " ${tests[*]} " =~ " ${test_file} " ]] && tests+=("$test_file") + done + done + + # Convert the list of tests to JSON and add to the GitHub Actions output + tests_json=$(printf '%s\n' "${tests[@]}" | jq -R . | jq -s -c .) + echo "tests=$tests_json" >> $GITHUB_OUTPUT + + run_tests: + name: Run tests + runs-on: ubuntu-latest + + needs: + - selproj + - find_tests + + env: + COUNT: 1 + PARALLEL: 10 + TIMEOUT: 180m + + strategy: + max-parallel: 5 + matrix: + test: ${{ fromJson(needs.find_tests.outputs.tests) }} + + steps: + - uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: Set up Go + uses: actions/setup-go@v5 + with: + go-version-file: go.mod + + - name: Run tests + run: go test ${{ matrix.test }} -v -count $COUNT -parallel $PARALLEL -timeout $TIMEOUT + env: + TF_ACC: 1 + CGO_ENABLED: 0 + AIVEN_TOKEN: ${{ secrets.AIVEN_TOKEN }} + AIVEN_PROJECT_NAME: ${{ secrets.AIVEN_PROJECT_NAME_PREFIX }}${{ needs.selproj.outputs.suffix }} + AIVEN_ORGANIZATION_NAME: ${{ secrets.AIVEN_ORGANIZATION_NAME }} + AIVEN_ACCOUNT_NAME: ${{ secrets.AIVEN_ORGANIZATION_NAME }}