mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-28 08:16:09 +03:00
317 lines
11 KiB
C++
317 lines
11 KiB
C++
|
// Copyright (c) 2013 The Chromium Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style license that can be
|
||
|
// found in the LICENSE file.
|
||
|
|
||
|
#include <algorithm>
|
||
|
#include <iostream>
|
||
|
|
||
|
#include "base/command_line.h"
|
||
|
#include "tools/gn/args.h"
|
||
|
#include "tools/gn/commands.h"
|
||
|
#include "tools/gn/err.h"
|
||
|
#include "tools/gn/functions.h"
|
||
|
#include "tools/gn/input_conversion.h"
|
||
|
#include "tools/gn/label.h"
|
||
|
#include "tools/gn/label_pattern.h"
|
||
|
#include "tools/gn/ninja_build_writer.h"
|
||
|
#include "tools/gn/parser.h"
|
||
|
#include "tools/gn/runtime_deps.h"
|
||
|
#include "tools/gn/setup.h"
|
||
|
#include "tools/gn/standard_out.h"
|
||
|
#include "tools/gn/string_utils.h"
|
||
|
#include "tools/gn/substitution_writer.h"
|
||
|
#include "tools/gn/switches.h"
|
||
|
#include "tools/gn/target.h"
|
||
|
#include "tools/gn/variables.h"
|
||
|
|
||
|
namespace commands {
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
void PrintToplevelHelp() {
|
||
|
PrintSectionHelp("Commands", "<command>", "commands");
|
||
|
for (const auto& cmd : commands::GetCommands())
|
||
|
PrintShortHelp(cmd.second.help_short);
|
||
|
|
||
|
// Target declarations.
|
||
|
PrintSectionHelp("Target declarations", "<function>", "targets");
|
||
|
for (const auto& func : functions::GetFunctions()) {
|
||
|
if (func.second.is_target)
|
||
|
PrintShortHelp(func.second.help_short);
|
||
|
}
|
||
|
|
||
|
// Functions.
|
||
|
PrintSectionHelp("Buildfile functions", "<function>", "functions");
|
||
|
for (const auto& func : functions::GetFunctions()) {
|
||
|
if (!func.second.is_target)
|
||
|
PrintShortHelp(func.second.help_short);
|
||
|
}
|
||
|
|
||
|
// Built-in variables.
|
||
|
PrintSectionHelp("Built-in predefined variables", "<variable>",
|
||
|
"predefined_variables");
|
||
|
for (const auto& builtin : variables::GetBuiltinVariables())
|
||
|
PrintShortHelp(builtin.second.help_short);
|
||
|
|
||
|
// Target variables.
|
||
|
PrintSectionHelp("Variables you set in targets", "<variable>",
|
||
|
"target_variables");
|
||
|
for (const auto& target : variables::GetTargetVariables())
|
||
|
PrintShortHelp(target.second.help_short);
|
||
|
|
||
|
PrintSectionHelp("Other help topics", "", "other");
|
||
|
PrintShortHelp("all: Print all the help at once");
|
||
|
PrintShortHelp("buildargs: How build arguments work.");
|
||
|
PrintShortHelp("dotfile: Info about the toplevel .gn file.");
|
||
|
PrintShortHelp("execution: Build graph and execution overview.");
|
||
|
PrintShortHelp("grammar: Language and grammar for GN build files.");
|
||
|
PrintShortHelp(
|
||
|
"input_conversion: Processing input from exec_script and read_file.");
|
||
|
PrintShortHelp("label_pattern: Matching more than one label.");
|
||
|
PrintShortHelp("labels: About labels.");
|
||
|
PrintShortHelp("ninja_rules: How Ninja build rules are named.");
|
||
|
PrintShortHelp("nogncheck: Annotating includes for checking.");
|
||
|
PrintShortHelp("runtime_deps: How runtime dependency computation works.");
|
||
|
PrintShortHelp("source_expansion: Map sources to outputs for scripts.");
|
||
|
PrintShortHelp("switches: Show available command-line switches.");
|
||
|
}
|
||
|
|
||
|
void PrintSwitchHelp() {
|
||
|
const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess();
|
||
|
bool is_markdown = cmdline->HasSwitch(switches::kMarkdown);
|
||
|
|
||
|
OutputString("Available global switches\n", DECORATION_YELLOW);
|
||
|
OutputString(
|
||
|
" Do \"gn help --the_switch_you_want_help_on\" for more. Individual\n"
|
||
|
" commands may take command-specific switches not listed here. See the\n"
|
||
|
" help on your specific command for more.\n\n");
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("```\n", DECORATION_NONE);
|
||
|
|
||
|
for (const auto& s : switches::GetSwitches())
|
||
|
PrintShortHelp(s.second.short_help);
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("```\n", DECORATION_NONE);
|
||
|
|
||
|
OutputString("\n");
|
||
|
}
|
||
|
|
||
|
void PrintAllHelp() {
|
||
|
const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess();
|
||
|
bool is_markdown = cmdline->HasSwitch(switches::kMarkdown);
|
||
|
|
||
|
if (is_markdown) {
|
||
|
OutputString("# GN Reference\n\n");
|
||
|
OutputString("*This page is automatically generated from* "
|
||
|
"`gn help --markdown all`.\n\n");
|
||
|
|
||
|
// Generate our own table of contents so that we have more control
|
||
|
// over what's in and out.
|
||
|
OutputString("## Contents\n\n");
|
||
|
}
|
||
|
|
||
|
PrintToplevelHelp();
|
||
|
|
||
|
OutputString("\n");
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("## <a name=\"commands\"></a>Commands\n\n");
|
||
|
for (const auto& c: commands::GetCommands())
|
||
|
PrintLongHelp(c.second.help);
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("## <a name=\"targets\"></a>Target declarations\n\n");
|
||
|
for (const auto& f : functions::GetFunctions()) {
|
||
|
if (f.second.is_target)
|
||
|
PrintLongHelp(f.second.help);
|
||
|
}
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("## <a name=\"functions\"></a>Buildfile functions\n\n");
|
||
|
for (const auto& f : functions::GetFunctions()) {
|
||
|
if (!f.second.is_target)
|
||
|
PrintLongHelp(f.second.help);
|
||
|
}
|
||
|
|
||
|
if (is_markdown) {
|
||
|
OutputString(
|
||
|
"## <a name=\"predefined_variables\"></a>"
|
||
|
"Built-in predefined variables\n\n");
|
||
|
}
|
||
|
for (const auto& v: variables::GetBuiltinVariables())
|
||
|
PrintLongHelp(v.second.help);
|
||
|
|
||
|
if (is_markdown) {
|
||
|
OutputString(
|
||
|
"## <a name=\"target_variables\"></a>"
|
||
|
"Variables you set in targets\n\n");
|
||
|
}
|
||
|
for (const auto& v: variables::GetTargetVariables())
|
||
|
PrintLongHelp(v.second.help);
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("## <a name=\"other\"></a>Other help topics\n\n");
|
||
|
PrintLongHelp(kBuildArgs_Help, "buildargs");
|
||
|
PrintLongHelp(kDotfile_Help, "dotfile");
|
||
|
PrintLongHelp(kExecution_Help, "execution");
|
||
|
PrintLongHelp(kGrammar_Help, "grammar");
|
||
|
PrintLongHelp(kInputConversion_Help, "input_conversion");
|
||
|
PrintLongHelp(kLabelPattern_Help, "label_pattern");
|
||
|
PrintLongHelp(kLabels_Help, "labels");
|
||
|
PrintLongHelp(kNinjaRules_Help, "ninja_rules");
|
||
|
PrintLongHelp(kNoGnCheck_Help, "nogncheck");
|
||
|
PrintLongHelp(kRuntimeDeps_Help, "runtime_deps");
|
||
|
PrintLongHelp(kSourceExpansion_Help, "source_expansion");
|
||
|
|
||
|
if (is_markdown)
|
||
|
OutputString("## <a name=\"switches\"></a>Command Line Switches\n\n");
|
||
|
PrintSwitchHelp();
|
||
|
}
|
||
|
|
||
|
// Prints help on the given switch. There should be no leading hyphens. Returns
|
||
|
// true if the switch was found and help was printed. False means the switch is
|
||
|
// unknown.
|
||
|
bool PrintHelpOnSwitch(const std::string& what) {
|
||
|
const switches::SwitchInfoMap& all = switches::GetSwitches();
|
||
|
switches::SwitchInfoMap::const_iterator found =
|
||
|
all.find(base::StringPiece(what));
|
||
|
if (found == all.end())
|
||
|
return false;
|
||
|
PrintLongHelp(found->second.long_help);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
const char kHelp[] = "help";
|
||
|
const char kHelp_HelpShort[] =
|
||
|
"help: Does what you think.";
|
||
|
const char kHelp_Help[] =
|
||
|
R"(gn help <anything>
|
||
|
|
||
|
Yo dawg, I heard you like help on your help so I put help on the help in the
|
||
|
help.
|
||
|
|
||
|
You can also use "all" as the parameter to get all help at once.
|
||
|
|
||
|
Switches
|
||
|
|
||
|
--markdown
|
||
|
Format output in markdown syntax.
|
||
|
|
||
|
Example
|
||
|
|
||
|
gn help --markdown all
|
||
|
Dump all help to stdout in markdown format.
|
||
|
)";
|
||
|
|
||
|
int RunHelp(const std::vector<std::string>& args) {
|
||
|
std::string what;
|
||
|
if (args.size() == 0) {
|
||
|
// If no argument is specified, check for switches to allow things like
|
||
|
// "gn help --args" for help on the args switch.
|
||
|
const base::CommandLine::SwitchMap& switches =
|
||
|
base::CommandLine::ForCurrentProcess()->GetSwitches();
|
||
|
if (switches.empty()) {
|
||
|
// Still nothing, show help overview.
|
||
|
PrintToplevelHelp();
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
// Switch help needs to be done separately. The CommandLine will strip the
|
||
|
// switch separators so --args will come out as "args" which is then
|
||
|
// ambiguous with the variable named "args".
|
||
|
if (!PrintHelpOnSwitch(switches.begin()->first))
|
||
|
PrintToplevelHelp();
|
||
|
return 0;
|
||
|
} else {
|
||
|
what = args[0];
|
||
|
}
|
||
|
|
||
|
std::vector<base::StringPiece> all_help_topics;
|
||
|
|
||
|
// Check commands.
|
||
|
const commands::CommandInfoMap& command_map = commands::GetCommands();
|
||
|
auto found_command = command_map.find(what);
|
||
|
if (found_command != command_map.end()) {
|
||
|
PrintLongHelp(found_command->second.help);
|
||
|
return 0;
|
||
|
}
|
||
|
for (const auto& entry : command_map)
|
||
|
all_help_topics.push_back(entry.first);
|
||
|
|
||
|
// Check functions.
|
||
|
const functions::FunctionInfoMap& function_map = functions::GetFunctions();
|
||
|
auto found_function = function_map.find(what);
|
||
|
if (found_function != function_map.end())
|
||
|
PrintLongHelp(found_function->second.help);
|
||
|
for (const auto& entry : function_map)
|
||
|
all_help_topics.push_back(entry.first);
|
||
|
|
||
|
// Builtin variables.
|
||
|
const variables::VariableInfoMap& builtin_vars =
|
||
|
variables::GetBuiltinVariables();
|
||
|
auto found_builtin_var = builtin_vars.find(what);
|
||
|
if (found_builtin_var != builtin_vars.end())
|
||
|
PrintLongHelp(found_builtin_var->second.help);
|
||
|
for (const auto& entry : builtin_vars)
|
||
|
all_help_topics.push_back(entry.first);
|
||
|
|
||
|
// Target variables.
|
||
|
const variables::VariableInfoMap& target_vars =
|
||
|
variables::GetTargetVariables();
|
||
|
auto found_target_var = target_vars.find(what);
|
||
|
if (found_target_var != target_vars.end())
|
||
|
PrintLongHelp(found_target_var->second.help);
|
||
|
for (const auto& entry : target_vars)
|
||
|
all_help_topics.push_back(entry.first);
|
||
|
|
||
|
if (found_function != function_map.end() ||
|
||
|
found_builtin_var != builtin_vars.end() ||
|
||
|
found_target_var != target_vars.end())
|
||
|
return 0;
|
||
|
|
||
|
// Random other topics.
|
||
|
std::map<std::string, void(*)()> random_topics;
|
||
|
random_topics["all"] = PrintAllHelp;
|
||
|
random_topics["execution"] = []() { PrintLongHelp(kExecution_Help); };
|
||
|
random_topics["buildargs"] = []() { PrintLongHelp(kBuildArgs_Help); };
|
||
|
random_topics["dotfile"] = []() { PrintLongHelp(kDotfile_Help); };
|
||
|
random_topics["grammar"] = []() { PrintLongHelp(kGrammar_Help); };
|
||
|
random_topics["input_conversion"] = []() {
|
||
|
PrintLongHelp(kInputConversion_Help);
|
||
|
};
|
||
|
random_topics["label_pattern"] = []() { PrintLongHelp(kLabelPattern_Help); };
|
||
|
random_topics["labels"] = []() { PrintLongHelp(kLabels_Help); };
|
||
|
random_topics["ninja_rules"] = []() { PrintLongHelp(kNinjaRules_Help); };
|
||
|
random_topics["nogncheck"] = []() { PrintLongHelp(kNoGnCheck_Help); };
|
||
|
random_topics["runtime_deps"] = []() { PrintLongHelp(kRuntimeDeps_Help); };
|
||
|
random_topics["source_expansion"] = []() {
|
||
|
PrintLongHelp(kSourceExpansion_Help);
|
||
|
};
|
||
|
random_topics["switches"] = PrintSwitchHelp;
|
||
|
auto found_random_topic = random_topics.find(what);
|
||
|
if (found_random_topic != random_topics.end()) {
|
||
|
found_random_topic->second();
|
||
|
return 0;
|
||
|
}
|
||
|
for (const auto& entry : random_topics)
|
||
|
all_help_topics.push_back(entry.first);
|
||
|
|
||
|
// No help on this.
|
||
|
Err(Location(), "No help on \"" + what + "\".").PrintToStdout();
|
||
|
base::StringPiece suggestion = SpellcheckString(what, all_help_topics);
|
||
|
if (suggestion.empty()) {
|
||
|
OutputString("Run `gn help` for a list of available topics.\n",
|
||
|
DECORATION_NONE);
|
||
|
} else {
|
||
|
OutputString("Did you mean `gn help " + suggestion.as_string() + "`?\n",
|
||
|
DECORATION_NONE);
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
} // namespace commands
|