170141561SBorawski.Lukasz /* 270141561SBorawski.Lukasz // Copyright (c) 2018 Intel Corporation 370141561SBorawski.Lukasz // 470141561SBorawski.Lukasz // Licensed under the Apache License, Version 2.0 (the "License"); 570141561SBorawski.Lukasz // you may not use this file except in compliance with the License. 670141561SBorawski.Lukasz // You may obtain a copy of the License at 770141561SBorawski.Lukasz // 870141561SBorawski.Lukasz // http://www.apache.org/licenses/LICENSE-2.0 970141561SBorawski.Lukasz // 1070141561SBorawski.Lukasz // Unless required by applicable law or agreed to in writing, software 1170141561SBorawski.Lukasz // distributed under the License is distributed on an "AS IS" BASIS, 1270141561SBorawski.Lukasz // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 1370141561SBorawski.Lukasz // See the License for the specific language governing permissions and 1470141561SBorawski.Lukasz // limitations under the License. 1570141561SBorawski.Lukasz */ 1670141561SBorawski.Lukasz #pragma once 1770141561SBorawski.Lukasz 183a8a0088SKowalski, Kamil #include "error_messages.hpp" 1970141561SBorawski.Lukasz #include "node.hpp" 2070141561SBorawski.Lukasz 2120e6ea5dSraviteja-b #include <optional> 2220e6ea5dSraviteja-b #include <utils/json_utils.hpp> 23abf2add6SEd Tanous #include <variant> 241abe55efSEd Tanous namespace redfish 251abe55efSEd Tanous { 2670141561SBorawski.Lukasz 271abe55efSEd Tanous enum NetworkProtocolUnitStructFields 281abe55efSEd Tanous { 293a8a0088SKowalski, Kamil NET_PROTO_UNIT_NAME, 303a8a0088SKowalski, Kamil NET_PROTO_UNIT_DESC, 313a8a0088SKowalski, Kamil NET_PROTO_UNIT_LOAD_STATE, 323a8a0088SKowalski, Kamil NET_PROTO_UNIT_ACTIVE_STATE, 333a8a0088SKowalski, Kamil NET_PROTO_UNIT_SUB_STATE, 343a8a0088SKowalski, Kamil NET_PROTO_UNIT_DEVICE, 353a8a0088SKowalski, Kamil NET_PROTO_UNIT_OBJ_PATH, 363a8a0088SKowalski, Kamil NET_PROTO_UNIT_ALWAYS_0, 373a8a0088SKowalski, Kamil NET_PROTO_UNIT_ALWAYS_EMPTY, 383a8a0088SKowalski, Kamil NET_PROTO_UNIT_ALWAYS_ROOT_PATH 393a8a0088SKowalski, Kamil }; 403a8a0088SKowalski, Kamil 411abe55efSEd Tanous enum NetworkProtocolListenResponseElements 421abe55efSEd Tanous { 433a8a0088SKowalski, Kamil NET_PROTO_LISTEN_TYPE, 443a8a0088SKowalski, Kamil NET_PROTO_LISTEN_STREAM 453a8a0088SKowalski, Kamil }; 463a8a0088SKowalski, Kamil 473a8a0088SKowalski, Kamil /** 483a8a0088SKowalski, Kamil * @brief D-Bus Unit structure returned in array from ListUnits Method 493a8a0088SKowalski, Kamil */ 503a8a0088SKowalski, Kamil using UnitStruct = 513a8a0088SKowalski, Kamil std::tuple<std::string, std::string, std::string, std::string, std::string, 523a8a0088SKowalski, Kamil std::string, sdbusplus::message::object_path, uint32_t, 533a8a0088SKowalski, Kamil std::string, sdbusplus::message::object_path>; 543a8a0088SKowalski, Kamil 55ec4974ddSAppaRao Puli const static boost::container::flat_map<const char*, std::string> 56ec4974ddSAppaRao Puli protocolToDBus{{"SSH", "dropbear"}, 57ec4974ddSAppaRao Puli {"HTTPS", "bmcweb"}, 58ec4974ddSAppaRao Puli {"IPMI", "phosphor-ipmi-net"}}; 593a8a0088SKowalski, Kamil 60d24bfc7aSJennifer Lee inline void 61d24bfc7aSJennifer Lee extractNTPServersAndDomainNamesData(const GetManagedObjects& dbus_data, 62d24bfc7aSJennifer Lee std::vector<std::string>& ntpData, 63d24bfc7aSJennifer Lee std::vector<std::string>& dnData) 6420e6ea5dSraviteja-b { 6520e6ea5dSraviteja-b for (const auto& obj : dbus_data) 6620e6ea5dSraviteja-b { 6720e6ea5dSraviteja-b for (const auto& ifacePair : obj.second) 6820e6ea5dSraviteja-b { 6920e6ea5dSraviteja-b if (obj.first == "/xyz/openbmc_project/network/eth0") 7020e6ea5dSraviteja-b { 7120e6ea5dSraviteja-b if (ifacePair.first == 7220e6ea5dSraviteja-b "xyz.openbmc_project.Network.EthernetInterface") 7320e6ea5dSraviteja-b { 7420e6ea5dSraviteja-b for (const auto& propertyPair : ifacePair.second) 7520e6ea5dSraviteja-b { 7620e6ea5dSraviteja-b if (propertyPair.first == "NTPServers") 7720e6ea5dSraviteja-b { 7820e6ea5dSraviteja-b const std::vector<std::string>* ntpServers = 7920e6ea5dSraviteja-b sdbusplus::message::variant_ns::get_if< 8020e6ea5dSraviteja-b std::vector<std::string>>( 8120e6ea5dSraviteja-b &propertyPair.second); 8220e6ea5dSraviteja-b if (ntpServers != nullptr) 8320e6ea5dSraviteja-b { 8420e6ea5dSraviteja-b ntpData = std::move(*ntpServers); 8520e6ea5dSraviteja-b } 8620e6ea5dSraviteja-b } 87d24bfc7aSJennifer Lee else if (propertyPair.first == "DomainName") 88d24bfc7aSJennifer Lee { 89d24bfc7aSJennifer Lee const std::vector<std::string>* domainNames = 90d24bfc7aSJennifer Lee sdbusplus::message::variant_ns::get_if< 91d24bfc7aSJennifer Lee std::vector<std::string>>( 92d24bfc7aSJennifer Lee &propertyPair.second); 93d24bfc7aSJennifer Lee if (domainNames != nullptr) 94d24bfc7aSJennifer Lee { 95d24bfc7aSJennifer Lee dnData = std::move(*domainNames); 96d24bfc7aSJennifer Lee } 97d24bfc7aSJennifer Lee } 9820e6ea5dSraviteja-b } 9920e6ea5dSraviteja-b } 10020e6ea5dSraviteja-b } 10120e6ea5dSraviteja-b } 10220e6ea5dSraviteja-b } 10320e6ea5dSraviteja-b } 10420e6ea5dSraviteja-b 10520e6ea5dSraviteja-b template <typename CallbackFunc> 10620e6ea5dSraviteja-b void getEthernetIfaceData(CallbackFunc&& callback) 10720e6ea5dSraviteja-b { 10820e6ea5dSraviteja-b crow::connections::systemBus->async_method_call( 10920e6ea5dSraviteja-b [callback{std::move(callback)}]( 11020e6ea5dSraviteja-b const boost::system::error_code error_code, 11120e6ea5dSraviteja-b const GetManagedObjects& dbus_data) { 11220e6ea5dSraviteja-b std::vector<std::string> ntpServers; 113d24bfc7aSJennifer Lee std::vector<std::string> domainNames; 11420e6ea5dSraviteja-b 11520e6ea5dSraviteja-b if (error_code) 11620e6ea5dSraviteja-b { 117d24bfc7aSJennifer Lee callback(false, ntpServers, domainNames); 11820e6ea5dSraviteja-b return; 11920e6ea5dSraviteja-b } 12020e6ea5dSraviteja-b 121d24bfc7aSJennifer Lee extractNTPServersAndDomainNamesData(dbus_data, ntpServers, 122d24bfc7aSJennifer Lee domainNames); 12320e6ea5dSraviteja-b 124d24bfc7aSJennifer Lee callback(true, ntpServers, domainNames); 12520e6ea5dSraviteja-b }, 12620e6ea5dSraviteja-b "xyz.openbmc_project.Network", "/xyz/openbmc_project/network", 12720e6ea5dSraviteja-b "org.freedesktop.DBus.ObjectManager", "GetManagedObjects"); 128271584abSEd Tanous } 12920e6ea5dSraviteja-b 1301abe55efSEd Tanous class NetworkProtocol : public Node 1311abe55efSEd Tanous { 13270141561SBorawski.Lukasz public: 1331abe55efSEd Tanous NetworkProtocol(CrowApp& app) : 13497b7ccf3SEd Tanous Node(app, "/redfish/v1/Managers/bmc/NetworkProtocol") 1351abe55efSEd Tanous { 1364b1b8683SBorawski.Lukasz entityPrivileges = { 1374b1b8683SBorawski.Lukasz {boost::beast::http::verb::get, {{"Login"}}}, 138e0d918bcSEd Tanous {boost::beast::http::verb::head, {{"Login"}}}, 139e0d918bcSEd Tanous {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 140e0d918bcSEd Tanous {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 141e0d918bcSEd Tanous {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 142e0d918bcSEd Tanous {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 14370141561SBorawski.Lukasz } 14470141561SBorawski.Lukasz 14570141561SBorawski.Lukasz private: 14655c7b7a2SEd Tanous void doGet(crow::Response& res, const crow::Request& req, 1471abe55efSEd Tanous const std::vector<std::string>& params) override 1481abe55efSEd Tanous { 1493a8a0088SKowalski, Kamil std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res); 1503a8a0088SKowalski, Kamil 1513a8a0088SKowalski, Kamil getData(asyncResp); 15270141561SBorawski.Lukasz } 15370141561SBorawski.Lukasz 1541abe55efSEd Tanous std::string getHostName() const 1551abe55efSEd Tanous { 15670141561SBorawski.Lukasz std::string hostName; 15770141561SBorawski.Lukasz 15870141561SBorawski.Lukasz std::array<char, HOST_NAME_MAX> hostNameCStr; 1591abe55efSEd Tanous if (gethostname(hostNameCStr.data(), hostNameCStr.size()) == 0) 1601abe55efSEd Tanous { 16170141561SBorawski.Lukasz hostName = hostNameCStr.data(); 16270141561SBorawski.Lukasz } 16370141561SBorawski.Lukasz return hostName; 16470141561SBorawski.Lukasz } 16570141561SBorawski.Lukasz 16620e6ea5dSraviteja-b void getNTPProtocolEnabled(const std::shared_ptr<AsyncResp>& asyncResp) 16720e6ea5dSraviteja-b { 16820e6ea5dSraviteja-b crow::connections::systemBus->async_method_call( 16920e6ea5dSraviteja-b [asyncResp](const boost::system::error_code error_code, 17020e6ea5dSraviteja-b const std::variant<std::string>& timeSyncMethod) { 17120e6ea5dSraviteja-b const std::string* s = 17220e6ea5dSraviteja-b std::get_if<std::string>(&timeSyncMethod); 17320e6ea5dSraviteja-b 17420e6ea5dSraviteja-b if (*s == "xyz.openbmc_project.Time.Synchronization.Method.NTP") 17520e6ea5dSraviteja-b { 17620e6ea5dSraviteja-b asyncResp->res.jsonValue["NTP"]["ProtocolEnabled"] = true; 17720e6ea5dSraviteja-b } 17820e6ea5dSraviteja-b else if (*s == "xyz.openbmc_project.Time.Synchronization." 17920e6ea5dSraviteja-b "Method.Manual") 18020e6ea5dSraviteja-b { 18120e6ea5dSraviteja-b asyncResp->res.jsonValue["NTP"]["ProtocolEnabled"] = false; 18220e6ea5dSraviteja-b } 18320e6ea5dSraviteja-b }, 18420e6ea5dSraviteja-b "xyz.openbmc_project.Settings", 18520e6ea5dSraviteja-b "/xyz/openbmc_project/time/sync_method", 18620e6ea5dSraviteja-b "org.freedesktop.DBus.Properties", "Get", 18720e6ea5dSraviteja-b "xyz.openbmc_project.Time.Synchronization", "TimeSyncMethod"); 18820e6ea5dSraviteja-b } 18920e6ea5dSraviteja-b 1901abe55efSEd Tanous void getData(const std::shared_ptr<AsyncResp>& asyncResp) 1911abe55efSEd Tanous { 1920f74e643SEd Tanous asyncResp->res.jsonValue["@odata.type"] = 1934ed69245SGunnar Mills "#ManagerNetworkProtocol.v1_4_0.ManagerNetworkProtocol"; 1940f74e643SEd Tanous asyncResp->res.jsonValue["@odata.id"] = 1950f74e643SEd Tanous "/redfish/v1/Managers/bmc/NetworkProtocol"; 1960f74e643SEd Tanous asyncResp->res.jsonValue["Id"] = "NetworkProtocol"; 1970f74e643SEd Tanous asyncResp->res.jsonValue["Name"] = "Manager Network Protocol"; 1980f74e643SEd Tanous asyncResp->res.jsonValue["Description"] = "Manager Network Service"; 1990f74e643SEd Tanous asyncResp->res.jsonValue["Status"]["Health"] = "OK"; 2000f74e643SEd Tanous asyncResp->res.jsonValue["Status"]["HealthRollup"] = "OK"; 2010f74e643SEd Tanous asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 2020f74e643SEd Tanous 203*818ea7b8SJoshi-Mansi // HTTP is Mandatory attribute as per OCP Baseline Profile � v1.0.0, 204*818ea7b8SJoshi-Mansi // but from security perspective it is not recommended to use. 205*818ea7b8SJoshi-Mansi // Hence using protocolEnabled as false to make it OCP and security-wise 206*818ea7b8SJoshi-Mansi // compliant 207*818ea7b8SJoshi-Mansi asyncResp->res.jsonValue["HTTP"]["Port"] = 0; 208*818ea7b8SJoshi-Mansi asyncResp->res.jsonValue["HTTP"]["ProtocolEnabled"] = false; 209*818ea7b8SJoshi-Mansi 2100f74e643SEd Tanous for (auto& protocol : protocolToDBus) 2110f74e643SEd Tanous { 2120f74e643SEd Tanous asyncResp->res.jsonValue[protocol.first]["ProtocolEnabled"] = false; 2130f74e643SEd Tanous } 2140f74e643SEd Tanous 215d24bfc7aSJennifer Lee std::string hostName = getHostName(); 216d24bfc7aSJennifer Lee 217d24bfc7aSJennifer Lee asyncResp->res.jsonValue["HostName"] = hostName; 2183a8a0088SKowalski, Kamil 21920e6ea5dSraviteja-b getNTPProtocolEnabled(asyncResp); 22020e6ea5dSraviteja-b 22120e6ea5dSraviteja-b // TODO Get eth0 interface data, and call the below callback for JSON 22220e6ea5dSraviteja-b // preparation 223271584abSEd Tanous getEthernetIfaceData( 224271584abSEd Tanous [hostName, asyncResp](const bool& success, 225d24bfc7aSJennifer Lee const std::vector<std::string>& ntpServers, 226d24bfc7aSJennifer Lee const std::vector<std::string>& domainNames) { 22720e6ea5dSraviteja-b if (!success) 22820e6ea5dSraviteja-b { 229271584abSEd Tanous messages::resourceNotFound(asyncResp->res, 230271584abSEd Tanous "EthernetInterface", "eth0"); 23120e6ea5dSraviteja-b return; 23220e6ea5dSraviteja-b } 23320e6ea5dSraviteja-b asyncResp->res.jsonValue["NTP"]["NTPServers"] = ntpServers; 234d24bfc7aSJennifer Lee if (hostName.empty() == false) 235d24bfc7aSJennifer Lee { 236749dad7dSRatan Gupta std::string FQDN = std::move(hostName); 237d24bfc7aSJennifer Lee if (domainNames.empty() == false) 238d24bfc7aSJennifer Lee { 239749dad7dSRatan Gupta FQDN += "." + domainNames[0]; 240d24bfc7aSJennifer Lee } 241749dad7dSRatan Gupta asyncResp->res.jsonValue["FQDN"] = std::move(FQDN); 242d24bfc7aSJennifer Lee } 24320e6ea5dSraviteja-b }); 24420e6ea5dSraviteja-b 245865fbb75SEd Tanous crow::connections::systemBus->async_method_call( 246271584abSEd Tanous [asyncResp](const boost::system::error_code e, 247271584abSEd Tanous const std::vector<UnitStruct>& r) { 248271584abSEd Tanous if (e) 2491abe55efSEd Tanous { 2503a8a0088SKowalski, Kamil asyncResp->res.jsonValue = nlohmann::json::object(); 251f12894f8SJason M. Bills messages::internalError(asyncResp->res); 252865fbb75SEd Tanous return; 2533a8a0088SKowalski, Kamil } 2545968caeeSMarri Devender Rao asyncResp->res.jsonValue["HTTPS"]["Certificates"] = { 2555968caeeSMarri Devender Rao {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol/" 256659dd62eSJason M. Bills "HTTPS/Certificates"}}; 2573a8a0088SKowalski, Kamil 258271584abSEd Tanous for (auto& unit : r) 2591abe55efSEd Tanous { 260ec4974ddSAppaRao Puli /* Only traverse through <xyz>.socket units */ 261ec4974ddSAppaRao Puli std::string unitName = std::get<NET_PROTO_UNIT_NAME>(unit); 262ec4974ddSAppaRao Puli if (!boost::ends_with(unitName, ".socket")) 2631abe55efSEd Tanous { 264865fbb75SEd Tanous continue; 26570141561SBorawski.Lukasz } 26670141561SBorawski.Lukasz 267ec4974ddSAppaRao Puli for (auto& kv : protocolToDBus) 268ec4974ddSAppaRao Puli { 269ec4974ddSAppaRao Puli // We are interested in services, which starts with 270ec4974ddSAppaRao Puli // mapped service name 271ec4974ddSAppaRao Puli if (!boost::starts_with(unitName, kv.second)) 272ec4974ddSAppaRao Puli { 273ec4974ddSAppaRao Puli continue; 274ec4974ddSAppaRao Puli } 275ec4974ddSAppaRao Puli const char* rfServiceKey = kv.first; 276ec4974ddSAppaRao Puli std::string socketPath = 277ec4974ddSAppaRao Puli std::get<NET_PROTO_UNIT_OBJ_PATH>(unit); 278ec4974ddSAppaRao Puli std::string unitState = 279ec4974ddSAppaRao Puli std::get<NET_PROTO_UNIT_SUB_STATE>(unit); 280ec4974ddSAppaRao Puli 281ec4974ddSAppaRao Puli asyncResp->res 282ec4974ddSAppaRao Puli .jsonValue[rfServiceKey]["ProtocolEnabled"] = 283ec4974ddSAppaRao Puli (unitState == "running") || 284ec4974ddSAppaRao Puli (unitState == "listening"); 285ec4974ddSAppaRao Puli 286865fbb75SEd Tanous crow::connections::systemBus->async_method_call( 287ec4974ddSAppaRao Puli [asyncResp, 288ec4974ddSAppaRao Puli rfServiceKey{std::string(rfServiceKey)}]( 289865fbb75SEd Tanous const boost::system::error_code ec, 290abf2add6SEd Tanous const std::variant<std::vector<std::tuple< 291abf2add6SEd Tanous std::string, std::string>>>& resp) { 2921abe55efSEd Tanous if (ec) 2931abe55efSEd Tanous { 294a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 295865fbb75SEd Tanous return; 2963a8a0088SKowalski, Kamil } 297abf2add6SEd Tanous const std::vector< 298abf2add6SEd Tanous std::tuple<std::string, std::string>>* 299abf2add6SEd Tanous responsePtr = std::get_if<std::vector< 300abf2add6SEd Tanous std::tuple<std::string, std::string>>>( 3011b6b96c5SEd Tanous &resp); 3021abe55efSEd Tanous if (responsePtr == nullptr || 3031abe55efSEd Tanous responsePtr->size() < 1) 3041abe55efSEd Tanous { 305865fbb75SEd Tanous return; 30670141561SBorawski.Lukasz } 30770141561SBorawski.Lukasz 308865fbb75SEd Tanous const std::string& listenStream = 3091abe55efSEd Tanous std::get<NET_PROTO_LISTEN_STREAM>( 3101abe55efSEd Tanous (*responsePtr)[0]); 3111abe55efSEd Tanous std::size_t lastColonPos = 3121abe55efSEd Tanous listenStream.rfind(":"); 3131abe55efSEd Tanous if (lastColonPos == std::string::npos) 3141abe55efSEd Tanous { 315865fbb75SEd Tanous // Not a port 316865fbb75SEd Tanous return; 317865fbb75SEd Tanous } 3181abe55efSEd Tanous std::string portStr = 3191abe55efSEd Tanous listenStream.substr(lastColonPos + 1); 320ec4974ddSAppaRao Puli if (portStr.empty()) 321ec4974ddSAppaRao Puli { 322ec4974ddSAppaRao Puli return; 323ec4974ddSAppaRao Puli } 324865fbb75SEd Tanous char* endPtr = nullptr; 325ec4974ddSAppaRao Puli errno = 0; 3261abe55efSEd Tanous // Use strtol instead of stroi to avoid 3271abe55efSEd Tanous // exceptions 3281abe55efSEd Tanous long port = 3291abe55efSEd Tanous std::strtol(portStr.c_str(), &endPtr, 10); 330ec4974ddSAppaRao Puli if ((errno == 0) && (*endPtr == '\0')) 3311abe55efSEd Tanous { 332ec4974ddSAppaRao Puli asyncResp->res 333ec4974ddSAppaRao Puli .jsonValue[rfServiceKey]["Port"] = port; 3341abe55efSEd Tanous } 335ec4974ddSAppaRao Puli return; 336865fbb75SEd Tanous }, 337865fbb75SEd Tanous "org.freedesktop.systemd1", socketPath, 338865fbb75SEd Tanous "org.freedesktop.DBus.Properties", "Get", 339865fbb75SEd Tanous "org.freedesktop.systemd1.Socket", "Listen"); 340ec4974ddSAppaRao Puli 341ec4974ddSAppaRao Puli // We found service, break the inner loop. 342ec4974ddSAppaRao Puli break; 343865fbb75SEd Tanous } 344865fbb75SEd Tanous } 345865fbb75SEd Tanous }, 346865fbb75SEd Tanous "org.freedesktop.systemd1", "/org/freedesktop/systemd1", 347865fbb75SEd Tanous "org.freedesktop.systemd1.Manager", "ListUnits"); 348865fbb75SEd Tanous } 349501be32bSraviteja-b 350501be32bSraviteja-b void handleHostnamePatch(const std::string& hostName, 351501be32bSraviteja-b const std::shared_ptr<AsyncResp>& asyncResp) 352501be32bSraviteja-b { 353501be32bSraviteja-b crow::connections::systemBus->async_method_call( 354501be32bSraviteja-b [asyncResp](const boost::system::error_code ec) { 355501be32bSraviteja-b if (ec) 356501be32bSraviteja-b { 357501be32bSraviteja-b messages::internalError(asyncResp->res); 358501be32bSraviteja-b return; 359501be32bSraviteja-b } 360501be32bSraviteja-b }, 361501be32bSraviteja-b "xyz.openbmc_project.Network", 362501be32bSraviteja-b "/xyz/openbmc_project/network/config", 363501be32bSraviteja-b "org.freedesktop.DBus.Properties", "Set", 364501be32bSraviteja-b "xyz.openbmc_project.Network.SystemConfiguration", "HostName", 365501be32bSraviteja-b std::variant<std::string>(hostName)); 366501be32bSraviteja-b } 367501be32bSraviteja-b 36820e6ea5dSraviteja-b void handleNTPProtocolEnabled(const bool& ntpEnabled, 36920e6ea5dSraviteja-b const std::shared_ptr<AsyncResp>& asyncResp) 37020e6ea5dSraviteja-b { 37120e6ea5dSraviteja-b std::string timeSyncMethod; 37220e6ea5dSraviteja-b if (ntpEnabled) 37320e6ea5dSraviteja-b { 37420e6ea5dSraviteja-b timeSyncMethod = 37520e6ea5dSraviteja-b "xyz.openbmc_project.Time.Synchronization.Method.NTP"; 37620e6ea5dSraviteja-b } 37720e6ea5dSraviteja-b else 37820e6ea5dSraviteja-b { 37920e6ea5dSraviteja-b timeSyncMethod = 38020e6ea5dSraviteja-b "xyz.openbmc_project.Time.Synchronization.Method.Manual"; 38120e6ea5dSraviteja-b } 38220e6ea5dSraviteja-b 38320e6ea5dSraviteja-b crow::connections::systemBus->async_method_call( 384cf05f9dcSJohnathan Mantey [asyncResp](const boost::system::error_code error_code) {}, 38520e6ea5dSraviteja-b "xyz.openbmc_project.Settings", 38620e6ea5dSraviteja-b "/xyz/openbmc_project/time/sync_method", 38720e6ea5dSraviteja-b "org.freedesktop.DBus.Properties", "Set", 38820e6ea5dSraviteja-b "xyz.openbmc_project.Time.Synchronization", "TimeSyncMethod", 38920e6ea5dSraviteja-b std::variant<std::string>{timeSyncMethod}); 39020e6ea5dSraviteja-b } 39120e6ea5dSraviteja-b 39220e6ea5dSraviteja-b void handleNTPServersPatch(const std::vector<std::string>& ntpServers, 39320e6ea5dSraviteja-b const std::shared_ptr<AsyncResp>& asyncResp) 39420e6ea5dSraviteja-b { 39520e6ea5dSraviteja-b crow::connections::systemBus->async_method_call( 396cf05f9dcSJohnathan Mantey [asyncResp](const boost::system::error_code ec) { 39720e6ea5dSraviteja-b if (ec) 39820e6ea5dSraviteja-b { 39920e6ea5dSraviteja-b messages::internalError(asyncResp->res); 40020e6ea5dSraviteja-b return; 40120e6ea5dSraviteja-b } 40220e6ea5dSraviteja-b }, 40320e6ea5dSraviteja-b "xyz.openbmc_project.Network", "/xyz/openbmc_project/network/eth0", 40420e6ea5dSraviteja-b "org.freedesktop.DBus.Properties", "Set", 40520e6ea5dSraviteja-b "xyz.openbmc_project.Network.EthernetInterface", "NTPServers", 40620e6ea5dSraviteja-b std::variant<std::vector<std::string>>{ntpServers}); 40720e6ea5dSraviteja-b } 40820e6ea5dSraviteja-b 409501be32bSraviteja-b void doPatch(crow::Response& res, const crow::Request& req, 410501be32bSraviteja-b const std::vector<std::string>& params) override 411501be32bSraviteja-b { 412501be32bSraviteja-b std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res); 413501be32bSraviteja-b std::optional<std::string> newHostName; 414cf05f9dcSJohnathan Mantey std::optional<nlohmann::json> ntp; 415501be32bSraviteja-b 416cf05f9dcSJohnathan Mantey if (!json_util::readJson(req, res, "HostName", newHostName, "NTP", ntp)) 417501be32bSraviteja-b { 418501be32bSraviteja-b return; 419501be32bSraviteja-b } 420cf05f9dcSJohnathan Mantey 421cf05f9dcSJohnathan Mantey res.result(boost::beast::http::status::no_content); 422501be32bSraviteja-b if (newHostName) 423501be32bSraviteja-b { 424501be32bSraviteja-b handleHostnamePatch(*newHostName, asyncResp); 425cf05f9dcSJohnathan Mantey } 426cf05f9dcSJohnathan Mantey 427cf05f9dcSJohnathan Mantey if (ntp) 428cf05f9dcSJohnathan Mantey { 429cf05f9dcSJohnathan Mantey std::optional<std::vector<std::string>> ntpServers; 430cf05f9dcSJohnathan Mantey std::optional<bool> ntpEnabled; 431cf05f9dcSJohnathan Mantey if (!json_util::readJson(*ntp, res, "NTPServers", ntpServers, 432cf05f9dcSJohnathan Mantey "ProtocolEnabled", ntpEnabled)) 433cf05f9dcSJohnathan Mantey { 434501be32bSraviteja-b return; 435501be32bSraviteja-b } 436cf05f9dcSJohnathan Mantey 43720e6ea5dSraviteja-b if (ntpEnabled) 43820e6ea5dSraviteja-b { 43920e6ea5dSraviteja-b handleNTPProtocolEnabled(*ntpEnabled, asyncResp); 44020e6ea5dSraviteja-b } 441cf05f9dcSJohnathan Mantey 44220e6ea5dSraviteja-b if (ntpServers) 44320e6ea5dSraviteja-b { 44420e6ea5dSraviteja-b handleNTPServersPatch(*ntpServers, asyncResp); 44520e6ea5dSraviteja-b } 446501be32bSraviteja-b } 447cf05f9dcSJohnathan Mantey } 44870141561SBorawski.Lukasz }; 44970141561SBorawski.Lukasz 45070141561SBorawski.Lukasz } // namespace redfish 451