# typed: strict # frozen_string_literal: true require "abstract_command" require "fileutils" require "tap" require "utils/uid" module Homebrew module DevCmd class TapNew < AbstractCommand include FileUtils cmd_args do usage_banner "`tap-new` [] `/`" description <<~EOS Generate the template files for a new tap. EOS switch "--no-git", description: "Don't initialize a Git repository for the tap." flag "--pull-label=", description: "Label name for pull requests ready to be pulled (default: `pr-pull`)." flag "--branch=", description: "Initialize Git repository and setup GitHub Actions workflows with the " \ "specified branch name (default: `main`)." switch "--github-packages", description: "Upload bottles to GitHub Packages." named_args :tap, number: 1 end sig { override.void } def run label = args.pull_label || "pr-pull" branch = args.branch || "main" tap = args.named.to_taps.fetch(0) odie "Invalid tap name '#{tap}'" unless tap.path.to_s.match?(HOMEBREW_TAP_PATH_REGEX) odie "Tap is already installed!" if tap.installed? titleized_user = tap.user.dup titleized_repository = tap.repository.dup titleized_user[0] = titleized_user[0].upcase titleized_repository[0] = titleized_repository[0].upcase root_url = GitHubPackages.root_url(tap.user, "homebrew-#{tap.repository}") if args.github_packages? (tap.path/"Formula").mkpath # FIXME: https://github.com/errata-ai/vale/issues/818 # readme = <<~MARKDOWN # #{titleized_user} #{titleized_repository} ## How do I install these formulae? `brew install #{tap}/` Or `brew tap #{tap}` and then `brew install `. Or, in a [`brew bundle`](https://github.com/Homebrew/homebrew-bundle) `Brewfile`: ```ruby tap "#{tap}" brew "" ``` ## Documentation `brew help`, `man brew` or check [Homebrew's documentation](https://docs.brew.sh). MARKDOWN # write_path(tap, "README.md", readme) actions_main = <<~YAML name: brew test-bot on: push: branches: - #{branch} pull_request: jobs: test-bot: strategy: matrix: os: [ubuntu-22.04, macos-13, macos-15] runs-on: ${{ matrix.os }} steps: - name: Set up Homebrew id: set-up-homebrew uses: Homebrew/actions/setup-homebrew@master - name: Cache Homebrew Bundler RubyGems uses: actions/cache@v4 with: path: ${{ steps.set-up-homebrew.outputs.gems-path }} key: ${{ matrix.os }}-rubygems-${{ steps.set-up-homebrew.outputs.gems-hash }} restore-keys: ${{ matrix.os }}-rubygems- - run: brew test-bot --only-cleanup-before - run: brew test-bot --only-setup - run: brew test-bot --only-tap-syntax - run: brew test-bot --only-formulae#{" --root-url='#{root_url}'" if root_url} if: github.event_name == 'pull_request' - name: Upload bottles as artifact if: always() && github.event_name == 'pull_request' uses: actions/upload-artifact@v4 with: name: bottles_${{ matrix.os }} path: '*.bottle.*' YAML pr_pull_permissions = { "contents" => "write", "pull-requests" => "write", } pr_pull_env = { "HOMEBREW_GITHUB_API_TOKEN" => "${{ github.token }}", } if args.github_packages? pr_pull_permissions["packages"] = "write" pr_pull_env["HOMEBREW_GITHUB_PACKAGES_TOKEN"] = "${{ github.token }}" pr_pull_env["HOMEBREW_GITHUB_PACKAGES_USER"] = "${{ github.repository_owner }}" end actions_publish = <<~YAML name: brew pr-pull on: pull_request_target: types: - labeled jobs: pr-pull: if: contains(github.event.pull_request.labels.*.name, '#{label}') runs-on: ubuntu-22.04 permissions: #{pr_pull_permissions.sort.map { |k, v| " #{k}: #{v}" }.join("\n")} steps: - name: Set up Homebrew uses: Homebrew/actions/setup-homebrew@master - name: Set up git uses: Homebrew/actions/git-user-config@master - name: Pull bottles env: #{pr_pull_env.sort.map { |k, v| " #{k}: #{v}" }.join("\n")} PULL_REQUEST: ${{ github.event.pull_request.number }} run: brew pr-pull --debug --tap="$GITHUB_REPOSITORY" "$PULL_REQUEST" - name: Push commits uses: Homebrew/actions/git-try-push@master with: token: ${{ github.token }} branch: #{branch} - name: Delete branch if: github.event.pull_request.head.repo.fork == false env: BRANCH: ${{ github.event.pull_request.head.ref }} run: git push --delete origin "$BRANCH" YAML (tap.path/".github/workflows").mkpath write_path(tap, ".github/workflows/tests.yml", actions_main) write_path(tap, ".github/workflows/publish.yml", actions_publish) unless args.no_git? cd tap.path do |path| Utils::Git.set_name_email! Utils::Git.setup_gpg! # Would be nice to use --initial-branch here but it's not available in # older versions of Git that we support. safe_system "git", "-c", "init.defaultBranch=#{branch}", "init" args = [] git_owner = File.stat(File.join(path, ".git")).uid if git_owner != Process.uid && git_owner == Process.euid # Under Homebrew user model, EUID is permitted to execute commands under the UID. # Root users are never allowed (see brew.sh). args << "-c" << "safe.directory=#{path}" end # Use the configuration of the original user, which will have author information and signing keys. Utils::UID.drop_euid do env = { HOME: Utils::UID.uid_home }.compact env[:TMPDIR] = nil if (tmpdir = ENV.fetch("TMPDIR", nil)) && !File.writable?(tmpdir) with_env(env) do safe_system "git", *args, "add", "--all" safe_system "git", *args, "commit", "-m", "Create #{tap} tap" safe_system "git", *args, "branch", "-m", branch end end end end ohai "Created #{tap}" puts <<~EOS #{tap.path} When a pull request making changes to a formula (or formulae) becomes green (all checks passed), then you can publish the built bottles. To do so, label your PR as `#{label}` and the workflow will be triggered. EOS end private sig { params(tap: Tap, filename: T.any(String, Pathname), content: String).void } def write_path(tap, filename, content) path = tap.path/filename tap.path.mkpath odie "#{path} already exists" if path.exist? path.write content end end end end