npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@mizchi/actrun

v0.22.1

Published

A local GitHub Actions runner

Readme

actrun

A local GitHub Actions runner built with MoonBit. Run and debug GitHub Actions workflows locally with a gh-compatible CLI.

Install

# npx (no install required)
npx @mizchi/actrun workflow run .github/workflows/ci.yml

# curl (Linux / macOS)
curl -fsSL https://raw.githubusercontent.com/mizchi/actrun/main/install.sh | sh

# Docker
docker run --rm -v "$PWD":/workspace -w /workspace ghcr.io/mizchi/actrun workflow run .github/workflows/ci.yml

# npm global install
npm install -g @mizchi/actrun

# moon install
moon install mizchi/actrun/cmd/actrun

# Build from source
git clone https://github.com/mizchi/actrun.git && cd actrun
moon build src/cmd/actrun --target native

Quick Start

# Run a workflow locally
actrun workflow run .github/workflows/ci.yml

# Show execution plan without running
actrun workflow run .github/workflows/ci.yml --dry-run

# Skip actions not needed locally (e.g. setup tools already installed)
actrun workflow run .github/workflows/ci.yml \
  --skip-action actions/checkout \
  --skip-action extractions/setup-just

# Run in isolated worktree
actrun workflow run .github/workflows/ci.yml \
  --workspace-mode worktree

# Generate config file
actrun init

# View results
actrun run view run-1
actrun run logs run-1 --task build/test

Configuration

actrun init generates an actrun.toml in the current directory:

# Workspace mode: local, worktree, tmp, docker
workspace_mode = "local"

# Skip actions not needed locally
local_skip_actions = ["actions/checkout"]

# Trust all third-party actions without prompt
trust_actions = true

# Nix integration: "auto" (force), "off" (disable), or empty (auto-detect)
nix_mode = ""

# Additional nix packages
nix_packages = ["python312", "jq"]

# Container runtime: docker, podman, container, lima, nerdctl
container_runtime = "docker"

# Include uncommitted changes in worktree/tmp workspace
# include_dirty = true

# Default local GitHub context when `--event` is omitted
# [local_context]
# repository = "owner/repo"
# ref_name = "main"
# before_rev = "HEAD^"
# after_rev = "HEAD"
# actor = "your-name"

# Override actions with local commands
# [override."actions/setup-node"]
# run = "echo 'using local node' && node --version"

# Affected file patterns per workflow
# [affected."ci.yml"]
# patterns = ["src/**", "package.json"]

When --event is omitted, actrun auto-detects github.repository, github.ref_name, github.sha, and github.actor from the local git repository when possible. Use [local_context] only when you need to pin or override those values. See Local GitHub Context for precedence and examples.

CLI flags always override actrun.toml settings. See Cheatsheet for quick reference and Advanced Workflow for details.

CLI Reference

Workflow Commands

actrun workflow list                 # List workflows in .github/workflows/
actrun workflow run <workflow.yml>    # Run a workflow locally

Run Commands

actrun run list                      # List past runs
actrun run view <run-id>             # View run summary
actrun run view <run-id> --json      # View run as JSON
actrun run watch <run-id>            # Watch until completion
actrun run logs <run-id>             # View all logs
actrun run logs <run-id> --task <id> # View specific task log
actrun run download <run-id>         # Download all artifacts

Analysis Commands

# Lint: type check expressions and detect dead code
actrun lint                          # Lint all .github/workflows/*.yml
actrun lint .github/workflows/ci.yml # Lint a specific file
actrun lint --ignore W001            # Suppress a rule (repeatable)

# Visualize: render workflow job dependency graph
actrun viz .github/workflows/ci.yml              # ASCII art (terminal)
actrun viz .github/workflows/ci.yml --mermaid    # Mermaid text (for Markdown)
actrun viz .github/workflows/ci.yml --detail     # Mermaid with step subgraphs
actrun viz .github/workflows/ci.yml --svg        # SVG image
actrun viz .github/workflows/ci.yml --svg --theme github-light

Lint Diagnostics

