blob: 02c3aeae79ed0d9974112237342f1eb8370b7c18 [file] [log] [blame]
/*
* Copyright 2019 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "general_systemd.hpp"
#include "status.hpp"
#include <sdbusplus/bus.hpp>
#include <fstream>
#include <memory>
#include <string>
#include <vector>
namespace ipmi_flash
{
static constexpr auto systemdService = "org.freedesktop.systemd1";
static constexpr auto systemdRoot = "/org/freedesktop/systemd1";
static constexpr auto systemdInterface = "org.freedesktop.systemd1.Manager";
static constexpr auto jobInterface = "org.freedesktop.systemd1.Job";
bool SystemdNoFile::trigger()
{
if (job)
{
std::fprintf(stderr, "Job alreading running %s: %s\n",
triggerService.c_str(), job->c_str());
return false;
}
try
{
jobMonitor.emplace(bus,
"type='signal',"
"sender='org.freedesktop.systemd1',"
"path='/org/freedesktop/systemd1',"
"interface='org.freedesktop.systemd1.Manager',"
"member='JobRemoved',",
[&](sdbusplus::message_t& m) { this->match(m); });
auto method = bus.new_method_call(systemdService, systemdRoot,
systemdInterface, "StartUnit");
method.append(triggerService);
method.append(mode);
sdbusplus::message::object_path obj_path;
bus.call(method).read(obj_path);
job = std::move(obj_path);
std::fprintf(stderr, "Triggered %s mode %s: %s\n",
triggerService.c_str(), mode.c_str(), job->c_str());
currentStatus = ActionStatus::running;
return true;
}
catch (const std::exception& e)
{
job = std::nullopt;
jobMonitor = std::nullopt;
currentStatus = ActionStatus::failed;
std::fprintf(stderr, "Failed to trigger %s mode %s: %s\n",
triggerService.c_str(), mode.c_str(), e.what());
return false;
}
}
void SystemdNoFile::abort()
{
if (!job)
{
return;
}
// Cancel the job
auto cancel_req = bus.new_method_call(systemdService, job->c_str(),
jobInterface, "Cancel");
try
{
bus.call_noreply(cancel_req);
std::fprintf(stderr, "Canceled %s: %s\n", triggerService.c_str(),
job->c_str());
}
catch (const sdbusplus::exception_t& ex)
{
std::fprintf(stderr, "Failed to cancel job %s %s: %s\n",
triggerService.c_str(), job->c_str(), ex.what());
}
}
ActionStatus SystemdNoFile::status()
{
return currentStatus;
}
const std::string& SystemdNoFile::getMode() const
{
return mode;
}
void SystemdNoFile::match(sdbusplus::message_t& m)
{
if (!job)
{
std::fprintf(stderr, "No running job %s\n", triggerService.c_str());
return;
}
uint32_t job_id;
sdbusplus::message::object_path job_path;
std::string unit;
std::string result;
try
{
m.read(job_id, job_path, unit, result);
}
catch (const sdbusplus::exception_t& e)
{
std::fprintf(stderr, "Bad JobRemoved signal %s: %s\n",
triggerService.c_str(), e.what());
return;
}
if (*job != job_path.str)
{
return;
}
std::fprintf(stderr, "Job Finished %s %s: %s\n", triggerService.c_str(),
job->c_str(), result.c_str());
jobMonitor = std::nullopt;
job = std::nullopt;
currentStatus = result == "done" ? ActionStatus::success
: ActionStatus::failed;
if (cb)
{
cb(*this);
}
}
std::unique_ptr<TriggerableActionInterface> SystemdNoFile::CreateSystemdNoFile(
sdbusplus::bus_t&& bus, const std::string& service, const std::string& mode)
{
return std::make_unique<SystemdNoFile>(std::move(bus), service, mode);
}
std::unique_ptr<TriggerableActionInterface>
SystemdWithStatusFile::CreateSystemdWithStatusFile(
sdbusplus::bus_t&& bus, const std::string& path,
const std::string& service, const std::string& mode)
{
return std::make_unique<SystemdWithStatusFile>(std::move(bus), path,
service, mode);
}
bool SystemdWithStatusFile::trigger()
{
if (SystemdNoFile::status() != ActionStatus::running)
{
try
{
std::ofstream ofs;
ofs.open(checkPath);
ofs << "unknown";
}
catch (const std::exception& e)
{
return false;
}
}
return SystemdNoFile::trigger();
}
ActionStatus SystemdWithStatusFile::status()
{
// Assume a status based on job execution if there is no file
ActionStatus result = SystemdNoFile::status() == ActionStatus::running
? ActionStatus::running
: ActionStatus::failed;
std::ifstream ifs;
ifs.open(checkPath);
if (ifs.good())
{
/*
* Check for the contents of the file, accepting:
* running, success, or failed.
*/
std::string status;
ifs >> status;
if (status == "running")
{
result = ActionStatus::running;
}
else if (status == "success")
{
result = ActionStatus::success;
}
else if (status == "failed")
{
result = ActionStatus::failed;
}
else
{
result = ActionStatus::unknown;
}
}
return result;
}
} // namespace ipmi_flash