243 lines
7.8 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
require "formula"
require "erb"
require "ostruct"
2019-04-17 18:25:08 +09:00
require "cli/parser"
module Homebrew
2016-09-26 01:44:51 +02:00
module_function
2019-04-19 21:46:20 +09:00
SOURCE_PATH = (HOMEBREW_LIBRARY_PATH/"manpages").freeze
TARGET_MAN_PATH = (HOMEBREW_REPOSITORY/"manpages").freeze
TARGET_DOC_PATH = (HOMEBREW_REPOSITORY/"docs").freeze
def man_args
Homebrew::CLI::Parser.new do
usage_banner <<~EOS
`man` [<options>]
2018-09-28 21:39:52 +05:30
Generate Homebrew's manpages.
EOS
switch "--fail-if-changed",
2019-04-30 08:44:35 +01:00
description: "Return a failing status code if changes are detected in the manpage outputs. This "\
"can be used to notify CI when the manpages are out of date. Additionally, "\
2019-04-30 08:44:35 +01:00
"the date used in new manpages will match those in the existing manpages (to allow "\
"comparison without factoring in the date)."
2018-09-28 21:39:52 +05:30
switch "--link",
2019-04-30 08:44:35 +01:00
description: "This is now done automatically by `brew update`."
max_named 0
end
end
def man
man_args.parse
2018-03-24 21:26:16 +05:30
odie "`brew man --link` is now done automatically by `brew update`." if args.link?
regenerate_man_pages
if system "git", "-C", HOMEBREW_REPOSITORY, "diff", "--quiet", "docs/Manpage.md", "manpages"
puts "No changes to manpage output detected."
elsif args.fail_if_changed?
Homebrew.failed = true
end
end
def regenerate_man_pages
Homebrew.install_bundler_gems!
markup = build_man_page
convert_man_page(markup, TARGET_DOC_PATH/"Manpage.md")
convert_man_page(markup, TARGET_MAN_PATH/"brew.1")
cask_markup = (SOURCE_PATH/"brew-cask.1.md").read
convert_man_page(cask_markup, TARGET_MAN_PATH/"brew-cask.1")
end
2016-09-05 21:46:40 +01:00
def build_man_page
template = (SOURCE_PATH/"brew.1.md.erb").read
variables = OpenStruct.new
variables[:commands] = generate_cmd_manpages(Commands.internal_commands_paths)
variables[:developer_commands] = generate_cmd_manpages(Commands.internal_developer_commands_paths)
variables[:official_external_commands] = generate_cmd_manpages(Commands.official_external_commands_paths)
2019-01-30 21:33:30 +00:00
variables[:global_options] = global_options_manpage
variables[:environment_variables] = env_vars_manpage
2018-09-28 21:39:52 +05:30
readme = HOMEBREW_REPOSITORY/"README.md"
variables[:lead] =
2019-02-15 10:54:30 +00:00
readme.read[/(Homebrew's \[Project Leader.*\.)/, 1]
.gsub(/\[([^\]]+)\]\([^)]+\)/, '\1')
variables[:plc] =
2019-02-15 10:54:30 +00:00
readme.read[/(Homebrew's \[Project Leadership Committee.*\.)/, 1]
.gsub(/\[([^\]]+)\]\([^)]+\)/, '\1')
variables[:tsc] =
2019-02-15 10:54:30 +00:00
readme.read[/(Homebrew's \[Technical Steering Committee.*\.)/, 1]
.gsub(/\[([^\]]+)\]\([^)]+\)/, '\1')
variables[:linux] =
readme.read[%r{(Homebrew/brew's Linux maintainers .*\.)}, 1]
.gsub(/\[([^\]]+)\]\([^)]+\)/, '\1')
variables[:maintainers] =
readme.read[/(Homebrew's other current maintainers .*\.)/, 1]
.gsub(/\[([^\]]+)\]\([^)]+\)/, '\1')
variables[:alumni] =
readme.read[/(Former maintainers .*\.)/, 1]
.gsub(/\[([^\]]+)\]\([^)]+\)/, '\1')
2016-06-14 21:01:50 +08:00
2019-10-13 10:09:38 +01:00
ERB.new(template, trim_mode: ">").result(variables.instance_eval { binding })
end
def sort_key_for_path(path)
# Options after regular commands (`~` comes after `z` in ASCII table).
path.basename.to_s.sub(/\.(rb|sh)$/, "").sub(/^--/, "~~")
end
def convert_man_page(markup, target)
manual = target.basename(".1")
organisation = "Homebrew"
# Set the manpage date to the existing one if we're checking for changes.
# This avoids the only change being e.g. a new date.
date = if args.fail_if_changed? &&
target.extname == ".1" && target.exist?
/"(\d{1,2})" "([A-Z][a-z]+) (\d{4})" "#{organisation}" "#{manual}"/ =~ target.read
Date.parse("#{Regexp.last_match(1)} #{Regexp.last_match(2)} #{Regexp.last_match(3)}")
else
Date.today
end
date = date.strftime("%Y-%m-%d")
shared_args = %W[
--pipe
--organization=#{organisation}
--manual=#{target.basename(".1")}
--date=#{date}
]
format_flag, format_desc = target_path_to_format(target)
puts "Writing #{format_desc} to #{target}"
Utils.popen(["ronn", format_flag] + shared_args, "rb+") do |ronn|
ronn.write markup
ronn.close_write
ronn_output = ronn.read
2019-10-08 10:49:02 +01:00
odie "Got no output from ronn!" if ronn_output.blank?
if format_flag == "--markdown"
ronn_output = ronn_output.gsub(%r{<var>(.*?)</var>}, "*`\\1`*")
2019-01-30 21:33:30 +00:00
.gsub(/\n\n\n+/, "\n\n")
elsif format_flag == "--roff"
2018-10-03 10:19:46 +05:30
ronn_output = ronn_output.gsub(%r{<code>(.*?)</code>}, "\\fB\\1\\fR")
.gsub(%r{<var>(.*?)</var>}, "\\fI\\1\\fR")
.gsub(/(^\[?\\fB.+): /, "\\1\n ")
2018-10-02 19:54:22 +05:30
end
target.atomic_write ronn_output
end
end
def target_path_to_format(target)
case target.basename
when /\.md$/ then ["--markdown", "markdown"]
when /\.\d$/ then ["--roff", "man page"]
else
odie "Failed to infer output format from '#{target.basename}'."
end
end
2018-09-08 22:21:04 +05:30
def generate_cmd_manpages(cmd_paths)
2018-09-28 21:39:52 +05:30
man_page_lines = []
man_args = Homebrew.args
2019-01-30 21:33:30 +00:00
# preserve existing manpage order
cmd_paths.sort_by(&method(:sort_key_for_path))
.each do |cmd_path|
cmd_man_page_lines = if cmd_parser = CLI::Parser.from_cmd_path(cmd_path)
2019-01-30 21:33:30 +00:00
next if cmd_parser.hide_from_man_page
2019-01-30 21:33:30 +00:00
cmd_parser_manpage_lines(cmd_parser).join
else
cmd_comment_manpage_lines(cmd_path)
2018-09-28 21:39:52 +05:30
end
2019-01-30 21:33:30 +00:00
man_page_lines << cmd_man_page_lines
2018-09-28 21:39:52 +05:30
end
Homebrew.args = man_args
2019-01-30 21:33:30 +00:00
man_page_lines.compact.join("\n")
2018-09-28 21:39:52 +05:30
end
2019-01-30 21:33:30 +00:00
def cmd_parser_manpage_lines(cmd_parser)
2018-10-02 19:54:22 +05:30
lines = [format_usage_banner(cmd_parser.usage_banner_text)]
2018-09-08 22:21:04 +05:30
lines += cmd_parser.processed_options.map do |short, long, _, desc|
next if !long.nil? && cmd_parser.global_option?(cmd_parser.option_to_name(long), desc)
generate_option_doc(short, long, desc)
2019-01-30 21:33:30 +00:00
end.reject(&:blank?)
lines
end
def cmd_comment_manpage_lines(cmd_path)
comment_lines = cmd_path.read.lines.grep(/^#:/)
return if comment_lines.empty?
return if comment_lines.first.include?("@hide_from_man_page")
2019-01-30 21:33:30 +00:00
lines = [format_usage_banner(comment_lines.first).chomp]
comment_lines.slice(1..-1)
.each do |line|
line = line.slice(4..-2)
unless line
lines.last << "\n"
next
end
# Omit the common global_options documented separately in the man page.
next if line.match?(/--(debug|force|help|quiet|verbose) /)
# Format one option or a comma-separated pair of short and long options.
lines << line.gsub(/^ +(-+[a-z-]+), (-+[a-z-]+) +/, "* `\\1`, `\\2`:\n ")
.gsub(/^ +(-+[a-z-]+) +/, "* `\\1`:\n ")
end
lines.last << "\n"
lines
end
2019-01-30 21:33:30 +00:00
def global_options_manpage
lines = ["These options are applicable across multiple subcommands.\n"]
lines += Homebrew::CLI::Parser.global_options.values.map do |names, _, desc|
short, long = names
2018-09-08 22:21:04 +05:30
generate_option_doc(short, long, desc)
end
2019-01-30 21:33:30 +00:00
lines.join("\n")
2018-09-08 22:21:04 +05:30
end
def env_vars_manpage
lines = Homebrew::EnvConfig::ENVS.flat_map do |env, hash|
entry = " * `#{env}`:\n #{hash[:description]}\n"
default = hash[:default_text]
default ||= "`#{hash[:default]}`." if hash[:default]
entry += "\n\n *Default:* #{default}\n" if default
entry
end
lines.join("\n")
end
2018-09-08 22:21:04 +05:30
def generate_option_doc(short, long, desc)
comma = (short && long) ? ", " : ""
<<~EOS
* #{format_short_opt(short)}#{comma}#{format_long_opt(long)}:
#{desc}
EOS
2018-09-08 22:21:04 +05:30
end
def format_short_opt(opt)
"`#{opt}`" unless opt.nil?
2018-09-08 22:21:04 +05:30
end
def format_long_opt(opt)
"`#{opt}`" unless opt.nil?
2018-09-08 22:21:04 +05:30
end
def format_usage_banner(usage_banner)
2019-01-30 21:33:30 +00:00
usage_banner&.sub(/^(#: *\* )?/, "### ")
end
end