mirror of
https://github.com/obi1kenobi/cargo-semver-checks-action.git
synced 2024-11-22 07:59:32 +01:00
Add initial cargo-semver-checks action implementation.
This commit is contained in:
parent
192d7c246b
commit
ebc272edf7
3 changed files with 140 additions and 0 deletions
57
action.yml
Normal file
57
action.yml
Normal file
|
@ -0,0 +1,57 @@
|
|||
name: 'cargo-semver-checks'
|
||||
description: 'Ensure your Rust crate's public API follows semantic versioning'
|
||||
inputs:
|
||||
crate-name:
|
||||
description: 'The crate whose API to check for semver'
|
||||
required: false
|
||||
default: ''
|
||||
version-tag-prefix:
|
||||
description: 'The prefix to use for the git tag for a version; the default "v" creates tags like "v1.0.0"'
|
||||
required: false
|
||||
default: 'v'
|
||||
runs:
|
||||
using: "composite"
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Install rust
|
||||
uses: actions-rs/toolchain@v1
|
||||
with:
|
||||
toolchain: nightly
|
||||
profile: minimal
|
||||
- run: |
|
||||
# Colorize output, since GitHub Actions terminals support color.
|
||||
export CARGO_TERM_COLOR=always
|
||||
|
||||
# Record the current git sha, so we can come back to it after generating the baseline.
|
||||
export CURRENT_GIT_SHA="$(git rev-parse HEAD)"
|
||||
|
||||
# Ensure this action's scripts are available to run on the path.
|
||||
echo "${{ github.action_path }}" >> $GITHUB_PATH
|
||||
|
||||
export PACKAGE_NAME="${{ inputs.crate-name }}"
|
||||
if [[ "$PACKAGE_NAME" == '' ]]; then
|
||||
export PACKAGE_NAME="$(find_workspace_crates.sh)"
|
||||
fi
|
||||
export PACKAGE_NAME_WITH_UNDERSCORES="$(echo $"PACKAGE_NAME" | tr '-' '_')"
|
||||
|
||||
# Switch to the tag for the correct baseline version,
|
||||
# then build rustdoc JSON.
|
||||
#
|
||||
# We *do not* want to record and reuse the target directory path
|
||||
# across different git commits, since it may be at a different location
|
||||
# in different commits.
|
||||
git checkout "${{ inputs.version-tag-prefix }}$(find_comparison_version.sh "$PACKAGE_NAME")"
|
||||
cargo +nightly rustdoc -- -Zunstable-options --output-format json
|
||||
mv "$(cargo metadata --format-version 1 | jq -r .target_directory)/doc/$PACKAGE_NAME_WITH_UNDERSCORES.json" /tmp/baseline.json
|
||||
|
||||
# Return to the original git sha.
|
||||
git checkout "$CURRENT_GIT_SHA"
|
||||
|
||||
# Build rustdoc JSON for the current version, and move it to /tmp/
|
||||
# so it doesn't get overwritten by the baseline build.
|
||||
cargo +nightly rustdoc -- -Zunstable-options --output-format json
|
||||
mv "$(cargo metadata --format-version 1 | jq -r .target_directory)/doc/$PACKAGE_NAME_WITH_UNDERSCORES.json" /tmp/current.json
|
||||
|
||||
# Check for semver violations.
|
||||
cargo install cargo-semver-checks
|
||||
cargo semver-checks check-release --current /tmp/current.json --baseline /tmp/baseline.json
|
60
find_comparison_version.sh
Normal file
60
find_comparison_version.sh
Normal file
|
@ -0,0 +1,60 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
# Script requirements:
|
||||
# - curl
|
||||
# - jq
|
||||
# - sort with `-V` flag, available in `coreutils-7`
|
||||
# On macOS this may require `brew install coreutils`.
|
||||
|
||||
# Fail on first error, on undefined variables, and on failures in pipelines.
|
||||
set -euo pipefail
|
||||
|
||||
# Go to the repo root directory.
|
||||
cd "$(git rev-parse --show-toplevel)"
|
||||
|
||||
# The first argument should be the name of a crate.
|
||||
CRATE_NAME="$1"
|
||||
|
||||
CURRENT_VERSION="$( \
|
||||
cargo metadata --format-version 1 | \
|
||||
jq --arg crate_name "$CRATE_NAME" --exit-status -r \
|
||||
'.packages[] | select(.name == $crate_name) | .version' \
|
||||
)" || (echo >&2 "No crate named $CRATE_NAME found in workspace."; exit 1)
|
||||
echo >&2 "Crate $CRATE_NAME current version: $CURRENT_VERSION"
|
||||
|
||||
# The leading whitespace is important! With it, we know that every version is both
|
||||
# preceded by and followed by whitespace. We use this fact to avoid matching
|
||||
# on substrings of versions.
|
||||
EXISTING_VERSIONS="
|
||||
$( \
|
||||
curl 2>/dev/null "https://crates.io/api/v1/crates/$CRATE_NAME" | \
|
||||
jq --exit-status -r .versions[].num \
|
||||
)"
|
||||
echo >&2 -e "Versions on crates.io:$EXISTING_VERSIONS\n"
|
||||
|
||||
# Use version sort (sort -V) to get all versions in ascending order, then use grep to:
|
||||
# - grab the first line that matches the current version (--max-count=1)
|
||||
# - only match full lines (--line-regexp)
|
||||
# - get one line of leading context (-B 1) i.e. the immediately-smaller version, if one exists
|
||||
# - explicitly opt out of trailing context lines (-A 0)
|
||||
# Finally, use `head` to output only the first of the up-to-two lines output.
|
||||
# Now, either:
|
||||
# - two lines were output, and we grabbed the immediately-smaller version, or
|
||||
# - one line was output with only our version, because there was no immediately-smaller version,
|
||||
# and we grabbed that one. We sort this out with the subsequent conditional.
|
||||
OUTPUT="$( \
|
||||
echo -e "$CURRENT_VERSION$EXISTING_VERSIONS" | \
|
||||
sort -V | \
|
||||
grep -B 1 -A 0 --line-regexp --max-count=1 "$CURRENT_VERSION" | \
|
||||
head -n 1 \
|
||||
)"
|
||||
|
||||
if [[ "$OUTPUT" == "$CURRENT_VERSION" ]]; then
|
||||
echo >&2 "There is no suitable comparison version."
|
||||
echo >&2 \
|
||||
"The current version $CURRENT_VERSION is smaller than any version published on crates.io"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo "Comparison version: $OUTPUT" >&2
|
||||
echo "$OUTPUT"
|
23
find_workspace_crates.sh
Normal file
23
find_workspace_crates.sh
Normal file
|
@ -0,0 +1,23 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
# Script requirements:
|
||||
# - jq
|
||||
|
||||
# Fail on first error, on undefined variables, and on failures in pipelines.
|
||||
set -euo pipefail
|
||||
|
||||
# Go to the repo root directory.
|
||||
cd "$(git rev-parse --show-toplevel)"
|
||||
|
||||
crates="$(cargo metadata --format-version 1 | \
|
||||
jq --exit-status -r \
|
||||
'.workspace_members[] as $key | .packages[] | select(.id == $key) | .name')"
|
||||
crate_count="$(echo -e "${crates}" | wc -l)"
|
||||
|
||||
if [[ "$crate_count" == "1" ]]; then
|
||||
echo -e "${crates}"
|
||||
exit 0
|
||||
else
|
||||
echo >&2 "Multiple crates in workspace, please specify a crate in the 'crate-name' setting."
|
||||
exit 1
|
||||
fi
|
Loading…
Reference in a new issue