mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
251 lines
8.7 KiB
Ruby
251 lines
8.7 KiB
Ruby
# typed: strict
|
|
# frozen_string_literal: true
|
|
|
|
require "abstract_command"
|
|
require "erb"
|
|
require "fileutils"
|
|
require "tap"
|
|
require "utils/uid"
|
|
|
|
module Homebrew
|
|
module DevCmd
|
|
class TapNew < AbstractCommand
|
|
include FileUtils
|
|
|
|
cmd_args do
|
|
usage_banner "`tap-new` [<options>] <user>`/`<repo>"
|
|
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
|
|
# Duplicate assignment to silence `assigned but unused variable` warning
|
|
root_url = root_url = GitHubPackages.root_url(tap.user, "homebrew-#{tap.repository}") if args.github_packages?
|
|
|
|
(tap.path/"Formula").mkpath
|
|
|
|
readme = <<~MARKDOWN
|
|
# #{titleized_user} #{titleized_repository}
|
|
|
|
## How do I install these formulae?
|
|
|
|
`brew install #{tap}/<formula>`
|
|
|
|
Or `brew tap #{tap}` and then `brew install <formula>`.
|
|
|
|
Or, in a `brew bundle` `Brewfile`:
|
|
|
|
```ruby
|
|
tap "#{tap}"
|
|
brew "<formula>"
|
|
```
|
|
|
|
## Documentation
|
|
|
|
`brew help`, `man brew` or check [Homebrew's documentation](https://docs.brew.sh).
|
|
MARKDOWN
|
|
write_path(tap, "README.md", readme)
|
|
|
|
tests_yml = <<~ERB
|
|
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 }}
|
|
permissions:
|
|
actions: read
|
|
checks: read
|
|
contents: read
|
|
<% if args.github_packages? -%>
|
|
packages: read
|
|
<% end -%>
|
|
pull-requests: read
|
|
steps:
|
|
- name: Set up Homebrew
|
|
id: set-up-homebrew
|
|
uses: Homebrew/actions/setup-homebrew@main
|
|
with:
|
|
token: ${{ github.token }}
|
|
|
|
- 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
|
|
<% if args.github_packages? -%>
|
|
- name: Base64-encode GITHUB_TOKEN for HOMEBREW_DOCKER_REGISTRY_TOKEN
|
|
id: base64-encode
|
|
if: github.event_name == 'pull_request'
|
|
env:
|
|
TOKEN: ${{ github.token }}
|
|
run: |
|
|
base64_token=$(echo -n "${TOKEN}" | base64 | tr -d "\\n")
|
|
echo "::add-mask::${base64_token}"
|
|
echo "token=${base64_token}" >> "${GITHUB_OUTPUT}"
|
|
<% end -%>
|
|
- run: brew test-bot --only-formulae#{" --root-url=#{root_url}" if root_url}
|
|
if: github.event_name == 'pull_request'
|
|
<% if args.github_packages? -%>
|
|
env:
|
|
HOMEBREW_DOCKER_REGISTRY_TOKEN: ${{ steps.base64-encode.outputs.token }}
|
|
<% end -%>
|
|
|
|
- name: Upload bottles as artifact
|
|
if: always() && github.event_name == 'pull_request'
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: bottles_${{ matrix.os }}
|
|
path: '*.bottle.*'
|
|
ERB
|
|
|
|
publish_yml = <<~ERB
|
|
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:
|
|
actions: read
|
|
checks: read
|
|
contents: write
|
|
issues: read
|
|
<% if args.github_packages? -%>
|
|
packages: write
|
|
<% end -%>
|
|
pull-requests: write
|
|
steps:
|
|
- name: Set up Homebrew
|
|
uses: Homebrew/actions/setup-homebrew@main
|
|
with:
|
|
token: ${{ github.token }}
|
|
|
|
- name: Set up git
|
|
uses: Homebrew/actions/git-user-config@main
|
|
|
|
- name: Pull bottles
|
|
env:
|
|
HOMEBREW_GITHUB_API_TOKEN: ${{ github.token }}
|
|
<% if args.github_packages? -%>
|
|
HOMEBREW_GITHUB_PACKAGES_TOKEN: ${{ github.token }}
|
|
HOMEBREW_GITHUB_PACKAGES_USER: ${{ github.repository_owner }}
|
|
<% end -%>
|
|
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@main
|
|
with:
|
|
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"
|
|
ERB
|
|
|
|
(tap.path/".github/workflows").mkpath
|
|
write_path(tap, ".github/workflows/tests.yml", ERB.new(tests_yml, trim_mode: "-").result(binding))
|
|
write_path(tap, ".github/workflows/publish.yml", ERB.new(publish_yml, trim_mode: "-").result(binding))
|
|
|
|
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
|