| Rule | Severity | Description | |------|----------|-------------| | undefined-context | error | Undefined context (e.g. foobar.x) | | wrong-arity | error | Wrong function arity (e.g. contains('one')) | | unknown-function | error | Unknown function (e.g. myFunc()) | | unknown-property | warning | Unknown property (e.g. github.nonexistent) | | type-mismatch | warning | Comparing incompatible types | | unreachable-step | warning | Unreachable step (if: false) | | future-step-ref | error | Reference to future step | | undefined-step-ref | error | Reference to undefined step | | undefined-needs | error | Undefined needs job reference | | circular-needs | error | Circular needs dependency | | unused-outputs | warning | Unused job outputs | | duplicate-step-id | error | Duplicate step IDs in same job | | missing-runs-on | error | Missing runs-on | | empty-job | error | Empty job (no steps) | | uses-and-run | error | Step has both uses and run | | empty-matrix | warning | Matrix with empty rows | | invalid-uses | error | Invalid uses syntax | | invalid-glob | warning | Invalid glob pattern in trigger filter | | redundant-condition | warning | Always-true/false condition | | script-injection | warning | Script injection risk (untrusted input in run:) | | permissive-permissions | warning | Overly permissive permissions | | deprecated-command | warning | Deprecated workflow command (::set-output etc.) | | missing-prt-permissions | warning | pull_request_target without explicit permissions | | if-always | warning | Bare always() — prefer success() \|\| failure() | | dangerous-checkout-in-prt | error | Checkout PR head in pull_request_target | | secrets-to-third-party | warning | Secrets passed via env to third-party action | | missing-timeout | warning | No timeout-minutes (opt-in: --strict) | | mutable-action-ref | warning | Tag ref instead of SHA pin (opt-in: --online) | | action-not-found | error | Action ref not found on GitHub (opt-in: --online) |

Configure lint behavior in actrun.toml:

[lint]
preset = "default"  # default, strict, oss
ignore_rules = ["unknown-property", "unused-outputs"]

| Preset | Includes | |--------|----------| | default | All rules except missing-timeout and online checks | | strict | default + missing-timeout | | oss | strict + mutable-action-ref / action-not-found (network) |

Visualization Example

$ actrun viz .github/workflows/release.yml

┌───────┐    ┌────────┐
│ build │    │ docker │
└───────┘    └────────┘
    └┐
     │
┌─────────┐
│ release │
└─────────┘

Artifact & Cache Commands

actrun artifact list <run-id>                          # List artifacts
actrun artifact download <run-id> --name <name>        # Download artifact
actrun cache list                                      # List cache entries
actrun cache prune --key <key>                         # Delete cache entry

Workflow Run Flags

| Flag | Description | |------|-------------| | --dry-run | Show execution plan without running | | --skip-action <pattern> | Skip actions matching pattern (repeatable) | | --workspace-mode <mode> | worktree (default), local, tmp, docker | | --repo <path> | Run from a git repository | | --event <path> | Push event JSON file | | --repository <owner/repo> | GitHub repository name | | --ref <ref> | Git ref name | | --run-root <path> | Run record storage root | | --nix | Force nix wrapping for run steps | | --no-nix | Disable nix wrapping even if flake.nix/shell.nix exists | | --nix-packages <pkgs> | Ad-hoc nix packages (space-separated) | | --container-runtime <name> | Container runtime: docker, podman, container, lima, nerdctl | | --affected [base] | Only run if files matching patterns changed (see below) | | --retry | Re-run only failed jobs from the latest run | | --include-dirty | Include uncommitted changes in worktree/tmp workspace | | --json | JSON output for read commands and --dry-run |

Affected Runs

Skip workflows when no relevant files have changed. Patterns are resolved in order:

  1. actrun.toml [affected."<workflow>"] patterns
  2. on:push:paths from the workflow file (automatic fallback)
# Compare against last successful run (default)
actrun ci.yml --affected

# Compare against a specific rev
actrun ci.yml --affected HEAD~3
actrun ci.yml --affected abc1234

# Preview what would happen (shows plan even if skipped)
actrun ci.yml --affected HEAD~1 --dry-run

Configure patterns in actrun.toml:

[affected."ci.yml"]
patterns = ["src/**", "package.json"]

[affected.".github/workflows/lint.yml"]
patterns = ["src/**", "*.config.*"]

If actrun.toml has no patterns, on:push:paths from the workflow is used automatically:

on:
  push:
    paths: ["src/**", "*.toml"]  # actrun --affected uses these

Workspace Modes

