dde-application-manager/apps/app-launch-helper/src/main.cpp

390 lines
12 KiB
C++
Raw Normal View History

// SPDX-FileCopyrightText: 2023 UnionTech Software Technology Co., Ltd.
//
// SPDX-License-Identifier: LGPL-3.0-or-later
#include <numeric>
#include <systemd/sd-bus.h>
#include <systemd/sd-journal.h>
#include <string_view>
#include <vector>
#include <deque>
#include <memory>
#include <iostream>
#include <string>
#include <algorithm>
#include <cstdlib>
#include <map>
#include <thread>
enum class ExitCode { SystemdError = -3, InvalidInput = -2, InternalError = -1, Done = 0, Waiting = 1 };
struct JobRemoveResult
{
std::string_view id;
int removedFlag{0};
ExitCode result{ExitCode::Waiting};
};
constexpr static auto SystemdService = "org.freedesktop.systemd1";
constexpr static auto SystemdObjectPath = "/org/freedesktop/systemd1";
constexpr static auto SystemdInterfaceName = "org.freedesktop.systemd1.Manager";
using msg_ptr = sd_bus_message *;
using bus_ptr = sd_bus *;
static ExitCode fromString(const std::string &str)
{
if (str == "done") {
return ExitCode::Done;
} else if (str == "canceled" or str == "timeout" or str == "failed" or str == "dependency" or str == "skipped") {
return ExitCode::SystemdError;
} else if (str == "internalError") {
return ExitCode::InternalError;
} else if (str == "invalidInput") {
return ExitCode::InvalidInput;
}
__builtin_unreachable();
}
static ExitCode fromString(const char *str)
{
if (!str) {
return ExitCode::Waiting;
}
std::string tmp{str};
return fromString(tmp);
}
[[noreturn]] static void releaseRes(sd_bus_error &error, msg_ptr &msg, bus_ptr &bus, ExitCode ret)
{
sd_bus_error_free(&error);
sd_bus_message_unref(msg);
sd_bus_unref(bus);
std::exit(static_cast<int>(ret));
}
static int processExecStart(msg_ptr &msg, const std::deque<std::string_view> &execArgs)
{
int ret;
if (ret = sd_bus_message_append(msg, "s", "ExecStart"); ret < 0) {
sd_journal_perror("append ExecStart failed.");
return ret;
}
if (ret = sd_bus_message_open_container(msg, SD_BUS_TYPE_VARIANT, "a(sasb)"); ret < 0) {
sd_journal_perror("open variant of execStart failed.");
if (auto tmp = sd_bus_message_close_container(msg))
return ret;
}
if (ret = sd_bus_message_open_container(msg, SD_BUS_TYPE_ARRAY, "(sasb)"); ret < 0) {
sd_journal_perror("open array of execStart failed.");
return ret;
}
if (ret = sd_bus_message_open_container(msg, SD_BUS_TYPE_STRUCT, "sasb"); ret < 0) {
sd_journal_perror("open struct of execStart failed.");
return ret;
}
if (ret = sd_bus_message_append(msg, "s", execArgs[0].data()); ret < 0) {
sd_journal_perror("append binary of execStart failed.");
return ret;
}
if (ret = sd_bus_message_open_container(msg, SD_BUS_TYPE_ARRAY, "s"); ret < 0) {
sd_journal_perror("open array of execStart variant failed.");
return ret;
}
for (std::size_t i = 0; i < execArgs.size(); ++i) {
if (ret = sd_bus_message_append(msg, "s", execArgs[i].data()); ret < 0) {
sd_journal_perror("append args of execStart failed.");
return ret;
}
}
if (ret = sd_bus_message_close_container(msg); ret < 0) {
sd_journal_perror("close array of execStart failed.");
return ret;
}
if (ret = sd_bus_message_append(msg, "b", 0);
ret < 0) { // this value indicate that systemd should be considered a failure if the process exits uncleanly
sd_journal_perror("append boolean of execStart failed.");
return ret;
}
if (ret = sd_bus_message_close_container(msg); ret < 0) {
sd_journal_perror("close struct of execStart failed.");
return ret;
}
if (ret = sd_bus_message_close_container(msg); ret < 0) {
sd_journal_perror("close array of execStart failed.");
return ret;
}
if (ret = sd_bus_message_close_container(msg); ret < 0) {
sd_journal_perror("close variant of execStart failed.");
return ret;
}
return 0;
}
static int processKVPair(msg_ptr &msg, const std::map<std::string_view, std::string_view> &props)
{
int ret;
if (!props.empty()) {
for (auto [key, value] : props) {
// NOTE: append if necessary
// usage:
// sd_bus_message_append(msg,"s",key.data());
// sd_bus_message_open_container(msg,SD_BUS_TYPE_VARIANT,"[corresponding type]");
// append content....
// sd_bus_message_close_container(msg);
}
}
return 0;
}
static std::string cmdParse(msg_ptr &msg, std::deque<std::string_view> &&cmdLines)
{
std::string serviceName{"internalError"};
std::map<std::string_view, std::string_view> props;
while (!cmdLines.empty()) { // NOTE: avoid stl exception
auto str = cmdLines.front();
if (str.size() < 2) {
sd_journal_print(LOG_WARNING, "invalid option %s.", str.data());
cmdLines.pop_front();
continue;
}
if (str.substr(0, 2) != "--") {
sd_journal_print(LOG_INFO, "unknown option %s.", str.data());
cmdLines.pop_front();
continue;
}
auto kvStr = str.substr(2);
if (!kvStr.empty()) [[likely]] {
auto it = kvStr.cbegin();
if (it = std::find(it, kvStr.cend(), '='); it == kvStr.cend()) {
sd_journal_print(LOG_WARNING, "invalid k-v pair: %s", kvStr.data());
cmdLines.pop_front();
continue;
}
auto splitIndex = std::distance(kvStr.cbegin(), it);
if (++it == kvStr.cend()) {
sd_journal_print(LOG_WARNING, "invalid k-v pair: %s", kvStr.data());
cmdLines.pop_front();
continue;
}
auto key = kvStr.substr(0, splitIndex);
if (key == "Type") {
// NOTE:
// Systemd service type must be "exec",
// this should not be configured in command line arguments.
cmdLines.pop_front();
continue;
}
props[key] = kvStr.substr(splitIndex + 1);
cmdLines.pop_front();
continue;
}
cmdLines.pop_front(); // NOTE: skip "--"
break;
}
// Processing of the binary file and its parameters that am want to launch
auto &execArgs = cmdLines;
if (execArgs.empty()) {
sd_journal_print(LOG_ERR, "param exec is empty.");
serviceName = "invalidInput";
return serviceName;
}
int ret;
if (props.find("unitName") == props.cend()) {
sd_journal_perror("unitName doesn't exists.");
serviceName = "invalidInput";
return serviceName;
}
if (ret = sd_bus_message_append(msg, "s", props["unitName"].data()); ret < 0) { // unitName
sd_journal_perror("append unitName failed.");
return serviceName;
} else {
serviceName = props["unitName"];
props.erase("unitName");
}
if (ret = sd_bus_message_append(msg, "s", "replace"); ret < 0) { // start mode
sd_journal_perror("append startMode failed.");
return serviceName;
}
// process properties: a(sv)
if (ret = sd_bus_message_open_container(msg, SD_BUS_TYPE_ARRAY, "(sv)"); ret < 0) {
sd_journal_perror("open array failed.");
return serviceName;
}
if (ret = sd_bus_message_append(msg, "(sv)", "Type", "s", "exec"); ret < 0) {
sd_journal_perror("append type failed.");
return serviceName;
}
if (ret = sd_bus_message_open_container(msg, SD_BUS_TYPE_STRUCT, "sv"); ret < 0) {
sd_journal_perror("open struct failed.");
return serviceName;
}
if (ret = processKVPair(msg, props); ret < 0) { // process props
serviceName = "invalidInput";
return serviceName;
}
if (ret = processExecStart(msg, execArgs); ret < 0) {
serviceName = "invalidInput";
return serviceName;
}
if (ret = sd_bus_message_close_container(msg); ret < 0) {
sd_journal_perror("close struct failed.");
return serviceName;
}
if (ret = sd_bus_message_close_container(msg); ret < 0) {
sd_journal_perror("close array failed.");
return serviceName;
}
// append aux, it's unused for now
if (ret = sd_bus_message_append(msg, "a(sa(sv))", 0); ret < 0) {
sd_journal_perror("append aux failed.");
return serviceName;
}
return serviceName;
}
int jobRemovedReceiver(sd_bus_message *m, void *userdata, sd_bus_error *ret_error)
{
int ret;
if (ret = sd_bus_error_is_set(ret_error); ret != 0) {
sd_journal_print(LOG_ERR, "JobRemoved error: [%s,%s]", ret_error->name, ret_error->message);
} else {
const char *serviceId{nullptr}, *jobResult{nullptr};
if (ret = sd_bus_message_read(m, "uoss", nullptr, nullptr, &serviceId, &jobResult); ret < 0) {
sd_journal_perror("read from JobRemoved failed.");
} else {
auto ptr = reinterpret_cast<JobRemoveResult *>(userdata);
if (ptr->id == serviceId) {
ptr->removedFlag = 1;
ptr->result = fromString(jobResult);
}
}
}
if (ret_error and ret_error->_need_free) {
sd_bus_error_free(ret_error);
}
return ret;
}
static int process_dbus_message(sd_bus *bus)
{
int ret;
ret = sd_bus_process(bus, nullptr);
if (ret < 0) {
sd_journal_print(LOG_ERR, "event loop error.");
return ret;
}
if (ret > 0) {
return 0;
}
ret = sd_bus_wait(bus, std::numeric_limits<uint64_t>::max());
if (ret < 0) {
sd_journal_print(LOG_ERR, "sd-bus event loop error:%s", strerror(-ret));
return ret;
}
return ret;
}
int main(int argc, const char *argv[])
{
sd_bus_error error{SD_BUS_ERROR_NULL};
sd_bus_message *msg{nullptr};
sd_bus *bus{nullptr};
std::string serviceId;
int ret;
if (ret = sd_bus_open_user(&bus); ret < 0) {
sd_journal_perror("Failed to connect to user bus.");
releaseRes(error, msg, bus, ExitCode::InternalError);
}
if (ret = sd_bus_message_new_method_call(
bus, &msg, SystemdService, SystemdObjectPath, SystemdInterfaceName, "StartTransientUnit");
ret < 0) {
sd_journal_perror("Failed to create D-Bus call message");
releaseRes(error, msg, bus, ExitCode::InternalError);
}
std::deque<std::string_view> args;
for (int i = 1; i < argc; ++i) {
args.emplace_back(argv[i]);
}
serviceId = cmdParse(msg, std::move(args));
if (serviceId == "internalError") {
releaseRes(error, msg, bus, ExitCode::InternalError);
} else if (serviceId == "invalidInput") {
releaseRes(error, msg, bus, ExitCode::InvalidInput);
}
const char *path{nullptr};
JobRemoveResult resultData{serviceId};
if (ret = sd_bus_match_signal(
bus, nullptr, SystemdService, SystemdObjectPath, SystemdInterfaceName, "JobRemoved", jobRemovedReceiver, &resultData);
ret < 0) {
sd_journal_perror("add signal matcher failed.");
releaseRes(error, msg, bus, ExitCode::InternalError);
}
sd_bus_message *reply{nullptr};
if (ret = sd_bus_call(bus, msg, 0, &error, &reply); ret < 0) {
sd_bus_error_get_errno(&error);
sd_journal_print(LOG_ERR, "launch failed: [%s,%s]", error.name, error.message);
releaseRes(error, msg, bus, ExitCode::InternalError);
} else {
sd_journal_print(LOG_INFO, "launch %s success.", serviceId.c_str());
}
if (ret = sd_bus_message_read(reply, "o", &path); ret < 0) {
sd_journal_perror("failed to parse response message.");
sd_bus_message_unref(reply);
releaseRes(error, msg, bus, ExitCode::InternalError);
}
sd_bus_message_unref(reply);
while (true) { // wait for jobRemoved
int ret = process_dbus_message(bus);
if (ret < 0) {
releaseRes(error, msg, bus, ExitCode::InternalError);
}
if (resultData.removedFlag) {
break;
}
}
releaseRes(error, msg, bus, resultData.result);
}