blob: 8e42507e1880cae22fa0d5d06d5a433d6ba7f44c [file] [log] [blame]
Ratan Gupta1dc91782018-04-19 16:47:12 +05301#include "config.h"
Patrick Williams1334b7b2021-02-22 17:15:12 -06002
Ratan Gupta1dc91782018-04-19 16:47:12 +05303#include "snmp_conf_manager.hpp"
Patrick Williams1334b7b2021-02-22 17:15:12 -06004
Ratan Gupta212f53e2018-04-30 17:28:05 +05305#include "snmp_serialize.hpp"
Ratan Gupta213517b2018-04-28 13:41:09 +05306#include "snmp_util.hpp"
7#include "xyz/openbmc_project/Common/error.hpp"
8
Patrick Williams1334b7b2021-02-22 17:15:12 -06009#include <arpa/inet.h>
10
Ratan Gupta213517b2018-04-28 13:41:09 +053011#include <phosphor-logging/elog-errors.hpp>
Ratan Gupta1dc91782018-04-19 16:47:12 +053012#include <phosphor-logging/log.hpp>
13
14#include <experimental/filesystem>
15
16namespace phosphor
17{
18namespace network
19{
20namespace snmp
21{
22
23using namespace phosphor::logging;
Ratan Gupta213517b2018-04-28 13:41:09 +053024using namespace sdbusplus::xyz::openbmc_project::Common::Error;
25using Argument = xyz::openbmc_project::Common::InvalidArgument;
Ratan Gupta1dc91782018-04-19 16:47:12 +053026
27ConfManager::ConfManager(sdbusplus::bus::bus& bus, const char* objPath) :
Ratan Gupta212f53e2018-04-30 17:28:05 +053028 details::CreateIface(bus, objPath, true),
29 dbusPersistentLocation(SNMP_CONF_PERSIST_PATH), bus(bus),
30 objectPath(objPath)
Patrick Williams1334b7b2021-02-22 17:15:12 -060031{}
Ratan Gupta1dc91782018-04-19 16:47:12 +053032
Ratan Guptad84e3272018-09-06 16:52:52 +053033std::string ConfManager::client(std::string address, uint16_t port)
Ratan Gupta1dc91782018-04-19 16:47:12 +053034{
Ratan Gupta9c4fed62018-11-16 17:47:54 +053035 // will throw exception if it is already configured.
36 checkClientConfigured(address, port);
Ratan Guptaa7ff3852018-11-16 14:05:57 +053037
38 lastClientId++;
Ratan Gupta213517b2018-04-28 13:41:09 +053039 try
40 {
41 // just to check whether given address is valid or not.
42 resolveAddress(address);
43 }
44 catch (InternalFailure& e)
45 {
46 log<level::ERR>("Not a valid address"),
47 entry("ADDRESS=%s", address.c_str());
48 elog<InvalidArgument>(Argument::ARGUMENT_NAME("Address"),
49 Argument::ARGUMENT_VALUE(address.c_str()));
50 }
Ratan Guptad84e3272018-09-06 16:52:52 +053051
Ratan Guptaa7ff3852018-11-16 14:05:57 +053052 // create the D-Bus object
Ratan Gupta213517b2018-04-28 13:41:09 +053053 std::experimental::filesystem::path objPath;
54 objPath /= objectPath;
Ratan Guptaa7ff3852018-11-16 14:05:57 +053055 objPath /= std::to_string(lastClientId);
56
Ratan Gupta212f53e2018-04-30 17:28:05 +053057 auto client = std::make_unique<phosphor::network::snmp::Client>(
58 bus, objPath.string().c_str(), *this, address, port);
Ratan Gupta212f53e2018-04-30 17:28:05 +053059
Ratan Guptaa7ff3852018-11-16 14:05:57 +053060 // save the D-Bus object
61 serialize(lastClientId, *client, dbusPersistentLocation);
62
63 this->clients.emplace(lastClientId, std::move(client));
Ratan Guptad84e3272018-09-06 16:52:52 +053064 return objPath.string();
Ratan Gupta1dc91782018-04-19 16:47:12 +053065}
66
Ratan Gupta9c4fed62018-11-16 17:47:54 +053067void ConfManager::checkClientConfigured(const std::string& address,
68 uint16_t port)
69{
70 if (address.empty())
71 {
72 log<level::ERR>("Invalid address");
73 elog<InvalidArgument>(Argument::ARGUMENT_NAME("ADDRESS"),
74 Argument::ARGUMENT_VALUE(address.c_str()));
75 }
76
77 for (const auto& val : clients)
78 {
79 if (val.second.get()->address() == address &&
80 val.second.get()->port() == port)
81 {
82 log<level::ERR>("Client already exist");
83 // TODO Add the error(Object already exist) in the D-Bus interface
84 // then make the change here,meanwhile send the Internal Failure.
85 elog<InvalidArgument>(
86 Argument::ARGUMENT_NAME("ADDRESS"),
87 Argument::ARGUMENT_VALUE("Client already exist."));
88 }
89 }
90}
91
Ratan Guptaa7ff3852018-11-16 14:05:57 +053092void ConfManager::deleteSNMPClient(Id id)
Ratan Gupta1dc91782018-04-19 16:47:12 +053093{
Ratan Guptaa7ff3852018-11-16 14:05:57 +053094 auto it = clients.find(id);
Ratan Gupta1dc91782018-04-19 16:47:12 +053095 if (it == clients.end())
96 {
97 log<level::ERR>("Unable to delete the snmp client.",
Ratan Guptaa7ff3852018-11-16 14:05:57 +053098 entry("ID=%d", id));
Ratan Gupta1dc91782018-04-19 16:47:12 +053099 return;
100 }
Ratan Gupta212f53e2018-04-30 17:28:05 +0530101
102 std::error_code ec;
103 // remove the persistent file
104 fs::path fileName = dbusPersistentLocation;
Ratan Guptaa7ff3852018-11-16 14:05:57 +0530105 fileName /= std::to_string(id);
Ratan Gupta212f53e2018-04-30 17:28:05 +0530106
107 if (fs::exists(fileName))
108 {
109 if (!fs::remove(fileName, ec))
110 {
111 log<level::ERR>("Unable to delete the file",
112 entry("FILE=%s", fileName.c_str()),
113 entry("ERROR=%d", ec.value()));
114 }
115 }
116 else
117 {
118 log<level::ERR>("File doesn't exist",
119 entry("FILE=%s", fileName.c_str()));
120 }
121 // remove the D-Bus Object.
Ratan Gupta1dc91782018-04-19 16:47:12 +0530122 this->clients.erase(it);
123}
124
Ratan Gupta212f53e2018-04-30 17:28:05 +0530125void ConfManager::restoreClients()
126{
127 if (!fs::exists(dbusPersistentLocation) ||
128 fs::is_empty(dbusPersistentLocation))
129 {
130 return;
131 }
132
133 for (auto& confFile :
134 fs::recursive_directory_iterator(dbusPersistentLocation))
135 {
136 if (!fs::is_regular_file(confFile))
137 {
138 continue;
139 }
140
141 auto managerID = confFile.path().filename().string();
Ratan Guptaa7ff3852018-11-16 14:05:57 +0530142 Id idNum = std::stol(managerID);
Ratan Gupta212f53e2018-04-30 17:28:05 +0530143
144 fs::path objPath = objectPath;
Ratan Guptaa7ff3852018-11-16 14:05:57 +0530145 objPath /= managerID;
Ratan Gupta212f53e2018-04-30 17:28:05 +0530146 auto manager =
147 std::make_unique<Client>(bus, objPath.string().c_str(), *this);
148 if (deserialize(confFile.path(), *manager))
149 {
150 manager->emit_object_added();
Ratan Guptaa7ff3852018-11-16 14:05:57 +0530151 this->clients.emplace(idNum, std::move(manager));
152 if (idNum > lastClientId)
153 {
154 lastClientId = idNum;
155 }
Ratan Gupta212f53e2018-04-30 17:28:05 +0530156 }
157 }
158}
159
Ratan Gupta1dc91782018-04-19 16:47:12 +0530160} // namespace snmp
161} // namespace network
162} // namespace phosphor