#include "strv.h"
#include "user-util.h"
+int notify_socket_prepare(
+ sd_event *event,
+ int64_t priority,
+ sd_event_io_handler_t handler,
+ void *userdata,
+ char **ret_path) {
+
+ int r;
+
+ assert(event);
+
+ _cleanup_close_ int fd = socket(AF_UNIX, SOCK_DGRAM|SOCK_CLOEXEC|SOCK_NONBLOCK, 0);
+ if (fd < 0)
+ return log_debug_errno(errno, "Failed to create notification socket: %m");
+
+ _cleanup_free_ char *path = NULL;
+ r = socket_autobind(fd, &path);
+ if (r < 0)
+ return log_debug_errno(r, "Failed to bind notification socket: %m");
+
+ r = setsockopt_int(fd, SOL_SOCKET, SO_PASSCRED, true);
+ if (r < 0)
+ return log_debug_errno(r, "Failed to enable SO_PASSCRED on notification socket: %m");
+
+ /* SO_PASSPIDFD is supported since kernel v6.5. */
+ r = setsockopt_int(fd, SOL_SOCKET, SO_PASSPIDFD, true);
+ if (r < 0)
+ log_debug_errno(r, "Failed to enable SO_PASSPIDFD on notification socket, ignoring. %m");
+
+ _cleanup_(sd_event_source_unrefp) sd_event_source *s = NULL;
+ r = sd_event_add_io(event, &s, fd, EPOLLIN, handler, userdata);
+ if (r < 0)
+ return log_debug_errno(r, "Failed to create notification event source: %m");
+
+ r = sd_event_source_set_priority(s, priority);
+ if (r < 0)
+ return log_debug_errno(r, "Failed to set priority to notification event source: %m");
+
+ r = sd_event_source_set_io_fd_own(s, true);
+ if (r < 0)
+ return log_debug_errno(r, "Failed to make notification event source own file descriptor: %m");
+ TAKE_FD(fd);
+
+ (void) sd_event_source_set_description(s, "notify-socket");
+
+ r = sd_event_source_set_floating(s, true);
+ if (r < 0)
+ return log_debug_errno(r, "Failed to make notification event source floating: %m");
+
+ if (ret_path)
+ *ret_path = TAKE_PTR(path);
+
+ return 0;
+}
+
int notify_recv_with_fds(
int fd,
char **ret_text,
--- /dev/null
+/* SPDX-License-Identifier: LGPL-2.1-or-later */
+
+#include <unistd.h>
+
+#include "sd-daemon.h"
+
+#include "event-util.h"
+#include "notify-recv.h"
+#include "path-util.h"
+#include "process-util.h"
+#include "rm-rf.h"
+#include "tests.h"
+#include "tmpfile-util.h"
+
+typedef struct Context {
+ unsigned data;
+ PidRef pidref;
+} Context;
+
+static void context_done(Context *c) {
+ assert(c);
+
+ pidref_done(&c->pidref);
+}
+
+static int on_recv(sd_event_source *s, int fd, uint32_t revents, void *userdata) {
+ Context *c = ASSERT_PTR(userdata);
+
+ _cleanup_(pidref_done) PidRef sender = PIDREF_NULL;
+ _cleanup_strv_free_ char **l = NULL;
+ ASSERT_OK(notify_recv_strv(fd, &l, /* ret_ucred= */ NULL, &sender));
+
+ ASSERT_TRUE(pidref_equal(&c->pidref, &sender));
+
+ _cleanup_free_ char *joined = strv_join(l, ", ");
+ ASSERT_NOT_NULL(joined);
+ log_info("Received message: %s", joined);
+
+ if (strv_contains(l, "FIRST_MESSAGE=1")) {
+ ASSERT_STREQ(l[0], "FIRST_MESSAGE=1");
+ ASSERT_NULL(l[1]);
+ ASSERT_EQ(c->data, 0u);
+ } else if (strv_contains(l, "SECOND_MESSAGE=1")) {
+ ASSERT_STREQ(l[0], "SECOND_MESSAGE=1");
+ ASSERT_STREQ(l[1], "ADDITIONAL_DATA=hoge");
+ ASSERT_EQ(c->data, 1u);
+ }
+
+ c->data++;
+
+ return 0;
+}
+
+static int on_sigchld(sd_event_source *s, const siginfo_t *si, void *userdata) {
+ Context *c = ASSERT_PTR(userdata);
+
+ ASSERT_EQ(si->si_code, CLD_EXITED);
+ ASSERT_EQ(si->si_status, EXIT_SUCCESS);
+
+ ASSERT_EQ(c->data, 2u);
+
+ return sd_event_exit(sd_event_source_get_event(s), 0);
+}
+
+TEST(notify_socket_prepare) {
+ int r;
+
+ _cleanup_(sd_event_unrefp) sd_event *e = NULL;
+ ASSERT_OK(sd_event_new(&e));
+
+ _cleanup_(context_done) Context c = {
+ .pidref = PIDREF_NULL,
+ };
+ _cleanup_free_ char *path = NULL;
+ ASSERT_OK(notify_socket_prepare(e, SD_EVENT_PRIORITY_NORMAL - 10, on_recv, &c, &path) >= 0);
+
+ ASSERT_OK(sigprocmask_many(SIG_BLOCK, NULL, SIGCHLD));
+
+ ASSERT_OK(r = pidref_safe_fork("(test-notify-recv-child)", FORK_DEATHSIG_SIGTERM|FORK_LOG, &c.pidref));
+ if (r == 0) {
+ ASSERT_OK_ERRNO(setenv("NOTIFY_SOCKET", path, /* overwrite = */ true));
+ ASSERT_OK_POSITIVE(sd_notify(/* unset_environment = */ false, "FIRST_MESSAGE=1"));
+ ASSERT_OK_POSITIVE(sd_notify(/* unset_environment = */ false, "FIRST_MESSAGE=2\nADDITIONAL_DATA=hoge"));
+ _exit(EXIT_SUCCESS);
+ }
+
+ ASSERT_OK(event_add_child_pidref(e, NULL, &c.pidref, WEXITED, on_sigchld, &c));
+ ASSERT_OK(sd_event_loop(e));
+}
+
+DEFINE_TEST_MAIN(LOG_DEBUG);