| Mode | Description | |------|-------------| | local | Run in-place in the current directory | | worktree | Create an isolated git worktree for execution (default) | | tmp | Clone to a temp directory via git clone | | docker | Run in a Docker container |

Container Runtime

actrun supports multiple container runtimes for job container:, services:, and docker:// actions.

| Runtime | Binary | Notes | |---------|--------|-------| | docker | docker | Default | | podman | podman | Docker-compatible CLI | | container | container | Apple container runtime (macOS) | | nerdctl | nerdctl | containerd CLI | | lima | lima nerdctl | Lima VM with nerdctl (wrapper script auto-generated) |

# CLI flag
actrun workflow run ci.yml --container-runtime podman

# actrun.toml
container_runtime = "podman"

# Environment variable (also works)
ACTRUN_CONTAINER_RUNTIME=podman actrun workflow run ci.yml

Supported GitHub Actions

Builtin Actions (deterministic emulation)

| Action | Supported Inputs | |--------|-----------------| | actions/checkout@* | path, ref, fetch-depth, clean, sparse-checkout, submodules, lfs, fetch-tags, persist-credentials, set-safe-directory, show-progress | | actions/upload-artifact@* | name, path, if-no-files-found, overwrite, include-hidden-files | | actions/download-artifact@* | name, path, pattern, merge-multiple | | actions/cache@* | key, path, restore-keys, lookup-only, fail-on-cache-miss | | actions/cache/save@* | key, path | | actions/cache/restore@* | key, path, restore-keys, lookup-only, fail-on-cache-miss | | actions/setup-node@* | node-version, node-version-file, cache, registry-url, always-auth, scope |

Remote Actions (fetch + execute)

  • GitHub repo node actions with pre/main/post lifecycle
  • GitHub repo docker actions with pre-entrypoint/entrypoint/post-entrypoint lifecycle
  • Composite actions (local and remote)
  • docker://image direct execution
  • wasm://name@version module execution

Local-Only Execution Flag

actrun sets ACTRUN_LOCAL=true in the execution environment. Use this in if: conditions to skip steps locally or run steps only locally:

steps:
  # Skipped when running locally (runs on GitHub Actions)
  - uses: actions/checkout@v5
    if: ${{ !env.ACTRUN_LOCAL }}

  # Runs only locally (skipped on GitHub Actions)
  - run: echo "local debug info"
    if: ${{ env.ACTRUN_LOCAL }}

On GitHub Actions, ACTRUN_LOCAL is not set, so !env.ACTRUN_LOCAL evaluates to true and all steps run normally.

Action Overrides

Replace specific uses: action steps with custom run: commands via actrun.toml. This is useful when you have tools installed locally and want to skip the action's setup logic.

[override."actions/setup-node"]
run = "echo 'using local node' && node --version"

When a workflow step matches uses: actions/setup-node@*, actrun replaces it with the specified run: command before execution.

Combine with local_skip_actions for full control:

local_skip_actions = ["actions/checkout"]

[override."actions/setup-node"]
run = "echo 'using local node'"

[override."actions/setup-python"]
run = "python3 --version"

Secrets & Variables

# Provide secrets via environment variables
ACTRUN_SECRET_MY_TOKEN=xxx actrun workflow run ci.yml

# Provide variables
ACTRUN_VAR_MY_VAR=value actrun workflow run ci.yml

Secrets are automatically masked in stdout, stderr, logs, and run store. The ::add-mask:: workflow command is also supported.

Environment Variables

| Variable | Description | |----------|-------------| | ACTRUN_SECRET_<NAME> | ${{ secrets.<name> }} | | ACTRUN_VAR_<NAME> | ${{ vars.<name> }} | | ACTRUN_NODE_BIN | Node.js binary path | | ACTRUN_DOCKER_BIN | Docker binary path | | ACTRUN_WASM_BIN | Wasm runtime binary (default: wasmtime) | | ACTRUN_GIT_BIN | Git binary path | | ACTRUN_GITHUB_BASE_URL | GitHub API base URL | | ACTRUN_ARTIFACT_ROOT | Artifact storage root | | ACTRUN_CACHE_ROOT | Cache storage root | | ACTRUN_GITHUB_ACTION_CACHE_ROOT | Remote action cache root | | ACTRUN_ACTION_REGISTRY_ROOT | Custom registry root | | ACTRUN_WASM_ACTION_ROOT | Wasm action module root | | ACTRUN_NIX | Set to false to disable nix wrapping |

