--- /dev/null
+/*#############################################################################
+# #
+# Pakfire - The IPFire package management system #
+# Copyright (C) 2023 Pakfire development team #
+# #
+# This program is free software: you can redistribute it and/or modify #
+# it under the terms of the GNU General Public License as published by #
+# the Free Software Foundation, either version 3 of the License, or #
+# (at your option) any later version. #
+# #
+# This program is distributed in the hope that it will be useful, #
+# but WITHOUT ANY WARRANTY; without even the implied warranty of #
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
+# GNU General Public License for more details. #
+# #
+# You should have received a copy of the GNU General Public License #
+# along with this program. If not, see <http://www.gnu.org/licenses/>. #
+# #
+#############################################################################*/
+
+#include <errno.h>
+#include <sys/epoll.h>
+
+#include <pakfire/ctx.h>
+#include <pakfire/httpclient.h>
+#include <pakfire/logging.h>
+
+#include "daemon.h"
+
+#define EPOLL_MAX_EVENTS 8
+
+struct cli_daemon_ctx {
+ struct pakfire_ctx* ctx;
+};
+
+static int cli_daemon_loop(struct pakfire_ctx* ctx) {
+ struct pakfire_httpclient* httpclient = NULL;
+ struct epoll_event events[EPOLL_MAX_EVENTS];
+ struct epoll_event event;
+ int epollfd = -1;
+ int num = 0;
+ int fd = -1;
+ int e;
+ int r;
+
+ // Setup the HTTP client
+ r = pakfire_httpclient_create(&httpclient, ctx);
+ if (r)
+ goto ERROR;
+
+ // Setup epoll()
+ epollfd = epoll_create1(EPOLL_CLOEXEC);
+ if (epollfd < 0) {
+ CTX_ERROR(ctx, "Could not setup main loop: %s\n", strerror(errno));
+ r = -errno;
+ goto ERROR;
+ }
+
+ CTX_DEBUG(ctx, "Entering main loop...\n");
+
+ // Main loop, loop, loop, looooop....
+ for (;;) {
+ num = epoll_wait(epollfd, events, EPOLL_MAX_EVENTS, -1);
+ if (num < 1) {
+ switch (errno) {
+ case EINTR:
+ break;
+
+ default:
+ CTX_ERROR(ctx, "epoll_wait() failed: %m\n");
+ r = -errno;
+ break;
+ }
+ goto ERROR;
+ }
+
+ // Handle all events
+ for (int i = 0; i < num; i++) {
+ fd = events[i].data.fd;
+ e = events[i].events;
+
+ // XXX Do the work...
+
+ // Remove the file descriptor if it has been closed
+ if (e & EPOLLHUP) {
+ r = epoll_ctl(epollfd, EPOLL_CTL_DEL, fd, NULL);
+ if (r) {
+ CTX_ERROR(ctx, "Could not remove closed fd %d: %s\n", fd, strerror(errno));
+ r = -errno;
+ goto ERROR;
+ }
+ }
+ }
+ }
+
+ERROR:
+ CTX_DEBUG(ctx, "Exited main loop...\n");
+
+ if (httpclient)
+ pakfire_httpclient_unref(httpclient);
+ if (epollfd >= 0)
+ close(epollfd);
+
+ return r;
+}
+
+int cli_daemon_main(struct pakfire_ctx* ctx) {
+ int r;
+
+ CTX_INFO(ctx, "Pakfire Daemon has been successfully initialized\n");
+
+ // Run the main loop
+ r = cli_daemon_loop(ctx);
+ if (r)
+ goto ERROR;
+
+ERROR:
+ return r;
+}
--- /dev/null
+/*#############################################################################
+# #
+# Pakfire - The IPFire package management system #
+# Copyright (C) 2023 Pakfire development team #
+# #
+# This program is free software: you can redistribute it and/or modify #
+# it under the terms of the GNU General Public License as published by #
+# the Free Software Foundation, either version 3 of the License, or #
+# (at your option) any later version. #
+# #
+# This program is distributed in the hope that it will be useful, #
+# but WITHOUT ANY WARRANTY; without even the implied warranty of #
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
+# GNU General Public License for more details. #
+# #
+# You should have received a copy of the GNU General Public License #
+# along with this program. If not, see <http://www.gnu.org/licenses/>. #
+# #
+#############################################################################*/
+
+#ifndef PAKFIRE_CLI_DAEMON_H
+#define PAKFIRE_CLI_DAEMON_H
+
+#include <pakfire/ctx.h>
+
+int cli_daemon_main(struct pakfire_ctx* ctx);
+
+#endif /* PAKFIRE_CLI_DAEMON_H */