#include <signal.h>
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <syslog.h>
#include <unistd.h>
#include "config.h"
#include "git.h"
#include "macros.h"
#include "network.h"
#define UNIX_PATH_MAX 108
pid_t fork_proc() {
pid_t pid = fork();
if (pid) {
exit((pid < 0) ? EXIT_FAILURE : EXIT_SUCCESS);
}
return pid;
}
void child_handler(int sig_num) {
signal(SIGCHLD, child_handler);
}
void hangup_handler(int sig_num) {
signal(SIGHUP, hangup_handler);
}
void init_daemon(int change_dir, char *path) {
pid_t pid = fork_proc();
if (setsid() < 0) {
exit(EXIT_FAILURE);
}
signal(SIGCHLD, child_handler);
signal(SIGHUP, hangup_handler);
pid = fork_proc();
umask(0);
if (change_dir) {
chdir((!path) ? "/" : path);
}
for (int i = sysconf(_SC_OPEN_MAX); i >= 0; close(i--));
}
int init_config(char *config_file, config **cfg) {
log(LOG_INFO, "Reading config file %s.", config_file);
*cfg = parse_config(config_file);
if (*cfg != NULL) {
log(LOG_INFO, "Successfully read %s.", config_file);
} else {
log(LOG_ERR, "Error reading %s.", config_file);
}
return (*cfg != NULL);
}
int init_socket(config *cfg) {
union sock_addr {
struct sockaddr sa;
struct sockaddr_un sa_un;
struct sockaddr_in sa_in;
} sock_addr;
size_t sockaddr_size = 0;
const int sock_type = (strcasecmp(cfg->sock_type, "unix") == 0) ? AF_UNIX : AF_INET;
if (sock_type == AF_UNIX) {
size_t path_length = strlen(cfg->sock);
if (path_length >= UNIX_PATH_MAX) {
log(LOG_ERR, "The socket path is too long, and exceeded %i characters.", UNIX_PATH_MAX);
return -1;
}
sockaddr_size = sizeof(sock_addr.sa_un);
sock_addr.sa_un.sun_family = sock_type;
memcpy(sock_addr.sa_un.sun_path, cfg->sock, path_length+1);
} else {
struct addrinfo hints, *ainfo_root, *ainfo;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_SEQPACKET;
hints.ai_flags = AI_PASSIVE;
const int gai_ret = getaddrinfo(cfg->sock, cfg->port, &hints, &ainfo_root);
if (gai_ret) {
log_reason(LOG_ERR, "getaddrinfo() failed for hostname %s.", cfg->sock, gai_strerror(gai_ret));
return -1;
}
TODO
sockaddr_size = ainfo_root->ai_addrlen;
sock_addr.sa = *ainfo_root->ai_addr;
freeaddrinfo(ainfo_root);
}
const int fd = socket(sock_addr.sa.sa_family, SOCK_SEQPACKET, 0);
if (fd < 0) {
log_reason(LOG_ERR, "Failed to create listen socket.", strerror(errno));
return -1;
}
if (bind(fd, &sock_addr.sa, sockaddr_size) < 0) {
log_reason(LOG_ERR, "Failed to bind listen socket.", strerror(errno));
return -1;
}
int one = 1;
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one)) < 0) {
log_reason(LOG_ERR, "Failed to enable SO_REUSEADDR.", strerror(errno));
return -1;
}
if (listen(fd, 20) < 0) {
log_reason(LOG_ERR, "Failed to listen to socket.", strerror(errno));
return -1;
}
log(LOG_INFO, "Successfully created socket descriptor %i for address %s.", fd, cfg->sock);
return fd;
}
int main_loop(config *cfg, int listen_socket, git_repository **repos) {
int done = 0;
struct stat st;
if (stat(cfg->pr_root, &st) < 0) {
mkdir(cfg->pr_root, 0755);
}
for (; !done;) {
break;
}
return 0;
}
void cleanup(config *cfg, int listen_socket, git_repository **repos) {
if (listen_socket > 0) {
close(listen_socket);
if (strcasecmp(cfg->sock_type, "unix") == 0) {
unlink(cfg->sock);
}
}
if (cfg) {
cleanup_config(cfg);
}
if (repos) {
cleanup_git(repos);
}
}
int main(int argc, char **argv) {
config *cfg = NULL;
git_repository **repos = NULL;
char *config_file = "test.conf";
int config_read = 0;
int listen_socket = -1;
int exit_status = EXIT_FAILURE;
init_daemon(1, "/");
openlog("pullreqd", LOG_PID, LOG_DAEMON);
log(LOG_NOTICE, "pullreqd started.");
config_read = init_config(config_file, &cfg);
if (config_read) {
listen_socket = init_socket(cfg);
}
if (config_read && listen_socket >= 0) {
repos = init_git(cfg);
if (repos != NULL) {
exit_status = main_loop(cfg, listen_socket, repos);
}
}
cleanup(cfg, listen_socket, repos);
log(LOG_NOTICE, "pullreqd stopped.");
closelog();
return exit_status;
}