2010-11-22 20:09:52 +00:00
|
|
|
|
/**
|
2010-11-22 21:17:08 +00:00
|
|
|
|
* inotify.c from Lsyncd - Live (Mirror) Syncing Demon
|
2010-11-22 20:09:52 +00:00
|
|
|
|
*
|
|
|
|
|
* License: GPLv2 (see COPYING) or any later version
|
|
|
|
|
*
|
|
|
|
|
* Authors: Axel Kittenberger <axkibe@gmail.com>
|
|
|
|
|
*
|
2010-11-22 21:17:08 +00:00
|
|
|
|
* -----------------------------------------------------------------------
|
|
|
|
|
*
|
2010-11-22 20:09:52 +00:00
|
|
|
|
* Event interface for Lsyncd to Linux´ inotify.
|
|
|
|
|
*/
|
2010-11-22 21:17:08 +00:00
|
|
|
|
|
2010-11-22 20:09:52 +00:00
|
|
|
|
#include "lsyncd.h"
|
|
|
|
|
|
|
|
|
|
#ifndef HAVE_SYS_INOTIFY_H
|
|
|
|
|
# error Missing <sys/inotify.h>; supply kernel-headers and rerun configure.
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <sys/times.h>
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <sys/wait.h>
|
|
|
|
|
#include <sys/inotify.h>
|
|
|
|
|
#include <dirent.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <signal.h>
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <syslog.h>
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
|
|
#include <lua.h>
|
|
|
|
|
#include <lualib.h>
|
|
|
|
|
#include <lauxlib.h>
|
|
|
|
|
|
2010-11-24 18:19:13 +00:00
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
|
* Event types.
|
|
|
|
|
*/
|
2010-11-24 21:38:48 +00:00
|
|
|
|
static const char * ATTRIB = "Attrib";
|
|
|
|
|
static const char * MODIFY = "Modify";
|
|
|
|
|
static const char * CREATE = "Create";
|
|
|
|
|
static const char * DELETE = "Delete";
|
|
|
|
|
static const char * MOVE = "Move";
|
2010-11-24 18:19:13 +00:00
|
|
|
|
|
2010-11-22 20:09:52 +00:00
|
|
|
|
/**
|
|
|
|
|
* The inotify file descriptor.
|
|
|
|
|
*/
|
2010-11-22 21:17:08 +00:00
|
|
|
|
static int inotify_fd = -1;
|
2010-11-22 20:09:52 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* TODO allow configure.
|
|
|
|
|
*/
|
|
|
|
|
static 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;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Adds an inotify watch
|
|
|
|
|
*
|
|
|
|
|
* @param dir (Lua stack) path to directory
|
|
|
|
|
* @return (Lua stack) numeric watch descriptor
|
|
|
|
|
*/
|
|
|
|
|
static int
|
|
|
|
|
l_addwatch(lua_State *L)
|
|
|
|
|
{
|
|
|
|
|
const char *path = luaL_checkstring(L, 1);
|
2010-11-26 10:59:18 +00:00
|
|
|
|
int wd = inotify_add_watch(inotify_fd, path, standard_event_mask);
|
2010-11-24 18:19:13 +00:00
|
|
|
|
printlogf(L, "Inotify", "addwatch(%s)->%d", path, wd);
|
2010-11-22 20:09:52 +00:00
|
|
|
|
lua_pushinteger(L, wd);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Removes an inotify watch
|
|
|
|
|
*
|
|
|
|
|
* @param dir (Lua stack) numeric watch descriptor
|
|
|
|
|
* @return nil
|
|
|
|
|
*/
|
|
|
|
|
static int
|
|
|
|
|
l_rmwatch(lua_State *L)
|
|
|
|
|
{
|
2010-11-26 10:59:18 +00:00
|
|
|
|
int wd = luaL_checkinteger(L, 1);
|
2010-11-22 20:09:52 +00:00
|
|
|
|
inotify_rm_watch(inotify_fd, wd);
|
2010-11-24 18:19:13 +00:00
|
|
|
|
printlogf(L, "Inotify", "rmwatch()<-%d", wd);
|
2010-11-22 20:09:52 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const luaL_reg linotfylib[] = {
|
|
|
|
|
{"addwatch", l_addwatch },
|
|
|
|
|
{"rmwatch", l_rmwatch },
|
|
|
|
|
{NULL, NULL}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Buffer for MOVE_FROM events.
|
2010-11-22 21:17:08 +00:00
|
|
|
|
* Lsyncd buffers MOVE_FROM events to check if
|
|
|
|
|
* they are followed by MOVE_TO events with identical cookie
|
|
|
|
|
* then they are condensed into one move event to be sent to the
|
|
|
|
|
* runner
|
2010-11-22 20:09:52 +00:00
|
|
|
|
*/
|
|
|
|
|
static struct inotify_event * move_event_buf = NULL;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Memory allocated for move_event_buf
|
|
|
|
|
*/
|
|
|
|
|
static size_t move_event_buf_size = 0;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* true if the buffer is used.
|
|
|
|
|
*/
|
|
|
|
|
static bool move_event = false;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Handles an inotify event.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
handle_event(lua_State *L,
|
|
|
|
|
struct inotify_event *event)
|
|
|
|
|
{
|
2010-11-24 18:19:13 +00:00
|
|
|
|
const char *event_type = NULL;
|
2010-11-22 20:09:52 +00:00
|
|
|
|
|
|
|
|
|
/* used to execute two events in case of unmatched MOVE_FROM buffer */
|
|
|
|
|
struct inotify_event *after_buf = NULL;
|
|
|
|
|
if (event && (IN_Q_OVERFLOW & event->mask)) {
|
|
|
|
|
/* and overflow happened, tells the runner */
|
|
|
|
|
load_runner_func(L, "overflow");
|
|
|
|
|
if (lua_pcall(L, 0, 0, -2)) {
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
|
|
|
|
lua_pop(L, 1);
|
|
|
|
|
hup = 1;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
/* cancel on ignored or resetting */
|
|
|
|
|
if (event && (IN_IGNORED & event->mask)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (event && event->len == 0) {
|
|
|
|
|
/* sometimes inotify sends such strange events,
|
|
|
|
|
* (e.g. when touching a dir */
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (event == NULL) {
|
|
|
|
|
/* a buffered MOVE_FROM is not followed by anything,
|
|
|
|
|
thus it is unary */
|
|
|
|
|
event = move_event_buf;
|
2010-11-24 18:19:13 +00:00
|
|
|
|
event_type = "Delete";
|
2010-11-22 20:09:52 +00:00
|
|
|
|
move_event = false;
|
|
|
|
|
} else if (move_event &&
|
|
|
|
|
( !(IN_MOVED_TO & event->mask) ||
|
|
|
|
|
event->cookie != move_event_buf->cookie) ) {
|
|
|
|
|
/* there is a MOVE_FROM event in the buffer and this is not the match
|
2010-11-23 23:19:01 +00:00
|
|
|
|
* continue in this function iteration to handle the buffer instead */
|
2010-11-24 18:19:13 +00:00
|
|
|
|
logstring("Inotify", "icore, changing unary MOVE_FROM into DELETE")
|
2010-11-22 20:09:52 +00:00
|
|
|
|
after_buf = event;
|
|
|
|
|
event = move_event_buf;
|
2010-11-24 18:19:13 +00:00
|
|
|
|
event_type = "Delete";
|
2010-11-22 20:09:52 +00:00
|
|
|
|
move_event = false;
|
|
|
|
|
} else if ( move_event &&
|
|
|
|
|
(IN_MOVED_TO & event->mask) &&
|
|
|
|
|
event->cookie == move_event_buf->cookie ) {
|
|
|
|
|
/* this is indeed a matched move */
|
2010-11-24 18:19:13 +00:00
|
|
|
|
event_type = "Move";
|
2010-11-22 20:09:52 +00:00
|
|
|
|
move_event = false;
|
|
|
|
|
} else if (IN_MOVED_FROM & event->mask) {
|
|
|
|
|
/* just the MOVE_FROM, buffers this event, and wait if next event is
|
|
|
|
|
* a matching MOVED_TO of this was an unary move out of the watched
|
|
|
|
|
* tree. */
|
|
|
|
|
size_t el = sizeof(struct inotify_event) + event->len;
|
|
|
|
|
if (move_event_buf_size < el) {
|
|
|
|
|
move_event_buf_size = el;
|
|
|
|
|
move_event_buf = s_realloc(move_event_buf, el);
|
|
|
|
|
}
|
|
|
|
|
memcpy(move_event_buf, event, el);
|
|
|
|
|
move_event = true;
|
|
|
|
|
return;
|
|
|
|
|
} else if (IN_MOVED_TO & event->mask) {
|
|
|
|
|
/* must be an unary move-to */
|
|
|
|
|
event_type = CREATE;
|
|
|
|
|
} else if (IN_MOVED_FROM & event->mask) {
|
|
|
|
|
/* must be an unary move-from */
|
|
|
|
|
event_type = DELETE;
|
|
|
|
|
} else if (IN_ATTRIB & event->mask) {
|
|
|
|
|
/* just attrib change */
|
|
|
|
|
event_type = ATTRIB;
|
|
|
|
|
} else if (IN_CLOSE_WRITE & event->mask) {
|
|
|
|
|
/* closed after written something */
|
|
|
|
|
event_type = MODIFY;
|
|
|
|
|
} else if (IN_CREATE & event->mask) {
|
|
|
|
|
/* a new file */
|
|
|
|
|
event_type = CREATE;
|
|
|
|
|
} else if (IN_DELETE & event->mask) {
|
|
|
|
|
/* rm'ed */
|
|
|
|
|
event_type = DELETE;
|
|
|
|
|
} else {
|
2010-11-24 18:19:13 +00:00
|
|
|
|
logstring("Inotify", "icore, skipped some inotify event.");
|
2010-11-22 20:09:52 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* and hands over to runner */
|
|
|
|
|
load_runner_func(L, "inotifyEvent");
|
2010-11-24 18:19:13 +00:00
|
|
|
|
if (!event_type) {
|
2010-11-22 20:09:52 +00:00
|
|
|
|
logstring("Error", "Internal: unknown event in handle_event()");
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
2010-11-24 18:19:13 +00:00
|
|
|
|
lua_pushstring(L, event_type);
|
2010-11-22 20:09:52 +00:00
|
|
|
|
if (event_type != MOVE) {
|
|
|
|
|
lua_pushnumber(L, event->wd);
|
|
|
|
|
} else {
|
|
|
|
|
lua_pushnumber(L, move_event_buf->wd);
|
|
|
|
|
}
|
|
|
|
|
lua_pushboolean(L, (event->mask & IN_ISDIR) != 0);
|
|
|
|
|
lua_pushinteger(L, times(NULL));
|
|
|
|
|
if (event_type == MOVE) {
|
|
|
|
|
lua_pushstring(L, move_event_buf->name);
|
|
|
|
|
lua_pushnumber(L, event->wd);
|
|
|
|
|
lua_pushstring(L, event->name);
|
|
|
|
|
} else {
|
|
|
|
|
lua_pushstring(L, event->name);
|
|
|
|
|
lua_pushnil(L);
|
|
|
|
|
lua_pushnil(L);
|
|
|
|
|
}
|
|
|
|
|
if (lua_pcall(L, 7, 0, -9)) {
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
|
|
|
|
lua_pop(L, 1);
|
2010-11-23 23:19:01 +00:00
|
|
|
|
|
2010-11-22 20:09:52 +00:00
|
|
|
|
/* if there is a buffered event executes it */
|
|
|
|
|
if (after_buf) {
|
2010-11-24 18:19:13 +00:00
|
|
|
|
logstring("Inotify", "icore, handling buffered event.");
|
2010-11-22 20:09:52 +00:00
|
|
|
|
handle_event(L, after_buf);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-22 21:17:08 +00:00
|
|
|
|
/**
|
|
|
|
|
* buffer to read inotify events into
|
|
|
|
|
*/
|
|
|
|
|
static size_t readbuf_size = 2048;
|
|
|
|
|
static char * readbuf = NULL;
|
2010-11-22 20:09:52 +00:00
|
|
|
|
|
|
|
|
|
/**
|
2010-11-22 21:17:08 +00:00
|
|
|
|
* Called by function pointer from when the inotify file descriptor
|
|
|
|
|
* became ready. Reads it contents and forward all received events
|
|
|
|
|
* to the runner.
|
2010-11-22 20:09:52 +00:00
|
|
|
|
*/
|
|
|
|
|
static void
|
2010-11-25 22:10:24 +00:00
|
|
|
|
inotify_ready(lua_State *L, struct observance *observance)
|
2010-11-22 20:09:52 +00:00
|
|
|
|
{
|
2010-11-25 22:10:24 +00:00
|
|
|
|
if (observance->fd != inotify_fd) {
|
|
|
|
|
logstring("Error",
|
|
|
|
|
"internal fail, inotify_ready on non-inotify file descriptor.");
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
2010-11-22 20:09:52 +00:00
|
|
|
|
while(true) {
|
2010-11-23 23:19:01 +00:00
|
|
|
|
ptrdiff_t len;
|
|
|
|
|
int err;
|
2010-11-22 20:09:52 +00:00
|
|
|
|
do {
|
|
|
|
|
len = read (inotify_fd, readbuf, readbuf_size);
|
2010-11-23 23:19:01 +00:00
|
|
|
|
err = errno;
|
|
|
|
|
if (len < 0 && err == EINVAL) {
|
2010-11-22 20:09:52 +00:00
|
|
|
|
/* kernel > 2.6.21 indicates that way that way that
|
|
|
|
|
* the buffer was too small to fit a filename.
|
|
|
|
|
* double its size and try again. When using a lower
|
|
|
|
|
* kernel and a filename > 2KB appears lsyncd
|
|
|
|
|
* will fail. (but does a 2KB filename really happen?)
|
|
|
|
|
*/
|
|
|
|
|
readbuf_size *= 2;
|
|
|
|
|
readbuf = s_realloc(readbuf, readbuf_size);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
} while(0);
|
|
|
|
|
if (len == 0) {
|
|
|
|
|
/* nothing more inotify */
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (len < 0) {
|
2010-11-23 23:19:01 +00:00
|
|
|
|
if (err == EAGAIN) {
|
2010-11-22 20:09:52 +00:00
|
|
|
|
/* nothing more inotify */
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
printlogf(L, "Error", "Read fail on inotify");
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-11-22 21:17:08 +00:00
|
|
|
|
{
|
|
|
|
|
int i = 0;
|
|
|
|
|
while (i < len && !hup && !term) {
|
|
|
|
|
struct inotify_event *event =
|
|
|
|
|
(struct inotify_event *) &readbuf[i];
|
|
|
|
|
handle_event(L, event);
|
|
|
|
|
i += sizeof(struct inotify_event) + event->len;
|
|
|
|
|
}
|
2010-11-22 20:09:52 +00:00
|
|
|
|
}
|
|
|
|
|
if (!move_event) {
|
|
|
|
|
/* give it a pause if not endangering splitting a move */
|
|
|
|
|
break;
|
|
|
|
|
}
|
2010-11-22 21:17:08 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-11-22 20:09:52 +00:00
|
|
|
|
/* checks if there is an unary MOVE_FROM left in the buffer */
|
|
|
|
|
if (move_event) {
|
2010-11-24 18:19:13 +00:00
|
|
|
|
logstring("Inotify", "icore, handling unary move from.");
|
2010-11-22 20:09:52 +00:00
|
|
|
|
handle_event(L, NULL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-25 22:10:24 +00:00
|
|
|
|
/**
|
|
|
|
|
* Called by function pointer when the core doesnt want the
|
|
|
|
|
* inotify fd anymore (term, hup or overflow)
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
inotify_tidy(struct observance *observance)
|
|
|
|
|
{
|
2010-11-25 22:13:57 +00:00
|
|
|
|
if (observance->fd != inotify_fd) {
|
|
|
|
|
logstring("Error",
|
|
|
|
|
"internal fail, inotify_ready on non-inotify file descriptor.");
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
|
|
|
|
close(inotify_fd);
|
|
|
|
|
inotify_fd = -1;
|
2010-11-25 22:10:24 +00:00
|
|
|
|
free(readbuf);
|
|
|
|
|
readbuf = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-22 20:09:52 +00:00
|
|
|
|
/**
|
|
|
|
|
* registers inotify functions.
|
|
|
|
|
*/
|
|
|
|
|
extern void
|
|
|
|
|
register_inotify(lua_State *L) {
|
|
|
|
|
lua_pushstring(L, "inotify");
|
|
|
|
|
luaL_register(L, "inotify", linotfylib);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* opens and initalizes inotify.
|
|
|
|
|
*/
|
|
|
|
|
extern void
|
|
|
|
|
open_inotify(lua_State *L) {
|
2010-11-22 21:17:08 +00:00
|
|
|
|
if (readbuf) {
|
|
|
|
|
logstring("Error",
|
|
|
|
|
"internal fail, inotify readbuf!=NULL in open_inotify()")
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
2010-11-22 20:09:52 +00:00
|
|
|
|
readbuf = s_malloc(readbuf_size);
|
|
|
|
|
|
|
|
|
|
inotify_fd = inotify_init();
|
|
|
|
|
if (inotify_fd == -1) {
|
|
|
|
|
printlogf(L, "Error",
|
|
|
|
|
"Cannot create inotify instance! (%d:%s)",
|
|
|
|
|
errno, strerror(errno));
|
|
|
|
|
exit(-1); // ERRNO
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
close_exec_fd(inotify_fd);
|
|
|
|
|
non_block_fd(inotify_fd);
|
2010-11-25 22:10:24 +00:00
|
|
|
|
observe_fd(inotify_fd, inotify_ready, NULL, inotify_tidy, NULL);
|
2010-11-22 20:09:52 +00:00
|
|
|
|
}
|
|
|
|
|
|