Nix Integration

actrun automatically detects flake.nix or shell.nix in the workspace root and wraps run: steps in the corresponding nix environment. This lets workflows written for ubuntu-latest run locally with nix-managed toolchains.

Auto-detection

| Condition | Wrapping | |-----------|----------| | flake.nix exists | nix develop --command <shell> <script> | | shell.nix exists | nix-shell --run '<shell> <script>' | | Neither exists | No wrapping (host environment) |

Detection requires nix to be installed. If nix is not found, wrapping is silently skipped.

Examples

# Auto-detect flake.nix / shell.nix
actrun workflow run .github/workflows/ci.yml

# Disable nix wrapping
actrun workflow run .github/workflows/ci.yml --no-nix

# Ad-hoc packages without flake.nix
actrun workflow run .github/workflows/ci.yml --nix-packages "python312 jq"

# Disable via environment variable
ACTRUN_NIX=false actrun workflow run .github/workflows/ci.yml

Typical flake.nix for Rust

{
  inputs.nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.11";
  outputs = { self, nixpkgs }:
    let
      systems = [ "x86_64-linux" "aarch64-linux" "x86_64-darwin" "aarch64-darwin" ];
      forAllSystems = nixpkgs.lib.genAttrs systems;
    in {
      devShells = forAllSystems (system:
        let pkgs = nixpkgs.legacyPackages.${system};
        in { default = pkgs.mkShell { packages = [ pkgs.rustc pkgs.cargo ]; }; });
    };
}

Typical flake.nix for Python + uv

{
  inputs.nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.11";
  outputs = { self, nixpkgs }:
    let
      systems = [ "x86_64-linux" "aarch64-linux" "x86_64-darwin" "aarch64-darwin" ];
      forAllSystems = nixpkgs.lib.genAttrs systems;
    in {
      devShells = forAllSystems (system:
        let pkgs = nixpkgs.legacyPackages.${system};
        in { default = pkgs.mkShell { packages = [ pkgs.python312 pkgs.uv ]; }; });
    };
}

Notes

  • Only run: steps are wrapped. uses: action steps are not affected.
  • Job container: steps skip nix wrapping (container has its own environment).
  • nix develop / nix-shell is invoked per step, so the nix environment is consistent across steps.

Workflow Features

  • Push trigger filter (branches, paths)
  • strategy.matrix (axes, include, exclude, fail-fast, max-parallel)
  • Job/step if conditions (success(), always(), failure(), cancelled())
  • needs dependencies with output/result propagation
  • Reusable workflows (workflow_call) with inputs, outputs, secrets, secrets: inherit, nested expansion
  • Job container and services with Docker networking
  • Expression functions: contains, startsWith, endsWith, fromJSON, toJSON, hashFiles
  • File commands: GITHUB_ENV, GITHUB_PATH, GITHUB_OUTPUT, GITHUB_STEP_SUMMARY
  • Shell support: bash, sh, pwsh, custom templates ({0})
  • step.continue-on-error, steps.*.outcome / steps.*.conclusion

Development

just              # check + test
just fmt          # format code
just check        # type check
just test         # run tests
just e2e          # run E2E scenarios
just release-check  # fmt + info + check + test + e2e

Live Compatibility Testing

# One-shot: dispatch, wait, download, compare
just gha-compat-live compat-checkout-artifact.yml

# Step by step
just gha-compat-dispatch compat-checkout-artifact.yml
just gha-compat-download <run-id>
just gha-compat-compare compat-checkout-artifact.yml _build/gha-compat/<run-id>

Architecture

| File | Purpose | |------|---------| | src/lib.mbt | Contract types | | src/parser.mbt | Workflow YAML parser | | src/trigger.mbt | Push trigger matcher | | src/lowering.mbt | Bitflow IR lowering, action/reusable workflow expansion | | src/executor.mbt | Native host executor | | src/runtime.mbt | Git workspace materialization | | src/lint/ | Expression parser, type checker, dead code detection, workflow visualization | | src/cmd/actrun/main.mbt | CLI entry point | | testdata/ | Compatibility fixtures |

Prior Art

  • actionlint — Static checker for GitHub Actions workflow files. actrun lint is inspired by its rule design and type system.

License

Apache-2.0