blob: 3118d9472615fe8b598ed0e47a2a948250192434 [file] [log] [blame]
Jan Kundrátdc2b0722018-03-02 14:13:37 +01001/*
2 * Copyright (C) 2018 CESNET, https://photonics.cesnet.cz/
Jan Kundráta2740442018-03-22 16:56:43 +01003 * Copyright (C) 2018 FIT CVUT, https://fit.cvut.cz/
Jan Kundrátdc2b0722018-03-02 14:13:37 +01004 *
Václav Kubernát624a8872018-03-02 17:28:47 +01005 * Written by Václav Kubernát <kubervac@fit.cvut.cz>
Jan Kundrátdc2b0722018-03-02 14:13:37 +01006 *
7*/
Václav Kubernátfd261162020-11-12 02:36:29 +01008#include <atomic>
Jan Kundrátdc2b0722018-03-02 14:13:37 +01009#include <docopt.h>
Václav Kubernát624a8872018-03-02 17:28:47 +010010#include <iostream>
Václav Kubernát435706e2019-02-20 18:05:59 +010011#include <optional>
Václav Kubernáta395d332019-02-13 16:49:20 +010012#include <replxx.hxx>
Václav Kubernát90de9502019-11-20 17:19:44 +010013#include <sstream>
Jan Kundrátdc2b0722018-03-02 14:13:37 +010014#include "NETCONF_CLI_VERSION.h"
Václav Kubernát96344a12018-05-28 16:33:39 +020015#include "interpreter.hpp"
Václav Kubernát48e9dfa2020-07-08 10:55:12 +020016#include "proxy_datastore.hpp"
Václav Kubernáte2e15ee2020-02-05 17:38:13 +010017#include "yang_schema.hpp"
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010018#if defined(SYSREPO_CLI)
Václav Kubernát6415b822018-08-22 17:40:01 +020019#include "sysrepo_access.hpp"
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010020#define PROGRAM_NAME "sysrepo-cli"
Václav Kubernát715c85c2020-04-14 01:46:08 +020021static const auto usage = R"(CLI interface to sysrepo
22
23Usage:
24 sysrepo-cli [-d <datastore>]
25 sysrepo-cli (-h | --help)
26 sysrepo-cli --version
27
28Options:
29 -d <datastore> can be "running" or "startup" [default: running])";
Václav Kubernát74487df2020-06-04 01:29:28 +020030#elif defined(YANG_CLI)
31#include <boost/spirit/home/x3.hpp>
Václav Kubernát619e6542020-06-29 14:13:43 +020032#include <filesystem>
Václav Kubernát74487df2020-06-04 01:29:28 +020033#include "yang_access.hpp"
34#define PROGRAM_NAME "yang-cli"
35static const auto usage = R"(CLI interface for creating local YANG data instances
36
Václav Kubernát619e6542020-06-29 14:13:43 +020037 The <schema_file_or_module_name> argument is treated as a file name if a file
38 with such a path exists, otherwise it's treated as a module name. Search dirs
39 will be used to find a schema for that module.
40
Václav Kubernát74487df2020-06-04 01:29:28 +020041Usage:
Václav Kubernát28cf3362020-06-29 17:52:51 +020042 yang-cli [--configonly] [-s <search_dir>] [-e enable_features]... [-i data_file]... <schema_file_or_module_name>...
Václav Kubernát74487df2020-06-04 01:29:28 +020043 yang-cli (-h | --help)
44 yang-cli --version
45
46Options:
47 -s <search_dir> Set search for schema lookup
Václav Kubernát548cb192020-06-26 14:00:42 +020048 -e <enable_features> Feature to enable after modules are loaded. This option can be supplied more than once. Format: <module_name>:<feature>
Václav Kubernát28cf3362020-06-29 17:52:51 +020049 -i <data_file> File to import data from
50 --configonly Disable editing of operational data)";
Václav Kubernáte2e15ee2020-02-05 17:38:13 +010051#elif defined(NETCONF_CLI)
52// FIXME: improve usage
53static const auto usage = R"(CLI interface for NETCONF
54
55Usage:
56 netconf-cli [-v] [-p <port>] <host>
57 netconf-cli (-h | --help)
58 netconf-cli --version
59
60Options:
61 -v enable verbose mode
62 -p <port> port number [default: 830]
63)";
Václav Kubernáte2e15ee2020-02-05 17:38:13 +010064#include "cli-netconf.hpp"
Václav Kubernátb4e5b182020-11-16 19:55:09 +010065#include "netconf_access.hpp"
Václav Kubernáte2e15ee2020-02-05 17:38:13 +010066#define PROGRAM_NAME "netconf-access"
Václav Kubernátfd261162020-11-12 02:36:29 +010067// FIXME: this should be replaced by C++20 std::jthread at some point
68struct PoorMansJThread {
69 ~PoorMansJThread()
70 {
71 if (thread.joinable()) {
72 thread.join();
73 }
74 }
75 std::thread thread;
76};
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010077#else
78#error "Unknown CLI backend"
79#endif
Jan Kundrátdc2b0722018-03-02 14:13:37 +010080
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010081const auto HISTORY_FILE_NAME = PROGRAM_NAME "_history";
82
Jan Kundrátdc2b0722018-03-02 14:13:37 +010083int main(int argc, char* argv[])
84{
85 auto args = docopt::docopt(usage,
86 {argv + 1, argv + argc},
87 true,
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010088 PROGRAM_NAME " " NETCONF_CLI_VERSION,
Jan Kundrátdc2b0722018-03-02 14:13:37 +010089 true);
Václav Kubernát28cf3362020-06-29 17:52:51 +020090 WritableOps writableOps = WritableOps::No;
Václav Kubernátff2c9f62018-05-16 20:26:31 +020091
Václav Kubernátfd261162020-11-12 02:36:29 +010092 using replxx::Replxx;
93 Replxx lineEditor;
94 std::atomic<int> backendReturnCode = 0;
95
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010096#if defined(SYSREPO_CLI)
Václav Kubernát715c85c2020-04-14 01:46:08 +020097 auto datastoreType = Datastore::Running;
98 if (const auto& ds = args["-d"]) {
99 if (ds.asString() == "startup") {
100 datastoreType = Datastore::Startup;
101 } else if (ds.asString() == "running") {
102 datastoreType = Datastore::Running;
103 } else {
Václav Kubernátb4e5b182020-11-16 19:55:09 +0100104 std::cerr << PROGRAM_NAME << ": unknown datastore: " << ds.asString() << "\n";
Václav Kubernát715c85c2020-04-14 01:46:08 +0200105 return 1;
106 }
107 }
Václav Kubernát654303f2020-07-31 13:16:54 +0200108 auto datastore = std::make_shared<SysrepoAccess>(datastoreType);
Václav Kubernát715c85c2020-04-14 01:46:08 +0200109 std::cout << "Connected to sysrepo [datastore: " << (datastoreType == Datastore::Startup ? "startup" : "running") << "]" << std::endl;
Václav Kubernát74487df2020-06-04 01:29:28 +0200110#elif defined(YANG_CLI)
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200111 auto datastore = std::make_shared<YangAccess>();
Václav Kubernát28cf3362020-06-29 17:52:51 +0200112 if (args["--configonly"].asBool()) {
113 writableOps = WritableOps::No;
114 } else {
115 writableOps = WritableOps::Yes;
116 std::cout << "ops is writable" << std::endl;
117 }
Václav Kubernát74487df2020-06-04 01:29:28 +0200118 if (const auto& search_dir = args["-s"]) {
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200119 datastore->addSchemaDir(search_dir.asString());
Václav Kubernát74487df2020-06-04 01:29:28 +0200120 }
Václav Kubernát619e6542020-06-29 14:13:43 +0200121 for (const auto& schemaFile : args["<schema_file_or_module_name>"].asStringList()) {
122 if (std::filesystem::exists(schemaFile)) {
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200123 datastore->addSchemaFile(schemaFile);
Václav Kubernát619e6542020-06-29 14:13:43 +0200124 } else if (schemaFile.find('/') == std::string::npos) { // Module names cannot have a slash
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200125 datastore->loadModule(schemaFile);
Václav Kubernát619e6542020-06-29 14:13:43 +0200126 } else {
127 std::cerr << "Cannot load YANG module " << schemaFile << "\n";
128 }
Václav Kubernát74487df2020-06-04 01:29:28 +0200129 }
130 if (const auto& enableFeatures = args["-e"]) {
131 namespace x3 = boost::spirit::x3;
132 auto grammar = +(x3::char_-":") >> ":" >> +(x3::char_-":");
133 for (const auto& enableFeature : enableFeatures.asStringList()) {
134 std::pair<std::string, std::string> parsed;
135 auto it = enableFeature.begin();
136 auto res = x3::parse(it, enableFeature.cend(), grammar, parsed);
137 if (!res || it != enableFeature.cend()) {
138 std::cerr << "Error parsing feature enable flags: " << enableFeature << "\n";
139 return 1;
140 }
141 try {
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200142 datastore->enableFeature(parsed.first, parsed.second);
Václav Kubernát74487df2020-06-04 01:29:28 +0200143 } catch (std::runtime_error& ex) {
144 std::cerr << ex.what() << "\n";
145 return 1;
146 }
Václav Kubernát74487df2020-06-04 01:29:28 +0200147 }
148 }
Václav Kubernát548cb192020-06-26 14:00:42 +0200149 if (const auto& dataFiles = args["-i"]) {
150 for (const auto& dataFile : dataFiles.asStringList()) {
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200151 datastore->addDataFile(dataFile);
Václav Kubernát548cb192020-06-26 14:00:42 +0200152 }
153 }
Václav Kubernáte2e15ee2020-02-05 17:38:13 +0100154#elif defined(NETCONF_CLI)
155 auto verbose = args.at("-v").asBool();
156 if (verbose) {
157 NetconfAccess::setNcLogLevel(NC_VERB_DEBUG);
158 }
159
160 SshProcess process;
Václav Kubernátfd261162020-11-12 02:36:29 +0100161 PoorMansJThread processWatcher;
Václav Kubernáte2e15ee2020-02-05 17:38:13 +0100162 std::shared_ptr<NetconfAccess> datastore;
163
164 try {
165 process = sshProcess(args.at("<host>").asString(), args.at("-p").asString());
Václav Kubernátfd261162020-11-12 02:36:29 +0100166 processWatcher.thread = std::thread([&process, &lineEditor, &backendReturnCode] () {
167 process.process.wait();
168 backendReturnCode = process.process.exit_code();
169 // CTRL-U clears from the cursor to the start of the line
170 // CTRL-K clears from the cursor to the end of the line
171 // CTRL-D send EOF
172 lineEditor.emulate_key_press(replxx::Replxx::KEY::control('U'));
173 lineEditor.emulate_key_press(replxx::Replxx::KEY::control('K'));
174 lineEditor.emulate_key_press(replxx::Replxx::KEY::control('D'));
175 });
Václav Kubernáte2e15ee2020-02-05 17:38:13 +0100176 datastore = std::make_shared<NetconfAccess>(process.std_out.native_source(), process.std_in.native_sink());
177 } catch (std::runtime_error& ex) {
178 std::cerr << "SSH connection failed: " << ex.what() << "\n";
179 return 1;
180 }
Václav Kubernátb79f3ca2020-02-04 15:56:01 +0100181#else
182#error "Unknown CLI backend"
183#endif
184
Václav Kubernáte2e15ee2020-02-05 17:38:13 +0100185#if defined(SYSREPO_CLI) || defined(NETCONF_CLI)
Václav Kubernáte7248b22020-06-26 15:38:59 +0200186 auto createTemporaryDatastore = [](const std::shared_ptr<DatastoreAccess>& datastore) {
187 return std::make_shared<YangAccess>(std::static_pointer_cast<YangSchema>(datastore->schema()));
188 };
189#elif defined(YANG_CLI)
190 auto createTemporaryDatastore = [](const std::shared_ptr<DatastoreAccess>&) {
191 return nullptr;
192 };
193#endif
194
195 ProxyDatastore proxyDatastore(datastore, createTemporaryDatastore);
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200196 auto dataQuery = std::make_shared<DataQuery>(*datastore);
197 Parser parser(datastore->schema(), writableOps, dataQuery);
Václav Kubernát395d92c2020-01-24 12:18:18 +0100198
Václav Kubernátb4e5b182020-11-16 19:55:09 +0100199 lineEditor.bind_key(Replxx::KEY::meta(Replxx::KEY::BACKSPACE), [&lineEditor](const auto& code) {
Václav Kubernát48637292020-07-08 16:54:13 +0200200 return lineEditor.invoke(Replxx::ACTION::KILL_TO_BEGINING_OF_WORD, code);
201 });
Václav Kubernátb4e5b182020-11-16 19:55:09 +0100202 lineEditor.bind_key(Replxx::KEY::control('W'), [&lineEditor](const auto& code) {
Václav Kubernát48637292020-07-08 16:54:13 +0200203 return lineEditor.invoke(Replxx::ACTION::KILL_TO_WHITESPACE_ON_LEFT, code);
204 });
Václav Kubernát395d92c2020-01-24 12:18:18 +0100205
206 lineEditor.set_word_break_characters("\t _[]/:'\"=-%");
207
Václav Kubernát1ed4aa32020-01-23 13:13:28 +0100208 lineEditor.set_completion_callback([&parser](const std::string& input, int& context) {
Václav Kubernáta395d332019-02-13 16:49:20 +0100209 std::stringstream stream;
Václav Kubernát1ed4aa32020-01-23 13:13:28 +0100210 auto completions = parser.completeCommand(input, stream);
Václav Kubernáta395d332019-02-13 16:49:20 +0100211
Jan Kundrát8d8efe82019-10-18 10:21:36 +0200212 std::vector<replxx::Replxx::Completion> res;
Václav Kubernát1ed4aa32020-01-23 13:13:28 +0100213 std::copy(completions.m_completions.begin(), completions.m_completions.end(), std::back_inserter(res));
214 context = completions.m_contextLength;
Václav Kubernáta395d332019-02-13 16:49:20 +0100215 return res;
216 });
Václav Kubernát2b684612018-08-09 18:55:24 +0200217
Václav Kubernát435706e2019-02-20 18:05:59 +0100218 std::optional<std::string> historyFile;
219 if (auto xdgHome = getenv("XDG_DATA_HOME")) {
220 historyFile = std::string(xdgHome) + "/" + HISTORY_FILE_NAME;
221 } else if (auto home = getenv("HOME")) {
222 historyFile = std::string(home) + "/.local/share/" + HISTORY_FILE_NAME;
223 }
224
Václav Kubernát3a433232020-07-08 17:52:50 +0200225 if (historyFile) {
Václav Kubernát435706e2019-02-20 18:05:59 +0100226 lineEditor.history_load(historyFile.value());
Václav Kubernát3a433232020-07-08 17:52:50 +0200227 }
Václav Kubernát435706e2019-02-20 18:05:59 +0100228
Václav Kubernátfd261162020-11-12 02:36:29 +0100229 while (backendReturnCode == 0) {
Václav Kubernáta395d332019-02-13 16:49:20 +0100230 auto line = lineEditor.input(parser.currentNode() + "> ");
231 if (!line) {
Václav Kubernát82bf1312019-11-05 11:19:26 +0100232 // If user pressed CTRL-C to abort the line, errno gets set to EAGAIN.
233 // If user pressed CTRL-D (for EOF), errno doesn't get set to EAGAIN, so we exit the program.
234 // I have no idea why replxx uses errno for this.
235 if (errno == EAGAIN) {
236 continue;
237 } else {
238 break;
239 }
Václav Kubernát5b80e522019-01-25 12:17:03 +0100240 }
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200241
Jan Kundráte3877022018-09-05 15:32:09 +0200242 std::locale C_locale("C");
Václav Kubernáta395d332019-02-13 16:49:20 +0100243 std::string_view view{line};
244 if (std::all_of(view.begin(), view.end(),
Jan Kundráte3877022018-09-05 15:32:09 +0200245 [C_locale](const auto c) { return std::isspace(c, C_locale);})) {
246 continue;
247 }
248
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200249 try {
Václav Kubernát5b80e522019-01-25 12:17:03 +0100250 command_ cmd = parser.parseCommand(line, std::cout);
Václav Kubernát48e9dfa2020-07-08 10:55:12 +0200251 boost::apply_visitor(Interpreter(parser, proxyDatastore), cmd);
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200252 } catch (InvalidCommandException& ex) {
253 std::cerr << ex.what() << std::endl;
Václav Kubernátc58e4aa2019-04-03 18:37:32 +0200254 } catch (DatastoreException& ex) {
255 std::cerr << ex.what() << std::endl;
Václav Kubernáte7248b22020-06-26 15:38:59 +0200256 } catch (std::runtime_error& ex) {
257 std::cerr << ex.what() << std::endl;
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200258 }
Václav Kubernát5b80e522019-01-25 12:17:03 +0100259
Václav Kubernáta395d332019-02-13 16:49:20 +0100260 lineEditor.history_add(line);
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200261 }
262
Václav Kubernát3a433232020-07-08 17:52:50 +0200263 if (historyFile) {
Václav Kubernát435706e2019-02-20 18:05:59 +0100264 lineEditor.history_save(historyFile.value());
Václav Kubernát3a433232020-07-08 17:52:50 +0200265 }
Václav Kubernát435706e2019-02-20 18:05:59 +0100266
Václav Kubernátfd261162020-11-12 02:36:29 +0100267 return backendReturnCode;
Jan Kundrátdc2b0722018-03-02 14:13:37 +0100268}