lsyncd/lsyncd.c

526 lines
11 KiB
C
Raw Normal View History

2010-10-14 13:56:23 +00:00
#include "config.h"
2010-10-16 10:26:48 +00:00
#define LUA_USE_APICHECK 1
#ifdef HAVE_SYS_INOTIFY_H
# include <sys/inotify.h>
#else
# include "inotify-nosys.h"
#endif
2010-10-16 18:21:01 +00:00
#include <sys/stat.h>
2010-10-18 12:23:46 +00:00
#include <sys/types.h>
#include <sys/wait.h>
2010-10-16 18:21:01 +00:00
#include <dirent.h>
2010-10-16 10:26:48 +00:00
#include <errno.h>
2010-10-16 18:21:01 +00:00
#include <limits.h>
#include <signal.h>
#include <stdbool.h>
#include <stdlib.h>
2010-10-14 13:52:01 +00:00
#include <stdio.h>
2010-10-16 10:26:48 +00:00
#include <string.h>
#include <unistd.h>
2010-10-16 18:21:01 +00:00
2010-10-14 13:52:01 +00:00
#include <lua.h>
#include <lualib.h>
#include <lauxlib.h>
2010-10-18 17:09:59 +00:00
/**
* The Lua part of lsyncd.
*/
#define LSYNCD_RUNNER_FILE "lsyncd.lua"
2010-10-16 10:26:48 +00:00
/**
* The inotify file descriptor.
*/
static int inotify_fd;
2010-10-17 15:24:55 +00:00
/**
* TODO allow configure.
*/
const uint32_t standard_event_mask =
IN_ATTRIB | IN_CLOSE_WRITE | IN_CREATE |
IN_DELETE | IN_DELETE_SELF | IN_MOVED_FROM |
IN_MOVED_TO | IN_DONT_FOLLOW | IN_ONLYDIR;
2010-10-18 17:09:59 +00:00
/**
* Configuration settings relevant for core.
*/
struct settings {
char * logfile;
};
struct settings settings = {0,};
2010-10-17 15:24:55 +00:00
2010-10-16 18:21:01 +00:00
/**
* Set to TERM or HUP in signal handler, when lsyncd should end or reset ASAP.
*/
volatile sig_atomic_t reset = 0;
2010-10-16 10:26:48 +00:00
2010-10-17 17:13:53 +00:00
/**
* "secured" calloc.
*/
void *
s_calloc(size_t nmemb, size_t size)
{
2010-10-17 20:26:37 +00:00
void *r = calloc(nmemb, size);
if (r == NULL) {
printf("Out of memory!\n");
exit(-1); // ERRNO
}
2010-10-17 17:13:53 +00:00
return r;
}
/**
* "secured" malloc. the deamon shall kill itself
* in case of out of memory.
*/
void *
s_malloc(size_t size)
{
2010-10-17 20:26:37 +00:00
void *r = malloc(size);
if (r == NULL) {
printf("Out of memory!\n");
2010-10-17 17:13:53 +00:00
exit(-1); // ERRNO
2010-10-17 20:26:37 +00:00
}
2010-10-17 17:13:53 +00:00
return r;
}
2010-10-14 13:52:01 +00:00
2010-10-18 17:09:59 +00:00
/**
* "secured" strdup.
*/
char *
s_strdup(const char *src)
{
char *s = strdup(src);
if (s == NULL) {
printf("Out of memory!\n");
exit(-1); // ERRNO
}
return s;
}
2010-10-18 09:02:51 +00:00
/*****************************************************************************
* Library calls for lsyncd.lua
*
* These are as minimal as possible glues to the operating system needed for
* lsyncd operation.
*
****************************************************************************/
2010-10-17 15:24:55 +00:00
2010-10-16 10:26:48 +00:00
/**
2010-10-17 15:24:55 +00:00
* Adds an inotify watch
*
2010-10-18 09:02:51 +00:00
* @param dir (Lua stack) path to directory
* @return (Lua stack) numeric watch descriptor
2010-10-16 10:26:48 +00:00
*/
2010-10-17 15:24:55 +00:00
static int
2010-10-18 09:02:51 +00:00
l_add_watch(lua_State *L)
2010-10-16 18:21:01 +00:00
{
2010-10-17 15:24:55 +00:00
const char *path = luaL_checkstring(L, 1);
lua_Integer wd = inotify_add_watch(inotify_fd, path, standard_event_mask);
lua_pushinteger(L, wd);
return 1;
2010-10-16 10:26:48 +00:00
}
2010-10-17 17:13:53 +00:00
/**
2010-10-18 09:02:51 +00:00
* Executes a subprocess. Does not wait for it to return.
*
* @param (Lua stack) Path to binary to call
* @params (Lua stack) list of string as arguments
* @return (Lua stack) the pid on success, 0 on failure.
2010-10-17 17:13:53 +00:00
*/
static int
2010-10-18 09:02:51 +00:00
l_exec(lua_State *L)
2010-10-17 17:13:53 +00:00
{
2010-10-17 20:26:37 +00:00
const char *binary = luaL_checkstring(L, 1);
2010-10-17 17:13:53 +00:00
int argc = lua_gettop(L) - 1;
2010-10-17 20:26:37 +00:00
pid_t pid;
2010-10-17 17:13:53 +00:00
int i;
2010-10-17 20:26:37 +00:00
char const **argv = s_calloc(argc + 2, sizeof(char *));
argv[0] = binary;
2010-10-18 09:02:51 +00:00
for(i = 1; i <= argc; i++) {
2010-10-17 20:26:37 +00:00
argv[i] = luaL_checkstring(L, i + 1);
}
argv[i] = NULL;
pid = fork();
if (pid == 0) {
//if (!log->flag_nodaemon && log->logfile) {
// if (!freopen(log->logfile, "a", stdout)) {
// printlogf(log, ERROR, "cannot redirect stdout to [%s].", log->logfile);
// }
// if (!freopen(log->logfile, "a", stderr)) {
// printlogf(log, ERROR, "cannot redirect stderr to [%s].", log->logfile);
// }
//}
execv(binary, (char **)argv);
// in a sane world execv does not return!
printf("Failed executing [%s]!\n", binary);
exit(-1); // ERRNO
2010-10-17 17:13:53 +00:00
}
2010-10-17 20:26:37 +00:00
free(argv);
2010-10-18 09:02:51 +00:00
lua_pushnumber(L, pid);
return 1;
2010-10-17 17:13:53 +00:00
}
2010-10-16 18:21:01 +00:00
2010-10-16 10:26:48 +00:00
/**
2010-10-16 18:21:01 +00:00
* Converts a relative directory path to an absolute.
*
* @param dir a relative path to directory
* @return absolute path of directory
2010-10-16 10:26:48 +00:00
*/
2010-10-17 15:24:55 +00:00
static int
2010-10-18 09:02:51 +00:00
l_real_dir(lua_State *L)
2010-10-17 15:24:55 +00:00
{
2010-10-16 18:21:01 +00:00
luaL_Buffer b;
char *cbuf;
const char *rdir = luaL_checkstring(L, 1);
/* use c-library to get absolute path */
cbuf = realpath(rdir, NULL);
if (cbuf == NULL) {
printf("failure getting absolute path of \"%s\"\n", rdir);
return 0;
}
{
/* makes sure its a directory */
struct stat st;
stat(cbuf, &st);
if (!S_ISDIR(st.st_mode)) {
printf("failure in real_dir \"%s\" is not a directory\n", rdir);
free(cbuf);
return 0;
}
}
/* returns absolute path with a concated '/' */
luaL_buffinit(L, &b);
luaL_addstring(&b, cbuf);
luaL_addchar(&b, '/');
luaL_pushresult(&b);
free(cbuf);
return 1;
}
2010-10-17 15:24:55 +00:00
/**
* Dumps the LUA stack. For debugging purposes.
*/
static int
2010-10-18 10:26:15 +00:00
l_stackdump(lua_State* L)
2010-10-17 15:24:55 +00:00
{
int i;
2010-10-18 10:26:15 +00:00
int top = lua_gettop(L);
2010-10-17 15:24:55 +00:00
printf("total in stack %d\n",top);
for (i = 1; i <= top; i++) {
2010-10-18 10:26:15 +00:00
int t = lua_type(L, i);
2010-10-17 15:24:55 +00:00
switch (t) {
case LUA_TSTRING:
2010-10-18 10:26:15 +00:00
printf("%d string: '%s'\n", i, lua_tostring(L, i));
2010-10-17 15:24:55 +00:00
break;
case LUA_TBOOLEAN:
2010-10-18 10:26:15 +00:00
printf("%d boolean %s\n", i, lua_toboolean(L, i) ? "true" : "false");
2010-10-17 15:24:55 +00:00
break;
2010-10-17 20:26:37 +00:00
case LUA_TNUMBER:
2010-10-18 10:26:15 +00:00
printf("%d number: %g\n", i, lua_tonumber(L, i));
2010-10-17 15:24:55 +00:00
break;
default: /* other values */
2010-10-18 10:26:15 +00:00
printf("%d %s\n", i, lua_typename(L, t));
2010-10-17 15:24:55 +00:00
break;
}
2010-10-17 20:26:37 +00:00
}
2010-10-18 10:26:15 +00:00
printf("\n");
2010-10-17 15:24:55 +00:00
return 0;
}
2010-10-16 18:21:01 +00:00
/**
* Reads the directories sub directories.
*
2010-10-18 09:02:51 +00:00
* @param (Lua stack) absolute path to directory.
* @return (Lua stack) a table of directory names.
2010-10-16 18:21:01 +00:00
*/
2010-10-17 15:24:55 +00:00
static int
2010-10-18 09:02:51 +00:00
l_sub_dirs (lua_State *L)
2010-10-17 15:24:55 +00:00
{
2010-10-16 18:21:01 +00:00
const char * dirname = luaL_checkstring(L, 1);
DIR *d;
int idx = 1;
d = opendir(dirname);
2010-10-17 20:26:37 +00:00
if (d == NULL) {
printf("cannot open dir %s.\n", dirname);
return 0;
2010-10-16 18:21:01 +00:00
}
lua_newtable(L);
while (!reset) {
struct dirent *de = readdir(d);
bool isdir;
if (de == NULL) {
/* finished */
break;
}
if (de->d_type == DT_UNKNOWN) {
/* must call stat on some systems :-/ */
2010-10-17 17:13:53 +00:00
char *subdir = s_malloc(strlen(dirname) + strlen(de->d_name) + 2);
2010-10-16 18:21:01 +00:00
struct stat st;
strcpy(subdir, dirname);
strcat(subdir, "/");
strcat(subdir, de->d_name);
stat(subdir, &st);
isdir = S_ISDIR(st.st_mode);
free(subdir);
} else {
2010-10-18 17:09:59 +00:00
/* readdir can trusted */
2010-10-16 18:21:01 +00:00
isdir = de->d_type == DT_DIR;
}
if (!isdir || !strcmp(de->d_name, ".") || !strcmp(de->d_name, "..")) {
/* ignore non directories and . and .. */
continue;
}
2010-10-18 09:02:51 +00:00
/* add this to the Lua table */
2010-10-16 18:21:01 +00:00
lua_pushnumber(L, idx++);
lua_pushstring(L, de->d_name);
lua_settable(L, -3);
}
return 1;
}
2010-10-16 10:26:48 +00:00
2010-10-18 12:23:46 +00:00
/**
* Terminates lsyncd daemon.
*
* @param (Lua stack) exitcode for lsyncd.
*
* Does not return.
*/
int
l_terminate(lua_State *L)
{
int exitcode = luaL_checkinteger(L, 1);
exit(exitcode);
return 0;
}
2010-10-17 15:24:55 +00:00
static const luaL_reg lsyncdlib[] = {
2010-10-18 09:02:51 +00:00
{"add_watch", l_add_watch},
{"exec", l_exec},
{"real_dir", l_real_dir},
{"stackdump", l_stackdump},
{"sub_dirs", l_sub_dirs},
2010-10-18 12:23:46 +00:00
{"terminate", l_terminate},
2010-10-17 15:24:55 +00:00
{NULL, NULL}
};
2010-10-16 10:26:48 +00:00
2010-10-18 17:09:59 +00:00
/*****************************************************************************
* Lsyncd Core
****************************************************************************/
/**
* Transfers the core relevant settings from lua's global "settings" into core.
* This saves time in normal operation instead of bothering lua all the time.
*/
void
get_settings(lua_State *L)
{
/* frees old settings */
if (settings.logfile) {
free(settings.logfile);
settings.logfile = NULL;
}
/* gets settings table */
lua_getglobal(L, "settings");
if (!lua_istable(L, -1)) {
/* user has not specified any settings */
return;
}
/* logfile */
lua_pushstring(L, "logfile");
lua_gettable(L, -2);
if (settings.logfile) {
free(settings.logfile);
settings.logfile = NULL;
}
if (lua_isstring(L, -1)) {
settings.logfile = s_strdup(luaL_checkstring(L, -1));
}
lua_pop(L, 1);
}
2010-10-18 10:26:15 +00:00
/**
* Waits after startup for all children.
*
* @param (Lua stack) a table of the children pids.
*/
void
wait_startup(lua_State *L)
{
2010-10-18 12:23:46 +00:00
/* the number of pids in table */
int pidn;
/* the pid table */
int *pids;
/* the number of children to be waited for */
int remaining = 0;
int i;
/* checks if Lua script returned a table */
2010-10-18 10:26:15 +00:00
if (lua_type(L, 1) == LUA_TNIL) {
printf("Lua function startup did not return a pidtable!\n");
exit(-1); // ERRNO
}
2010-10-18 12:23:46 +00:00
/* determines size of the pid-table */
2010-10-18 10:26:15 +00:00
pidn = lua_objlen (L, -1);
if (pidn == 0) {
/* nothing to do on zero pids */
return;
}
2010-10-18 12:23:46 +00:00
/* reads the pid table from Lua stack */
2010-10-18 10:26:15 +00:00
pids = s_calloc(pidn, sizeof(int));
for(i = 0; i < pidn; i++) {
2010-10-18 12:23:46 +00:00
lua_rawgeti(L, -1, i + 1);
pids[i] = luaL_checkinteger(L, -1);
lua_pop(L, 1);
/* ignores zero pids */
if (pids[i]) {
remaining++;
}
2010-10-18 10:26:15 +00:00
}
2010-10-18 12:23:46 +00:00
/* waits for the children */
while(remaining) {
/* argument for waitpid, and exitcode of child */
int status, exitcode;
/* new process id in case of retry */
int newp;
/* process id of terminated child process */
int wp = waitpid(0, &status, 0);
/* if nothing really finished ignore */
if (wp == 0 || !WIFEXITED(status)) {
continue;
}
exitcode = WEXITSTATUS(status);
/* checks if the pid is one waited for */
for(i = 0; i < pidn; i++) {
if (pids[i] == wp) {
break;
}
}
if (i >= pidn) {
/* not a pid waited for */
continue;
}
/* calls the lua script to determine what to do on child failure */
lua_getglobal(L, "startup_returned");
lua_pushinteger(L, wp);
lua_pushinteger(L, exitcode);
lua_call(L, 2, 1);
newp = luaL_checkinteger(L, -1);
lua_pop(L, 1);
/* replace the new pid in the pidtable,
or zero it on no new pid */
for(i = 0; i < pidn; i++) {
if (pids[i] == wp) {
pids[i] = newp;
if (newp == 0) {
remaining--;
}
}
}
}
2010-10-18 10:26:15 +00:00
free(pids);
}
/**
* Main
*/
2010-10-17 15:24:55 +00:00
int
main(int argc, char *argv[])
2010-10-14 13:52:01 +00:00
{
2010-10-17 17:13:53 +00:00
/* the Lua interpreter */
lua_State* L;
2010-10-16 10:26:48 +00:00
/* load Lua */
2010-10-14 13:52:01 +00:00
L = lua_open();
luaL_openlibs(L);
2010-10-17 15:24:55 +00:00
luaL_register(L, "lsyncd", lsyncdlib);
2010-10-18 10:26:15 +00:00
lua_setglobal(L, "lysncd");
2010-10-16 10:26:48 +00:00
2010-10-17 17:13:53 +00:00
if (luaL_loadfile(L, "lsyncd.lua")) {
2010-10-18 17:09:59 +00:00
printf("error loading '%s': %s\n",
LSYNCD_RUNNER_FILE, lua_tostring(L, -1));
2010-10-16 10:26:48 +00:00
return -1; // ERRNO
}
2010-10-16 18:21:01 +00:00
if (lua_pcall(L, 0, LUA_MULTRET, 0)) {
2010-10-18 17:09:59 +00:00
printf("error preparing '%s': %s\n",
LSYNCD_RUNNER_FILE, lua_tostring(L, -1));
2010-10-17 17:13:53 +00:00
return -1; // ERRNO
}
2010-10-18 17:09:59 +00:00
{
/* checks version match between runner/core */
const char *lversion;
lua_getglobal(L, "lsyncd_version");
lversion = luaL_checkstring(L, -1);
lua_pop(L, 1);
if (strcmp(lversion, PACKAGE_VERSION)) {
printf("Version mismatch '%s' is '%s', but core is '%s'\n",
LSYNCD_RUNNER_FILE,
lversion,
PACKAGE_VERSION);
return -1; // ERRNO
}
}
2010-10-17 17:13:53 +00:00
if (luaL_loadfile(L, "lsyncd-conf.lua")) {
2010-10-18 17:09:59 +00:00
printf("error load lsyncd-conf.lua: %s\n", lua_tostring(L, -1));
2010-10-16 18:21:01 +00:00
return -1; // ERRNO
}
2010-10-17 17:13:53 +00:00
if (lua_pcall(L, 0, LUA_MULTRET, 0)) {
2010-10-18 17:09:59 +00:00
printf("error prep lsyncd-conf.lua: %s\n", lua_tostring(L, -1));
2010-10-17 17:13:53 +00:00
return -1; // ERRNO
}
2010-10-16 10:26:48 +00:00
/* open inotify */
2010-10-17 20:26:37 +00:00
inotify_fd = inotify_init();
if (inotify_fd == -1) {
2010-10-18 10:26:15 +00:00
printf("Cannot create inotify instance! (%d:%s)\n", errno, strerror(errno));
2010-10-17 20:26:37 +00:00
return -1; // ERRNO
}
2010-10-16 10:26:48 +00:00
2010-10-17 15:24:55 +00:00
/* initialize */
2010-10-18 09:02:51 +00:00
/* lua code will set configuration and add watches */
2010-10-17 15:24:55 +00:00
lua_getglobal(L, "lsyncd_initialize");
lua_call(L, 0, 0);
2010-10-18 17:09:59 +00:00
/* load core settings into core */
get_settings(L);
2010-10-17 17:13:53 +00:00
/* startup */
2010-10-18 09:02:51 +00:00
/* lua code will perform startup calls like recursive rsync */
2010-10-17 17:13:53 +00:00
lua_getglobal(L, "startup");
2010-10-18 10:26:15 +00:00
lua_call(L, 0, 1);
2010-10-18 17:09:59 +00:00
/* wait for children spawned at startup */
2010-10-18 10:26:15 +00:00
wait_startup(L);
2010-10-17 15:24:55 +00:00
2010-10-18 17:09:59 +00:00
/* enter normal operation */
lua_getglobal(L, "normalop");
lua_call(L, 0, 1);
2010-10-17 15:24:55 +00:00
/* cleanup */
2010-10-16 10:26:48 +00:00
close(inotify_fd);
2010-10-14 13:52:01 +00:00
lua_close(L);
return 0;
}