| #!/bin/bash |
| # Copyright 2019 The Fuchsia Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| #### CATEGORY=Documentation |
| ### create markdown docs for fx subcommands |
| |
| ## usage: fx helpdoc [OPTIONS] OUTPUT_DIR |
| ## |
| ## Create markdown documentation for fx and its subcommands, based on |
| ## the same metadata that is used to create the output of `fx help` |
| ## |
| ## --vendor create docs for subcommands in //vendor/*/scripts/devshell. |
| ## If specified, only vendor docs are documented. There's |
| ## no way to generate docs for vendor and non-vendor at the |
| ## same time. |
| ## --toc-prefix URL_PREFIX use URL_PREFIX instead of "/reference/tools/fx" |
| ## as an URL prefix in generated _toc.yaml files. |
| ## --no-deprecated do not create docs for deprecated subcommands |
| ## --no-contrib do not create docs for contrib scripts in //tools/devshell/contrib/* |
| |
| source "$(cd "$(dirname "${BASH_SOURCE[0]}")" >/dev/null 2>&1 && pwd)"/lib/vars.sh || exit $? |
| fx-config-read |
| |
| categories=() |
| |
| function main { |
| show_vendor=0 |
| show_contrib=1 |
| show_deprecated=1 |
| toc_prefix="/reference/tools/fx" |
| while [[ $# -ne 0 ]]; do |
| case $1 in |
| --vendor) |
| show_vendor=1 |
| ;; |
| --toc-prefix) |
| if [[ $# -lt 2 ]]; then |
| fx-error "Invalid syntax" |
| fx-command-help |
| exit 1 |
| fi |
| shift |
| toc_prefix="$1" |
| ;; |
| --no-deprecated) |
| show_deprecated=0 |
| ;; |
| --no-contrib) |
| show_contrib=0 |
| ;; |
| -*) |
| fx-error "Unknown argument $1" |
| fx-command-help |
| exit 1 |
| ;; |
| *) |
| break |
| ;; |
| esac |
| shift |
| done |
| |
| if [[ $# -ne 1 ]]; then |
| fx-error "No output path specified" |
| fx-command-help |
| return 1 |
| fi |
| |
| base_out="$1" |
| mkdir -p "${base_out}/cmd" |
| |
| # initializes the main index.md |
| { |
| echo "# fx - Fuchsia development commands" |
| echo |
| echo "\`fx\` is the entry-point for a set of subcommands that make many tasks related to Fuchsia development easier." |
| echo |
| echo "It contains a large number of subcommands. Run \`fx help\` to see all the available subcommands." |
| echo "If you use bash or zsh as a shell, source \`scripts/fx-env.sh\` " |
| echo "to get some auto-completion." |
| echo |
| echo "Also see the [global options and flags](fx.md) that affect all subcommands." |
| echo |
| echo "# fx subcommands" |
| |
| } > "${base_out}/index.md" |
| |
| { |
| echo "# fx command" |
| echo |
| echo "For help with specific subcommands, see [Overview page](index.md)." |
| echo |
| echo '```none' |
| print-redacted-fx-help |
| echo '```' |
| echo |
| echo "[fx source code](https://fuchsia.googlesource.com/fuchsia/+/HEAD/scripts/fx)" |
| } > "${base_out}/fx.md" |
| |
| local dirs |
| if [[ "${show_vendor}" -eq 1 ]]; then |
| for d in "${FUCHSIA_DIR}"/vendor/*/scripts/devshell; do |
| if [[ -d "${d}" ]]; then |
| vendor="${d%/scripts/devshell}" |
| vendor="${vendor##*/}" |
| { |
| echo |
| echo "## ${vendor} subcommands" |
| echo |
| _md_main_table_header |
| } >> "${base_out}/index.md" |
| handle-directory "${d}" |
| fi |
| done |
| else |
| { |
| echo |
| echo "## Main subcommands" |
| echo |
| echo "Subcommands that are part of the [fx workflow](/docs/development/build/fx.md)." |
| echo |
| _md_main_table_header |
| } >> "${base_out}/index.md" |
| handle-directory "${FUCHSIA_DIR}/tools/devshell" |
| local d="${FUCHSIA_DIR}/tools/devshell/contrib" |
| if [[ "${show_contrib}" -eq 1 && -d "${d}" ]]; then |
| { |
| echo |
| echo "## Contrib subcommands" |
| echo |
| echo "Subcommands that have been contributed by project members that have other levels of support, ownership, or both." |
| echo "The [OWNERS file](https://fuchsia.googlesource.com/fuchsia/+/HEAD/tools/devshell/contrib/OWNERS) in the" |
| echo "contrib directory provides a pointer to the member that supports each script." |
| echo |
| _md_main_table_header |
| } >> "${base_out}/index.md" |
| handle-directory "${d}" |
| fi |
| fi |
| create-main-toc |
| } |
| |
| function _md_main_table_header { |
| echo "Command | Category | Description" |
| echo "------- | -------- | -----------" |
| } |
| |
| function print-redacted-fx-help { |
| local filter="$(mktemp)" |
| cat > "${filter}" <<EOF |
| NR==1 || NR==3, /^$/ { |
| print |
| } |
| |
| /^[^ ].*help flags.*:/,/^$/ { |
| print |
| } |
| /^Global .*:/,/^$/ { |
| print |
| } |
| EOF |
| fx help | awk -f "${filter}" |
| rm "${filter}" |
| } |
| |
| function handle-directory { |
| local d="$1" |
| local files=( "${d}"/* ) |
| local cmds=() |
| for f in "${files[@]}"; do |
| if [[ -f "${f}" && ( -x "${f}" || "${f: -3}" == ".fx" ) ]]; then |
| cmds+=( "${f}" ) |
| fi |
| done |
| |
| # sort cmds |
| IFS=$'\n' sorted=($(sort <<< "${cmds[*]}")) |
| unset IFS |
| |
| for c in "${sorted[@]}"; do |
| echo "Processing $(basename "${c}")" |
| handle-command "${c}" |
| done |
| } |
| |
| |
| function normalize { |
| local str="$1" |
| echo "${str}" | tr -s ', .' '_' | tr '[:upper:]' '[:lower:]' |
| } |
| |
| function get_binary_metadata { |
| local file="$1" |
| local key="$2" |
| awk -F ' *= *' -f - "${file}" <<EOF |
| /^#### +${key} */ { |
| print 1; |
| } |
| EOF |
| } |
| |
| function get_metadata { |
| local file="$1" |
| local key="$2" |
| awk -F ' *= *' -f - "${file}" <<EOF |
| /^#### +${key} */ { |
| print \$2; |
| } |
| EOF |
| } |
| |
| function get_metadata_camelcase { |
| local file="$1" |
| local key="$2" |
| awk -F ' *= *' -f - "${file}" <<EOF |
| /^#### +${key} */ { |
| print toupper(substr(\$2,1,1)) tolower(substr(\$2,2)); |
| } |
| EOF |
| } |
| |
| function get_summary { |
| local cmd_path="$1" |
| sed -n '1,/^### /s/^### //p' < "${cmd_path}" | sed 's/_/\\_/g; s/*/\\*/g; s/|/\\|/g' |
| } |
| |
| function create-main-toc { |
| # initializes the global toc |
| { |
| echo "toc:" |
| echo "- title: Overview" |
| echo " path: ${toc_prefix}/index.md" |
| echo "- title: fx command" |
| echo " path: ${toc_prefix}/fx.md" |
| } > "${base_out}/_toc.yaml" |
| |
| # sort categories |
| IFS=$'\n' sorted=($(sort -u <<< "${categories[*]}")) |
| unset IFS |
| |
| for category in "${sorted[@]}"; do |
| echo "Processing category ${category}" |
| local norm_cat="$(normalize "${category}")" |
| # Appends the category to the global _toc.yaml |
| { |
| echo "- title: \"${category} commands\"" |
| echo " path: ${toc_prefix}/${norm_cat}" |
| echo " section:" |
| echo " - include: ${toc_prefix}/${norm_cat}_toc.yaml" |
| } >> "${base_out}/_toc.yaml" |
| done |
| } |
| |
| function get-category-title { |
| local category="$1" |
| if [[ -n "${vendor}" ]]; then |
| echo "${category} fx subcommands for ${vendor}" |
| else |
| echo "${category} fx subcommands" |
| fi |
| } |
| |
| function maybe-add-category { |
| local category="$1" |
| local norm_cat="$(normalize "${category}")" |
| local cat_base="${base_out}/${norm_cat}" |
| local toc="${cat_base}_toc.yaml" |
| |
| if [[ ! -f "${toc}" ]]; then |
| # initializes the per-category _toc.yaml |
| { |
| echo "toc:" |
| } > "${toc}" |
| |
| # initializes the per-category index |
| { |
| echo "# $(get-category-title "${category}")" |
| echo |
| echo "Command | Description" |
| echo "------- | -----------" |
| } > "${cat_base}.md" |
| fi |
| echo "${norm_cat}" |
| } |
| |
| function handle-command { |
| local cmd_path="$1" |
| local category="$(get_metadata_camelcase "${cmd_path}" "CATEGORY")" |
| if [[ -z "${category}" ]]; then |
| category="Other" |
| fi |
| categories+=( "${category}" ) |
| local norm_cat="$(maybe-add-category "${category}")" |
| local cmd_name="$(basename "${cmd_path}" ".fx")" |
| local filename="${base_out}/cmd/${cmd_name}.md" |
| local deprecated_str="" |
| local deprecated="$(get_binary_metadata "${cmd_path}" "DEPRECATED")" |
| if [[ "${deprecated}" -eq 1 ]]; then |
| if [[ "${show_deprecated}" -eq 0 ]]; then |
| return |
| fi |
| deprecated_str="DEPRECATED! " |
| fi |
| local summary="${deprecated_str}$(get_summary "${cmd_path}")" |
| local cmd_url="cmd/${cmd_name}.md" |
| local cat_url="${norm_cat}" |
| |
| local rel_path="${cmd_path#${FUCHSIA_DIR}/}" |
| |
| # Creates the per-command markdown file |
| { |
| if [[ -n "${vendor}" ]]; then |
| echo "# ${deprecated_str}fx vendor ${vendor} ${cmd_name}" |
| else |
| echo "# ${deprecated_str}fx ${cmd_name}" |
| fi |
| echo "" |
| echo "${summary}" |
| echo |
| echo '```none' |
| fx-print-command-help "$cmd_path" |
| echo '```' |
| if [[ -z "${vendor}" ]]; then |
| echo |
| echo "[${cmd_name} source code](https://fuchsia.googlesource.com/fuchsia/+/HEAD/${rel_path})" |
| fi |
| } > "${filename}" |
| |
| # Appends the command to the per-category index |
| { |
| echo "[${cmd_name}](cmd/${cmd_name}.md) | ${summary}" |
| } >> "${base_out}/${norm_cat}.md" |
| |
| # Appends the command to the global index |
| { |
| echo "[${cmd_name}](${cmd_url}) | [${category}](${cat_url}) | ${summary}" |
| } >> "${base_out}/index.md" |
| |
| # Appends the command to the per-category _toc.yaml |
| { |
| echo "- title: \"${deprecated_str}${cmd_name}\"" |
| echo " path: ${toc_prefix}/${cmd_url}" |
| } >> "${base_out}/${norm_cat}_toc.yaml" |
| |
| } |
| |
| main "$@" |