サービスファイルを呼び出すには、Systemd -Dバスインタフェースを使用する必要があります。 D-Busが通信できるメソッドを使用するには、そのメソッドを呼び出すg_dbus_connection_call_sync()
必要があります。また、StartUnit()パラメータモードを " "として指定する必要があります。g_dbus_connection_call_sync()
StartUnit()
replace
================================================== =========
#define SYSTEMD_MNGR_IF "org.freedesktop.systemd1.Manager"
#define SYSTEMD_UNIT_IF "org.freedesktop.systemd1.Unit"
#define SYSTEMD_SERVICE_IF "org.freedesktop.systemd1.Service"
#define SYSTEMD_PROP_IF "org.freedesktop.DBus.Properties"
#define SYSTEMD_SERVICE "org.freedesktop.systemd1"
#define SYSTEMD_OBJ "/org/freedesktop/systemd1"
GDBusConnection* systemd_conn = NULL ;
ret = g_dbus_connection_call_sync(systemd_conn,
SYSTEMD_SERVICE,
SYSTEMD_OBJ,
SYSTEMD_MNGR_IF,
"StartUnit",
NULL,
NULL,
G_DBUS_CALL_FLAGS_NONE,
-1,
NULL,
&error);
================================================== ==================代替パラメータをどこに渡すのですか?新しいメソッドStartUnitを定義する必要がありますか?
答え1
必要なオリジナルのd-busソリューションではありませんが、systemd
systemdインターフェイスに合わせて調整されたdbus独自のラッパーを使用してください。
私は使うsd_bus_call_method()
使用libsystemd-dev
:
sd_bus* bus
は割り当てられたオブジェクトです。sd_bus_default()
。name
エスケープされていない単位名(例foo.service
:)。method
その一つです"StartUnit"
、、、"StopUnit"
"RestartUnit"
#include <systemd/sd-bus.h>
static void CallMethodSS(
sd_bus* bus,
const std::string& name,
const std::string& method)
{
sd_bus_error err = SD_BUS_ERROR_NULL;
sd_bus_message* msg = nullptr;
int r;
r = sd_bus_call_method(bus,
"org.freedesktop.systemd1", /* <service> */
"/org/freedesktop/systemd1", /* <path> */
"org.freedesktop.systemd1.Manager", /* <interface> */
method.c_str(), /* <method> */
&err, /* object to return error in */
&msg, /* return message on success */
"ss", /* <input_signature (string-string)> */
name.c_str(), "replace" ); /* <arguments...> */
if (r < 0)
{
std::string err_str("Could not send " + method +
" command to systemd for service: " + name +
". Error: " + err.message );
sd_bus_error_free(&err);
sd_bus_message_unref(msg);
throw exception(err_str);
}
// Extra stuff that might be useful: display the response...
char* response;
r = sd_bus_message_read(msg, "o", &response);
if (r < 0)
{
LogError("Failed to parse response message: %s\n", strerror(-r) );
}
sd_bus_error_free(&err);
sd_bus_message_unref(msg);
}
systemd開発者はここに良い説明を提供しました。
答え2
私の他の答えは別のライブラリを使用しています。この答えは私の経験sd_bus
とGDBusConnectionドキュメントを読んだことに基づいて推測されます。
署名g_dbus_connection_call_sync
例:
GVariant * g_dbus_connection_call_sync(
GDBusConnection *connection,
const gchar *bus_name,
const gchar *object_path,
const gchar *interface_name,
const gchar *method_name,
GVariant *parameters,
const GVariantType *reply_type,
GDBusCallFlags flags,
gint timeout_msec,
GCancellable *cancellable,
GError **error
);
GVariant* parameters
ここで最も興味深い部分がこれです。 Systemd
~の"StartUnit"
このメソッドには2つのパラメータがあります。
StartUnit(in s name, /* Name of the unit (foo.service) */
in s mode, /* Mode of operation (replace) */
out o job);
GVariant* parameters
したがって、私たちはこれら2つの文字列を埋める必要があります。私たちはこれを次のようにすることができます:
g_variant_new ("(ss)", "foo.service", "replace"),
2つの文字列を含むバリアントの生成を"(ss)"
指示する形式文字列です。g_variant_new
次の2つのパラメータは対応する文字列です。
これをコードに適用すると、次のような結果が得られます。
ret = g_dbus_connection_call_sync(
systemd_conn,
SYSTEMD_SERVICE,
SYSTEMD_OBJ,
SYSTEMD_MNGR_IF,
"StartUnit",
g_variant_new ("(ss)", "foo.service", "replace"),
NULL,
G_DBUS_CALL_FLAGS_NONE,
-1,
NULL,
&error );