2020-08-14 18:15:03 +02:00
|
|
|
/* Copyright (C) 2020 C. McEnroe <june@causal.agency>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero 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 Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2020-08-15 19:05:31 +02:00
|
|
|
#include <errno.h>
|
2020-08-15 20:46:11 +02:00
|
|
|
#include <stdbool.h>
|
2020-08-14 18:15:03 +02:00
|
|
|
#include <stdint.h>
|
2020-08-14 21:56:10 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2020-08-15 15:53:03 +02:00
|
|
|
#include <sys/time.h>
|
2020-08-14 22:17:29 +02:00
|
|
|
#include <unistd.h>
|
2020-08-14 18:15:03 +02:00
|
|
|
|
|
|
|
typedef unsigned char byte;
|
|
|
|
|
2020-08-14 21:56:10 +02:00
|
|
|
extern struct Prepend {
|
|
|
|
size_t cap, len;
|
|
|
|
char **commands;
|
|
|
|
} prepend;
|
|
|
|
|
|
|
|
static inline void prependClear(void) {
|
|
|
|
for (size_t i = 0; i < prepend.len; ++i) {
|
|
|
|
free(prepend.commands[i]);
|
|
|
|
}
|
|
|
|
prepend.len = 0;
|
|
|
|
}
|
|
|
|
static inline int prependAdd(const char *command) {
|
|
|
|
if (prepend.len == prepend.cap) {
|
|
|
|
size_t cap = (prepend.cap ? prepend.cap * 2 : 8);
|
|
|
|
void *ptr = realloc(prepend.commands, sizeof(*prepend.commands) * cap);
|
|
|
|
if (!ptr) return -1;
|
|
|
|
prepend.cap = cap;
|
|
|
|
prepend.commands = ptr;
|
|
|
|
}
|
|
|
|
prepend.commands[prepend.len] = strdup(command);
|
|
|
|
if (!prepend.commands[prepend.len]) return -1;
|
|
|
|
prepend.len++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-08-15 21:16:50 +02:00
|
|
|
enum { LineCap = 512 };
|
2020-08-15 19:05:31 +02:00
|
|
|
struct Line {
|
|
|
|
size_t len;
|
2020-08-15 21:16:50 +02:00
|
|
|
char buf[LineCap];
|
2020-08-15 19:05:31 +02:00
|
|
|
};
|
|
|
|
|
2020-08-15 20:46:11 +02:00
|
|
|
static inline const char *lineFlush(struct Line *line) {
|
2020-08-15 19:05:31 +02:00
|
|
|
if (!line->len) return NULL;
|
|
|
|
line->buf[line->len++] = '\0';
|
|
|
|
return line->buf;
|
|
|
|
}
|
|
|
|
|
2020-08-15 20:46:11 +02:00
|
|
|
static inline const char *lineRead(struct Line *line, int fd) {
|
2020-08-15 19:05:31 +02:00
|
|
|
char *nul = memchr(line->buf, '\0', line->len);
|
|
|
|
if (nul) {
|
|
|
|
nul++;
|
|
|
|
line->len -= nul - line->buf;
|
|
|
|
memmove(line->buf, nul, line->len);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t cap = sizeof(line->buf) - line->len - 1;
|
|
|
|
if (!cap) return lineFlush(line);
|
|
|
|
|
|
|
|
ssize_t len = read(fd, &line->buf[line->len], cap);
|
|
|
|
if (len < 0 && errno != EAGAIN) return NULL;
|
|
|
|
if (len > 0) line->len += len;
|
|
|
|
|
|
|
|
char *nl = memchr(line->buf, '\n', line->len);
|
|
|
|
if (nl) {
|
|
|
|
*nl = '\0';
|
|
|
|
return line->buf;
|
|
|
|
} else {
|
|
|
|
errno = EAGAIN;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-14 22:17:29 +02:00
|
|
|
enum {
|
|
|
|
SHELL,
|
|
|
|
PATH,
|
|
|
|
USER,
|
|
|
|
HOME,
|
|
|
|
EnvironNull,
|
|
|
|
EnvironLen,
|
|
|
|
};
|
|
|
|
|
2020-08-14 21:56:10 +02:00
|
|
|
extern const char *serviceDir;
|
2020-08-14 22:17:29 +02:00
|
|
|
extern uid_t serviceUID;
|
|
|
|
extern gid_t serviceGID;
|
|
|
|
extern char *serviceEnviron[EnvironLen];
|
2020-08-14 21:56:10 +02:00
|
|
|
|
2020-08-14 22:51:32 +02:00
|
|
|
enum State {
|
|
|
|
Stop,
|
|
|
|
Start,
|
2020-08-15 00:11:14 +02:00
|
|
|
Restart,
|
2020-08-14 22:51:32 +02:00
|
|
|
};
|
|
|
|
|
2020-08-14 21:56:10 +02:00
|
|
|
struct Service {
|
|
|
|
char *name;
|
|
|
|
char *command;
|
2020-08-16 23:28:57 +02:00
|
|
|
bool privileged;
|
2020-08-14 22:51:32 +02:00
|
|
|
enum State intent;
|
|
|
|
enum State state;
|
|
|
|
pid_t pid;
|
|
|
|
int outPipe[2];
|
|
|
|
int errPipe[2];
|
|
|
|
struct Line outLine;
|
|
|
|
struct Line errLine;
|
2020-08-16 00:25:40 +02:00
|
|
|
struct timespec startTime;
|
2020-08-14 23:44:27 +02:00
|
|
|
struct timespec restartInterval;
|
|
|
|
struct timespec restartDeadline;
|
2020-08-14 21:56:10 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
extern struct Services {
|
|
|
|
size_t cap, len;
|
|
|
|
struct Service *ptr;
|
|
|
|
} services;
|
|
|
|
|
|
|
|
int serviceAdd(const char *name, const char *command);
|
2020-08-17 05:01:25 +02:00
|
|
|
void serviceDrop(size_t index);
|
2020-08-15 21:29:58 +02:00
|
|
|
void serviceStatus(struct Service *service);
|
2020-08-14 23:36:31 +02:00
|
|
|
void serviceStart(struct Service *service);
|
2020-08-15 00:11:14 +02:00
|
|
|
void serviceStop(struct Service *service);
|
|
|
|
void serviceRestart(struct Service *service);
|
|
|
|
void serviceSignal(struct Service *service, int signal);
|
2020-08-15 20:46:11 +02:00
|
|
|
void serviceRead(struct Service *service);
|
2020-08-15 00:38:14 +02:00
|
|
|
void serviceReap(pid_t pid, int status);
|
2020-08-14 21:56:10 +02:00
|
|
|
|
2020-08-14 18:15:03 +02:00
|
|
|
struct Set256 {
|
|
|
|
uint32_t bits[8];
|
|
|
|
};
|
|
|
|
static inline void setClear(struct Set256 *set) {
|
|
|
|
for (int i = 0; i < 8; ++i) set->bits[i] = 0;
|
|
|
|
}
|
|
|
|
static inline void setAdd(struct Set256 *set, byte x) {
|
|
|
|
set->bits[x / 32] |= 1 << (uint32_t)(x & 31);
|
|
|
|
}
|
|
|
|
static inline uint32_t setTest(const struct Set256 *set, byte x) {
|
|
|
|
return set->bits[x / 32] & (1 << (uint32_t)(x & 31));
|
|
|
|
}
|
|
|
|
|
|
|
|
extern struct Set256 stopExits;
|
2020-08-14 23:44:27 +02:00
|
|
|
extern struct timespec restartInterval;
|
2020-08-16 00:25:40 +02:00
|
|
|
extern struct timespec resetInterval;
|