1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
|
// Copyright (C) 2018-2022 Internet Systems Consortium, Inc. ("ISC")
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
/// @file unix_control_socket.cc
/// Contains the UNIX socket derived class for control socket communication.
#include <config.h>
#include <asiolink/asio_wrapper.h>
#include <asiolink/io_service.h>
#include <cc/command_interpreter.h>
#include <cc/json_feed.h>
#include <config/client_connection.h>
#include <config/timeouts.h>
#include <netconf/unix_control_socket.h>
using namespace std;
using namespace isc::asiolink;
using namespace isc::config;
using namespace isc::data;
namespace isc {
namespace netconf {
template <> ControlSocketBasePtr
createControlSocket<CfgControlSocket::Type::UNIX>(CfgControlSocketPtr ctrl_sock) {
return (UnixControlSocketPtr(new UnixControlSocket(ctrl_sock)));
}
UnixControlSocket::UnixControlSocket(CfgControlSocketPtr ctrl_sock)
: ControlSocketBase(ctrl_sock) {
}
ConstElementPtr
UnixControlSocket::configGet(const string& /*service*/) {
return (sendCommand(createCommand("config-get")));
}
ConstElementPtr
UnixControlSocket::configTest(ElementPtr config,
const string& /*service*/) {
return (sendCommand(createCommand("config-test", config)));
}
ConstElementPtr
UnixControlSocket::configSet(ElementPtr config,
const string& /*service*/) {
return (sendCommand(createCommand("config-set", config)));
}
ConstElementPtr
UnixControlSocket::sendCommand(ConstElementPtr command) {
// We are using our own IO service because this method is synchronous.
IOServicePtr io_service(new IOService());
ClientConnection conn(*io_service);
boost::system::error_code received_ec;
ConstJSONFeedPtr received_feed;
conn.start(ClientConnection::SocketPath(getName()),
ClientConnection::ControlCommand(command->toWire()),
[&io_service, &received_ec, &received_feed]
(const boost::system::error_code& ec, ConstJSONFeedPtr feed) {
// Capture error code and parsed data.
received_ec = ec;
received_feed = feed;
// Got the IO service so stop IO service. This causes to
// stop IO service when all handlers have been invoked.
io_service->stopWork();
},
ClientConnection::Timeout(TIMEOUT_AGENT_FORWARD_COMMAND));
// Perform this synchronously.
io_service->run();
if (received_ec) {
// Got an error.
isc_throw(ControlSocketError, "communication error: "
<< received_ec.message());
}
if (!received_feed) {
// Failed to get the answer.
isc_throw(ControlSocketError, "empty response");
}
try {
return (received_feed->toElement());
} catch (exception const& ex) {
isc_throw(ControlSocketError, "unparsable response: " << ex.what());
}
}
} // namespace netconf
} // namespace isc
|