blob: b036ebcedfe7394af75248fcbce86d1b9254123b [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*/
Jan Kundrátdc2b0722018-03-02 14:13:37 +01008#include <docopt.h>
Václav Kubernát624a8872018-03-02 17:28:47 +01009#include <iostream>
Václav Kubernát435706e2019-02-20 18:05:59 +010010#include <optional>
Václav Kubernáta395d332019-02-13 16:49:20 +010011#include <replxx.hxx>
Václav Kubernát90de9502019-11-20 17:19:44 +010012#include <sstream>
Jan Kundrátdc2b0722018-03-02 14:13:37 +010013#include "NETCONF_CLI_VERSION.h"
Václav Kubernát96344a12018-05-28 16:33:39 +020014#include "interpreter.hpp"
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010015#if defined(SYSREPO_CLI)
Václav Kubernát6415b822018-08-22 17:40:01 +020016#include "sysrepo_access.hpp"
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010017#define PROGRAM_NAME "sysrepo-cli"
Václav Kubernát715c85c2020-04-14 01:46:08 +020018static const auto usage = R"(CLI interface to sysrepo
19
20Usage:
21 sysrepo-cli [-d <datastore>]
22 sysrepo-cli (-h | --help)
23 sysrepo-cli --version
24
25Options:
26 -d <datastore> can be "running" or "startup" [default: running])";
Václav Kubernát74487df2020-06-04 01:29:28 +020027#elif defined(YANG_CLI)
28#include <boost/spirit/home/x3.hpp>
29#include "yang_access.hpp"
30#define PROGRAM_NAME "yang-cli"
31static const auto usage = R"(CLI interface for creating local YANG data instances
32
33Usage:
34 yang-cli [-s <search_dir>] [-e enable_features]... <schema_file>...
35 yang-cli (-h | --help)
36 yang-cli --version
37
38Options:
39 -s <search_dir> Set search for schema lookup
40 -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átb79f3ca2020-02-04 15:56:01 +010041#else
42#error "Unknown CLI backend"
43#endif
Jan Kundrátdc2b0722018-03-02 14:13:37 +010044
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010045const auto HISTORY_FILE_NAME = PROGRAM_NAME "_history";
46
Jan Kundrátdc2b0722018-03-02 14:13:37 +010047int main(int argc, char* argv[])
48{
49 auto args = docopt::docopt(usage,
50 {argv + 1, argv + argc},
51 true,
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010052 PROGRAM_NAME " " NETCONF_CLI_VERSION,
Jan Kundrátdc2b0722018-03-02 14:13:37 +010053 true);
Václav Kubernátff2c9f62018-05-16 20:26:31 +020054
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010055#if defined(SYSREPO_CLI)
Václav Kubernát715c85c2020-04-14 01:46:08 +020056 auto datastoreType = Datastore::Running;
57 if (const auto& ds = args["-d"]) {
58 if (ds.asString() == "startup") {
59 datastoreType = Datastore::Startup;
60 } else if (ds.asString() == "running") {
61 datastoreType = Datastore::Running;
62 } else {
63 std::cerr << PROGRAM_NAME << ": unknown datastore: " << ds.asString() << "\n";
64 return 1;
65 }
66 }
67 SysrepoAccess datastore(PROGRAM_NAME, datastoreType);
68 std::cout << "Connected to sysrepo [datastore: " << (datastoreType == Datastore::Startup ? "startup" : "running") << "]" << std::endl;
Václav Kubernát74487df2020-06-04 01:29:28 +020069#elif defined(YANG_CLI)
70 YangAccess datastore;
71 if (const auto& search_dir = args["-s"]) {
72 datastore.addSchemaDir(search_dir.asString());
73 }
74 for (const auto& schemaFile : args["<schema_file>"].asStringList()) {
75 datastore.addSchemaFile(schemaFile);
76 }
77 if (const auto& enableFeatures = args["-e"]) {
78 namespace x3 = boost::spirit::x3;
79 auto grammar = +(x3::char_-":") >> ":" >> +(x3::char_-":");
80 for (const auto& enableFeature : enableFeatures.asStringList()) {
81 std::pair<std::string, std::string> parsed;
82 auto it = enableFeature.begin();
83 auto res = x3::parse(it, enableFeature.cend(), grammar, parsed);
84 if (!res || it != enableFeature.cend()) {
85 std::cerr << "Error parsing feature enable flags: " << enableFeature << "\n";
86 return 1;
87 }
88 try {
89 datastore.enableFeature(parsed.first, parsed.second);
90 } catch (std::runtime_error& ex) {
91 std::cerr << ex.what() << "\n";
92 return 1;
93 }
94
95 }
96 }
Václav Kubernátb79f3ca2020-02-04 15:56:01 +010097#else
98#error "Unknown CLI backend"
99#endif
100
Václav Kubernát43908fb2020-01-02 19:05:51 +0100101 auto dataQuery = std::make_shared<DataQuery>(datastore);
102 Parser parser(datastore.schema(), dataQuery);
Václav Kubernát395d92c2020-01-24 12:18:18 +0100103
104 using replxx::Replxx;
105
106 Replxx lineEditor;
107
108 lineEditor.bind_key(Replxx::KEY::meta(Replxx::KEY::BACKSPACE), std::bind(&Replxx::invoke, &lineEditor, Replxx::ACTION::KILL_TO_BEGINING_OF_WORD, std::placeholders::_1));
109 lineEditor.bind_key(Replxx::KEY::control('W'), std::bind(&Replxx::invoke, &lineEditor, Replxx::ACTION::KILL_TO_WHITESPACE_ON_LEFT, std::placeholders::_1));
110
111 lineEditor.set_word_break_characters("\t _[]/:'\"=-%");
112
Václav Kubernát1ed4aa32020-01-23 13:13:28 +0100113 lineEditor.set_completion_callback([&parser](const std::string& input, int& context) {
Václav Kubernáta395d332019-02-13 16:49:20 +0100114 std::stringstream stream;
Václav Kubernát1ed4aa32020-01-23 13:13:28 +0100115 auto completions = parser.completeCommand(input, stream);
Václav Kubernáta395d332019-02-13 16:49:20 +0100116
Jan Kundrát8d8efe82019-10-18 10:21:36 +0200117 std::vector<replxx::Replxx::Completion> res;
Václav Kubernát1ed4aa32020-01-23 13:13:28 +0100118 std::copy(completions.m_completions.begin(), completions.m_completions.end(), std::back_inserter(res));
119 context = completions.m_contextLength;
Václav Kubernáta395d332019-02-13 16:49:20 +0100120 return res;
121 });
Václav Kubernát2b684612018-08-09 18:55:24 +0200122
Václav Kubernát435706e2019-02-20 18:05:59 +0100123 std::optional<std::string> historyFile;
124 if (auto xdgHome = getenv("XDG_DATA_HOME")) {
125 historyFile = std::string(xdgHome) + "/" + HISTORY_FILE_NAME;
126 } else if (auto home = getenv("HOME")) {
127 historyFile = std::string(home) + "/.local/share/" + HISTORY_FILE_NAME;
128 }
129
130 if (historyFile)
131 lineEditor.history_load(historyFile.value());
132
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200133 while (true) {
Václav Kubernáta395d332019-02-13 16:49:20 +0100134 auto line = lineEditor.input(parser.currentNode() + "> ");
135 if (!line) {
Václav Kubernát82bf1312019-11-05 11:19:26 +0100136 // If user pressed CTRL-C to abort the line, errno gets set to EAGAIN.
137 // If user pressed CTRL-D (for EOF), errno doesn't get set to EAGAIN, so we exit the program.
138 // I have no idea why replxx uses errno for this.
139 if (errno == EAGAIN) {
140 continue;
141 } else {
142 break;
143 }
Václav Kubernát5b80e522019-01-25 12:17:03 +0100144 }
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200145
Jan Kundráte3877022018-09-05 15:32:09 +0200146 std::locale C_locale("C");
Václav Kubernáta395d332019-02-13 16:49:20 +0100147 std::string_view view{line};
148 if (std::all_of(view.begin(), view.end(),
Jan Kundráte3877022018-09-05 15:32:09 +0200149 [C_locale](const auto c) { return std::isspace(c, C_locale);})) {
150 continue;
151 }
152
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200153 try {
Václav Kubernát5b80e522019-01-25 12:17:03 +0100154 command_ cmd = parser.parseCommand(line, std::cout);
Václav Kubernát6415b822018-08-22 17:40:01 +0200155 boost::apply_visitor(Interpreter(parser, datastore), cmd);
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200156 } catch (InvalidCommandException& ex) {
157 std::cerr << ex.what() << std::endl;
Václav Kubernátc58e4aa2019-04-03 18:37:32 +0200158 } catch (DatastoreException& ex) {
159 std::cerr << ex.what() << std::endl;
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200160 }
Václav Kubernát5b80e522019-01-25 12:17:03 +0100161
Václav Kubernáta395d332019-02-13 16:49:20 +0100162 lineEditor.history_add(line);
Václav Kubernátff2c9f62018-05-16 20:26:31 +0200163 }
164
Václav Kubernát435706e2019-02-20 18:05:59 +0100165 if (historyFile)
166 lineEditor.history_save(historyFile.value());
167
Jan Kundrátdc2b0722018-03-02 14:13:37 +0100168 return 0;
169}