Skip to content

lint_random_files

lint_random_files #180

#/
# @license Apache-2.0
#
# Copyright (c) 2022 The Stdlib Authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#/
# Workflow name:
name: lint_random_files
# Workflow triggers:
on:
# Allow the workflow to be manually run:
workflow_dispatch:
inputs:
num:
type: string
description: 'Maximum number of files to lint:'
default: '100'
pattern:
description: 'Regular expression for files to include:'
default: .*
javascript:
type: boolean
description: 'Lint JavaScript and TypeScript'
default: true
markdown:
type: boolean
description: 'Lint Markdown'
default: true
json:
type: boolean
description: 'Lint JSON'
default: true
repl:
type: boolean
description: 'Lint REPL documentation and shell script files'
default: true
r:
type: boolean
description: 'Lint R'
default: true
c:
type: boolean
description: 'Lint C'
default: true
python:
type: boolean
description: 'Lint Python'
default: true
fix:
type: boolean
description: 'Fix lint errors and submit a PR (if possible)'
default: false
# Trigger the workflow every 24 hours:
schedule:
- cron: '0 0 * * *'
# Global permissions:
permissions:
# Allow read-only access to the repository contents:
contents: read
# Workflow jobs:
jobs:
# Define a job for linting committed code...
lint:
# Define a display name:
name: 'Lint Random Files'
# Ensure the job does not run on forks:
if: github.repository == 'stdlib-js/stdlib'
# Define the type of virtual host machine:
runs-on: ubuntu-latest
# Define the sequence of job steps...
steps:
# Checkout the repository:
- name: 'Checkout repository'
# Pin action to full length commit SHA
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
with:
# Specify whether to remove untracked files before checking out the repository:
clean: true
# Limit clone depth to the most recent commit:
fetch-depth: 1
# Specify whether to download Git-LFS files:
lfs: false
timeout-minutes: 10
# Install Node.js:
- name: 'Install Node.js'
# Pin action to full length commit SHA
uses: actions/setup-node@1d0ff469b7ec7b3cb9d8673fde0c81c44821de2a # v4.2.0
with:
node-version: '20' # 'lts/*'
timeout-minutes: 5
# Install dependencies (accounting for possible network failures, etc, when installing node module dependencies):
- name: 'Install dependencies'
run: |
make install-node-modules || make install-node-modules || make install-node-modules
timeout-minutes: 15
# Initialize development environment:
- name: 'Initialize development environment'
run: |
make init
timeout-minutes: 5
# Pick random files from the `lib/node_modules/@stdlib` directory:
- name: 'Pick random files from the `lib/node_modules/@stdlib` directory'
id: random-files
run: |
name=""
if [ "${{ github.event.inputs.javascript }}" != "false" ]; then
name="${name} -name '*.js'"
fi
if [ "${{ github.event.inputs.markdown }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.md'"
fi
if [ "${{ github.event.inputs.json }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.json'"
fi
if [ "${{ github.event.inputs.repl }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.repl\.txt' -o -name '*.sh'"
fi
if [ "${{ github.event.inputs.r }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.R'"
fi
if [ "${{ github.event.inputs.c }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.c'"
fi
if [ "${{ github.event.inputs.javascript }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.ts'"
fi
if [ "${{ github.event.inputs.python }}" != "false" ]; then
if [ -n "${name}" ]; then
name="${name} -o"
fi
name="${name} -name '*.py'"
fi
command="find lib/node_modules/@stdlib -type f \( ${name} \) |
grep -E '${{ github.event.inputs.pattern }}' |
grep -v '/fixtures/bad/' |
shuf -n ${{ github.event.inputs.num || 100 }} | tr '\n' ','"
files=$(eval ${command})
echo "files=$files" >> $GITHUB_OUTPUT
# Lint file names:
- name: 'Lint file names'
run: |
# Determine root directory:
root=$(git rev-parse --show-toplevel)
# Define the path to a utility for linting filenames:
lint_filenames="${root}/lib/node_modules/@stdlib/_tools/lint/filenames/bin/cli"
# Lint filenames:
echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | "${lint_filenames}"
# Lint files against EditorConfig:
- name: 'Lint against EditorConfig'
id: lint-editorconfig
run: |
set -o pipefail
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' ' ')
make lint-editorconfig-files FILES="${files}" 2>&1 | tee lint_editorconfig_errors.txt
# Create sub-issue for EditorConfig lint failures:
- name: 'Create sub-issue for EditorConfig lint failures'
if: failure() && contains(steps.lint-editorconfig.outcome, 'failure')
env:
GITHUB_TOKEN: ${{ secrets.STDLIB_BOT_PAT_REPO_WRITE }}
run: |
strip_ansi() {
sed -r 's/\x1B\[[0-9;]*[mK]//g'
}
BODY_FILE="$GITHUB_WORKSPACE/lint_issue_body.md"
cat << EOF > "$BODY_FILE"
## EditorConfig Linting Failures
Linting failures were detected in the automated EditorConfig lint workflow run.
### Workflow Details
- Run: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}
- Type: EditorConfig Linting
- Date: $(date -u +"%Y-%m-%d %H:%M:%S UTC")
### Error Details
\`\`\`
$(cat lint_editorconfig_errors.txt | strip_ansi)
\`\`\`
EOF
. "$GITHUB_WORKSPACE/.github/workflows/scripts/create_sub_issue" \
'Fix EditorConfig lint errors' \
"$BODY_FILE" \
"5156" \
"Good First Issue"
rm "$BODY_FILE"
# Lint Markdown files:
- name: 'Lint Markdown files'
if: ( github.event.inputs.markdown != 'false' ) && ( success() || failure() )
run: |
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep -E '\.md$' | tr '\n' ' ')
if [ -n "${files}" ]; then
make lint-markdown-files FAST_FAIL=0 FILES="${files}"
fi
# Lint package.json files:
- name: 'Lint package.json files'
if: ( github.event.inputs.json != 'false' ) && ( success() || failure() )
run: |
# Determine root directory:
root=$(git rev-parse --show-toplevel)
# Define the path to a utility for linting package.json files:
lint_package_json="${root}/lib/node_modules/@stdlib/_tools/lint/pkg-json/bin/cli"
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep 'package\.json$' | grep -v 'datapackage\.json$' | tr '\n' ' ' | sed 's/ $//')
if [ -n "${files}" ]; then
printf "${files}" | "${lint_package_json}" --split=" "
fi
# Lint REPL help files...
- name: 'Lint REPL help files'
if: ( github.event.inputs.repl != 'false' ) && ( success() || failure() )
run: |
# Determine root directory:
root=$(git rev-parse --show-toplevel)
# Define the path to a utility for linting REPL help files:
lint_repl_help="${root}/lib/node_modules/@stdlib/_tools/lint/repl-txt/bin/cli"
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep 'repl\.txt$' | tr '\n' ' ' | sed 's/ $//')
if [ -n "${files}" ]; then
printf "${files}" | "${lint_repl_help}" --split=" "
fi
# Lint shell script files:
- name: 'Lint shell script files'
if: ( github.event.inputs.repl != 'false' ) && ( success() || failure() )
run: |
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep -vE '\.(js|md|json|ts|c|h)$' | while read -r file; do head -n1 "$file" | grep -q '^\#\!/usr/bin/env bash' && echo "$file"; done | tr '\n' ' ')
if [[ -n "${files}" ]]; then
# Install shellcheck:
make install-deps-shellcheck
# Lint shell scripts:
make FILES="${files}" lint-shell-files
fi
# Lint JavaScript files:
- name: 'Lint JavaScript files'
id: lint-javascript
if: ( github.event.inputs.javascript != 'false' ) && ( success() || failure() )
run: |
# If any command in a pipeline fails, the entire pipeline should fail:
set -o pipefail
# Determine root directory:
root=$(git rev-parse --show-toplevel)
# Define the path to ESLint configuration file for linting examples:
eslint_examples_conf="${root}/etc/eslint/.eslintrc.examples.js"
# Define the path to ESLint configuration file for linting tests:
eslint_tests_conf="${root}/etc/eslint/.eslintrc.tests.js"
# Define the path to ESLint configuration file for linting benchmarks:
eslint_benchmarks_conf="${root}/etc/eslint/.eslintrc.benchmarks.js"
# Set `FIX` variable to `1` to enable automatic fixing of lint errors, `0` to disable:
if [ "${{ github.event.inputs.fix }}" == "true" ]; then
FIX=1
else
FIX=0
fi
# Combined error file:
ERR_FILE="lint_javascript_errors.txt"
> "$ERR_FILE" # Initialize a clean file
# Lint JavaScript source files:
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '\.js$' | grep -v -e '/examples' -e '/test' -e '/benchmark' -e '^dist/' | tr '\n' ' ')
# Build native addons if present:
packages=$(echo "${files}" | tr ' ' '\n' | sed 's/^lib\/node_modules\///g' | sed 's/\/lib\/.*//g' | sort | uniq)
for pkg in ${packages}; do
if [ -f "lib/node_modules/${pkg}/binding.gyp" ]; then
NODE_ADDONS_PATTERN="${pkg}" make install-node-addons
fi
done
if [[ -n "${files}" ]]; then
make lint-javascript-files FIX="${FIX}" FAST_FAIL=0 FILES="${files}" 2>&1 | tee -a "$ERR_FILE"
fi
# Lint JavaScript command-line interfaces...
file=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '\.js$' | grep -E '/bin/cli$' | tr '\n' ' ')
if [[ -n "${file}" ]]; then
make lint-javascript-files FIX="${FIX}" FAST_FAIL=0 FILES="${file}" 2>&1 | tee -a "$ERR_FILE"
fi
# Lint JavaScript example files:
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '/examples/.*\.js$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-javascript-files FIX="${FIX}" FAST_FAIL=0 FILES="${files}" ESLINT_CONF="${eslint_examples_conf}" 2>&1 | tee -a "$ERR_FILE"
fi
# Lint JavaScript test files:
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '/test/.*\.js$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-javascript-files FIX="${FIX}" FAST_FAIL=0 FILES="${files}" ESLINT_CONF="${eslint_tests_conf}" 2>&1 | tee -a "$ERR_FILE"
fi
# Lint JavaScript benchmark files:
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '/benchmark/.*\.js$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-javascript-files FIX="${FIX}" FAST_FAIL=0 FILES="${files}" ESLINT_CONF="${eslint_benchmarks_conf}" 2>&1 | tee -a "$ERR_FILE"
fi
# Create sub-issue for JavaScript lint failures:
- name: 'Create sub-issue for JavaScript lint failures'
if: failure() && contains(steps.lint-javascript.outcome, 'failure')
env:
GITHUB_TOKEN: ${{ secrets.STDLIB_BOT_PAT_REPO_WRITE }}
run: |
BODY_FILE="$GITHUB_WORKSPACE/lint_issue_body.md"
cat << EOF > "$BODY_FILE"
## JavaScript Linting Failures
Linting failures were detected in the automated JavaScript lint workflow run.
### Workflow Details
- Run: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}
- Type: JavaScript Linting
- Date: $(date -u +"%Y-%m-%d %H:%M:%S UTC")
### Error Details
\`\`\`
$(cat lint_javascript_errors.txt)
\`\`\`
EOF
. "$GITHUB_WORKSPACE/.github/workflows/scripts/create_sub_issue" \
'Fix JavaScript lint errors' \
"$BODY_FILE" \
"5377" \
"Good First Issue"
rm "$BODY_FILE"
# Lint Python files:
- name: 'Lint Python files'
if: ( github.event.inputs.python != 'false' ) && ( success() || failure() )
run: |
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep -E '\.py$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
# Install Python dependencies:
make install-deps-python
# Lint Python files:
make lint-python-files FAST_FAIL=0 FILES="${files}"
fi
# Setup R:
- name: 'Setup R'
if: ( github.event.inputs.r != 'false' ) && ( success() || failure() )
# Pin action to full length commit SHA
uses: r-lib/actions/setup-r@14a7e741c1cb130261263aa1593718ba42cf443b # v2.11.2
with:
r-version: '4.3.3'
# Lint R files:
- name: 'Lint R files'
if: ( github.event.inputs.r != 'false' ) && ( success() || failure() )
run: |
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep -E '\.R$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
# Install R dependencies:
make install-deps-r
# Lint R files:
make lint-r-files FAST_FAIL=0 FILES="${files}"
fi
# Lint C files:
- name: 'Lint C files'
id: lint-c
if: ( github.event.inputs.c != 'false' ) && ( success() || failure() )
run: |
# Determine root directory:
root=$(git rev-parse --show-toplevel)
# Install Cppcheck:
make install-deps-cppcheck
# Define the path to cppcheck configuration file for linting examples:
cppcheck_examples_suppressions_list="${root}/etc/cppcheck/suppressions.examples.txt"
# Define the path to cppcheck configuration file for linting test fixtures:
cppcheck_tests_fixtures_suppressions_list="${root}/etc/cppcheck/suppressions.tests_fixtures.txt"
# Define the path to cppcheck configuration file for linting benchmarks:
cppcheck_benchmarks_suppressions_list="${root}/etc/cppcheck/suppressions.benchmarks.txt"
# Define output files for each category:
out_source="lint_c_errors_source.txt"
out_examples="lint_c_errors_examples.txt"
out_tests="lint_c_errors_tests.txt"
out_benchmarks="lint_c_errors_benchmarks.txt"
combined="lint_c_errors.txt"
# Initialize output files with empty content:
echo "" > "$out_source"
echo "" > "$out_examples"
echo "" > "$out_tests"
echo "" > "$out_benchmarks"
echo "" > "$combined"
# Initialize an error flag:
error=0
# Lint C source files:
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep -E '\.c$' | grep -v -e '/examples' -e '/test' -e '/benchmark' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-c-files FILES="${files}" CPPCHECK_FLAGS="--output-file=${out_source}" || error=1
fi
# Lint C example files...
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '/examples/.*\.c$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-c-files FILES="${files}" CPPCHECK_FLAGS="--output-file=${out_examples}" CPPCHECK_SUPPRESSIONS_LIST="${cppcheck_examples_suppressions_list}" || error=1
fi
# Lint C test fixtures files...
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '/test/fixtures/.*\.c$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-c-files FILES="${files}" CPPCHECK_FLAGS="--output-file=${out_tests}" CPPCHECK_SUPPRESSIONS_LIST="${cppcheck_tests_fixtures_suppressions_list}" || error=1
fi
# Lint C benchmark files...
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep '/benchmark/.*\.c$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make lint-c-files FILES="${files}" CPPCHECK_FLAGS="--output-file=${out_benchmarks}" CPPCHECK_SUPPRESSIONS_LIST="${cppcheck_benchmarks_suppressions_list}" || error=1
fi
# Combine all outputs into one file:
cat "$out_source" "$out_examples" "$out_tests" "$out_benchmarks" > "$combined"
# Print the combined errors to the log:
cat "$combined"
# If any linting command failed, exit with error:
if [[ $error -ne 0 ]]; then
exit 1
fi
# Create sub-issue for C linting failures:
- name: 'Create sub-issue for C lint failures'
if: ( github.event.inputs.c != 'false' ) && failure() && contains(steps.*.outcome, 'failure') && contains(steps.lint-c.outcome, 'failure')
env:
GITHUB_TOKEN: ${{ secrets.STDLIB_BOT_PAT_REPO_WRITE }}
run: |
BODY_FILE="$GITHUB_WORKSPACE/lint_issue_body.md"
cat << EOF > "$BODY_FILE"
## C Linting Failures
Linting failures were detected in the automated lint workflow run.
### Workflow Details
- Run: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}
- Type: C Linting
- Date: $(date -u +"%Y-%m-%d %H:%M:%S UTC")
### Error Details
\`\`\`
$(grep -B 1 -A 2 "style:\|warning:\|error:" "lint_c_errors.txt")
\`\`\`
EOF
. "$GITHUB_WORKSPACE/.github/workflows/scripts/create_sub_issue" \
'Fix C lint errors' \
"$BODY_FILE" \
"3235" # Number of C lint error tracking issue
rm "$BODY_FILE"
# Lint TypeScript declarations files:
- name: 'Lint TypeScript declarations files'
if: ( github.event.inputs.javascript != 'false' ) && ( success() || failure() )
run: |
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' '\n' | grep -E '\.d\.ts$' | tr '\n' ' ')
if [[ -n "${files}" ]]; then
make TYPESCRIPT_DECLARATIONS_LINTER=eslint lint-typescript-declarations-files FAST_FAIL=0 FILES="${files}"
fi
# Lint license headers:
- name: 'Lint license headers'
if: success() || failure()
run: |
files=$(echo "${{ steps.random-files.outputs.files }}" | tr ',' ' ')
if [[ -n "${files}" ]]; then
make lint-license-headers-files FILES="${files}"
fi
# Disable Git hooks:
- name: 'Disable Git hooks'
if: ${{ github.event.inputs.fix == 'true' }} && ( success() || failure() )
run: |
rm -rf .git/hooks
# Import GPG key to sign commits:
- name: 'Import GPG key to sign commits'
if: ${{ github.event.inputs.fix == 'true' }} && ( success() || failure() )
# Pin action to full length commit SHA
uses: crazy-max/ghaction-import-gpg@cb9bde2e2525e640591a934b1fd28eef1dcaf5e5 # v6.2.0
with:
gpg_private_key: ${{ secrets.STDLIB_BOT_GPG_PRIVATE_KEY }}
passphrase: ${{ secrets.STDLIB_BOT_GPG_PASSPHRASE }}
git_user_signingkey: true
git_commit_gpgsign: true
# Create a pull request with the fixes (if applicable):
- name: 'Create pull request'
if: ${{ github.event.inputs.fix == 'true' }} && ( success() || failure() )
id: cpr
# Pin action to full length commit SHA
uses: peter-evans/create-pull-request@67ccf781d68cd99b580ae25a5c18a1cc84ffff1f # v7.0.6
with:
title: 'style: fix lint errors'
add-paths: ${{ steps.random-files.outputs.files }}
body: |
This PR
- fixes lint errors
commit-message: 'style: resolve lint errors'
committer: 'stdlib-bot <[email protected]>'
signoff: true
token: ${{ secrets.STDLIB_BOT_PAT_REPO_WRITE }}
labels: |
automated-pr
team-reviewers: |
reviewers
branch: 'fix-lint-errors'
delete-branch: true