From 767214e4f157ae4a74be0b557bd49e24d972970b Mon Sep 17 00:00:00 2001
From: Lukas Fleischer <lfleischer@calcurse.org>
Date: Thu, 7 Jan 2016 18:23:09 +0100
Subject: Add pre-save and post-save hooks

This adds support for hooks which are executed before/after saving
calcurse data. Hooks can be placed under hooks/pre-save and
hooks/post-save in the data directory and need to be executable.

Potential use cases include:

* Automatically commit any changes to the data files using a VCS.
* Automatically sync with some sever component on data file changes.

Signed-off-by: Lukas Fleischer <lfleischer@calcurse.org>
---
 src/Makefile.am |  1 +
 src/calcurse.h  |  6 ++++++
 src/hooks.c     | 63 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 src/io.c        |  4 ++++
 src/vars.c      |  1 +
 5 files changed, 75 insertions(+)
 create mode 100644 src/hooks.c

(limited to 'src')

diff --git a/src/Makefile.am b/src/Makefile.am
index edb83d8..823f7c9 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -20,6 +20,7 @@ calcurse_SOURCES = \
 	event.c \
 	getstring.c \
 	help.c \
+	hooks.c \
 	ical.c \
 	io.c \
 	keys.c \
diff --git a/src/calcurse.h b/src/calcurse.h
index efbdb81..1ab5363 100644
--- a/src/calcurse.h
+++ b/src/calcurse.h
@@ -93,6 +93,7 @@
 #define DPID_PATH_NAME   ".daemon.pid"
 #define DLOG_PATH_NAME   "daemon.log"
 #define NOTES_DIR_NAME   "notes/"
+#define HOOKS_DIR_NAME   "hooks/"
 
 #define TODO_PATH        DIR_NAME TODO_PATH_NAME
 #define APTS_PATH        DIR_NAME APTS_PATH_NAME
@@ -102,6 +103,7 @@
 #define DLOG_PATH        DIR_NAME DLOG_PATH_NAME
 #define DPID_PATH        DIR_NAME DPID_PATH_NAME
 #define NOTES_DIR        DIR_NAME NOTES_DIR_NAME
+#define HOOKS_DIR        DIR_NAME HOOKS_DIR_NAME
 
 #define DEFAULT_EDITOR     "vi"
 #define DEFAULT_PAGER      "less"
@@ -779,6 +781,9 @@ int updatestring(WINDOW *, char **, int, int);
 /* help.c */
 int display_help(const char *);
 
+/* hooks.c */
+int run_hook(const char *);
+
 /* ical.c */
 void ical_import_data(FILE *, FILE *, unsigned *, unsigned *, unsigned *,
 		      unsigned *, unsigned *);
@@ -1111,6 +1116,7 @@ extern char path_notes[BUFSIZ];
 extern char path_cpid[BUFSIZ];
 extern char path_dpid[BUFSIZ];
 extern char path_dmon_log[BUFSIZ];
+extern char path_hooks[BUFSIZ];
 extern struct conf conf;
 extern struct pad apad;
 extern struct nbar nbar;
diff --git a/src/hooks.c b/src/hooks.c
new file mode 100644
index 0000000..5fffb70
--- /dev/null
+++ b/src/hooks.c
@@ -0,0 +1,63 @@
+/*
+ * Calcurse - text-based organizer
+ *
+ * Copyright (c) 2004-2016 calcurse Development Team <misc@calcurse.org>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ *      - Redistributions of source code must retain the above
+ *        copyright notice, this list of conditions and the
+ *        following disclaimer.
+ *
+ *      - Redistributions in binary form must reproduce the above
+ *        copyright notice, this list of conditions and the
+ *        following disclaimer in the documentation and/or other
+ *        materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ *
+ * Send your feedback or comments to : misc@calcurse.org
+ * Calcurse home page : http://calcurse.org
+ *
+ */
+
+#include <stddef.h>
+
+#include "calcurse.h"
+
+int run_hook(const char *name)
+{
+	char *hook_path;
+	char const *arg[2];
+	int pid, ret = -127;
+
+	asprintf(&hook_path, "%s/%s", path_hooks, name);
+	arg[0] = hook_path;
+	arg[1] = NULL;
+
+	if (!io_file_exists(hook_path))
+		return 0;
+
+	wins_prepare_external();
+	if ((pid = shell_exec(NULL, NULL, *arg, arg))) {
+		ret = child_wait(NULL, NULL, pid);
+		if (ret)
+			press_any_key();
+	}
+	wins_unprepare_external();
+
+	return ret;
+}
diff --git a/src/io.c b/src/io.c
index f12bac9..7411a6b 100644
--- a/src/io.c
+++ b/src/io.c
@@ -239,6 +239,7 @@ void io_init(const char *cfile, const char *datadir)
 		snprintf(path_dpid, BUFSIZ, "%s/" DPID_PATH_NAME, home);
 		snprintf(path_dmon_log, BUFSIZ, "%s/" DLOG_PATH_NAME,
 			 home);
+		snprintf(path_hooks, BUFSIZ, "%s/" HOOKS_DIR_NAME, home);
 	} else {
 		home = getenv("HOME");
 		if (home == NULL) {
@@ -252,6 +253,7 @@ void io_init(const char *cfile, const char *datadir)
 		snprintf(path_dpid, BUFSIZ, "%s/" DPID_PATH, home);
 		snprintf(path_dmon_log, BUFSIZ, "%s/" DLOG_PATH, home);
 		snprintf(path_notes, BUFSIZ, "%s/" NOTES_DIR, home);
+		snprintf(path_hooks, BUFSIZ, "%s/" HOOKS_DIR, home);
 	}
 
 	if (cfile == NULL) {
@@ -391,6 +393,7 @@ void io_save_cal(enum save_display display)
 	if (read_only)
 		return;
 
+	run_hook("pre-save");
 	pthread_mutex_lock(&io_save_mutex);
 
 	show_bar = 0;
@@ -428,6 +431,7 @@ void io_save_cal(enum save_display display)
 	}
 
 	pthread_mutex_unlock(&io_save_mutex);
+	run_hook("post-save");
 }
 
 static void io_load_error(const char *filename, unsigned line,
diff --git a/src/vars.c b/src/vars.c
index f6fd685..d674393 100644
--- a/src/vars.c
+++ b/src/vars.c
@@ -111,6 +111,7 @@ char path_keys[] = "";
 char path_cpid[] = "";
 char path_dpid[] = "";
 char path_dmon_log[] = "";
+char path_hooks[] = "";
 
 /* Variable to store global configuration. */
 struct conf conf;
-- 
cgit v1.2.3-70-g09d2