[PATCH RFC nftables 4/4] cli: add linenoise CLI implementation.

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



By default, continue to use libreadline, but if `--with-cli=linenoise`
is passed to configure, build the linenoise implementation instead.

Signed-off-by: Jeremy Sowden <jeremy@xxxxxxxxxx>
---
 Makefile.am                               |  7 ++-
 configure.ac                              | 18 +++++--
 include/cli.h                             |  2 +-
 linenoise/.gitignore                      |  6 +++
 linenoise/Makefile.am                     | 13 +++++
 linenoise/{Makefile => Makefile.upstream} |  0
 src/Makefile.am                           |  6 +++
 src/cli.c                                 | 64 +++++++++++++++++++----
 8 files changed, 101 insertions(+), 15 deletions(-)
 create mode 100644 linenoise/.gitignore
 create mode 100644 linenoise/Makefile.am
 rename linenoise/{Makefile => Makefile.upstream} (100%)

diff --git a/Makefile.am b/Makefile.am
index 4a17424d45b8..0095e1958482 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -1,6 +1,11 @@
 ACLOCAL_AMFLAGS	= -I m4
 
-SUBDIRS = 	src	\
+if BUILD_CLI_LINENOISE
+SUBDIRS = linenoise
+else
+SUBDIRS =
+endif
+SUBDIRS += 	src	\
 		include	\
 		files	\
 		doc
diff --git a/configure.ac b/configure.ac
index 68f97f090535..347f3b0cc772 100644
--- a/configure.ac
+++ b/configure.ac
@@ -68,14 +68,23 @@ AC_CHECK_LIB([gmp],[__gmpz_init], , AC_MSG_ERROR([No suitable version of libgmp
 AM_CONDITIONAL([BUILD_MINIGMP], [test "x$with_mini_gmp" = xyes])
 
 AC_ARG_WITH([cli], [AS_HELP_STRING([--without-cli],
-            [disable interactive CLI (libreadline support)])],
-            [], [with_cli=yes])
-AS_IF([test "x$with_cli" != xno], [
+            [disable interactive CLI (libreadline or linenoise support)])],
+            [], [with_cli=readline])
+
+AS_IF([test "x$with_cli" = xreadline], [
 AC_CHECK_LIB([readline], [readline], ,
-	     AC_MSG_ERROR([No suitable version of libreadline found]))
+        AC_MSG_ERROR([No suitable version of libreadline found]))
 AC_DEFINE([HAVE_LIBREADLINE], [1], [])
+],
+      [test "x$with_cli" = xlinenoise], [
+AH_TEMPLATE([HAVE_LINENOISE], [])
+AC_DEFINE([HAVE_LINENOISE], [1], [])
+],
+      [test "x$with_cli" != xno], [
+AC_MSG_ERROR([unexpected CLI value: $with_cli])
 ])
 AM_CONDITIONAL([BUILD_CLI], [test "x$with_cli" != xno])
+AM_CONDITIONAL([BUILD_CLI_LINENOISE], [test "x$with_cli" = xlinenoise])
 
 AC_ARG_WITH([xtables], [AS_HELP_STRING([--with-xtables],
             [Use libxtables for iptables interaction])],
@@ -118,6 +127,7 @@ AM_CONDITIONAL([HAVE_PYTHON], [test "$enable_python" != "no"])
 AC_CONFIG_FILES([					\
 		Makefile				\
 		libnftables.pc				\
+		linenoise/Makefile			\
 		src/Makefile				\
 		include/Makefile			\
 		include/nftables/Makefile		\
diff --git a/include/cli.h b/include/cli.h
index 023f004b8dab..ea1bd3267d64 100644
--- a/include/cli.h
+++ b/include/cli.h
@@ -4,7 +4,7 @@
 #include <nftables/libnftables.h>
 #include <config.h>
 
-#ifdef HAVE_LIBREADLINE
+#if defined(HAVE_LIBREADLINE) || defined(HAVE_LINENOISE)
 extern int cli_init(struct nft_ctx *nft);
 #else
 static inline int cli_init(struct nft_ctx *nft)
diff --git a/linenoise/.gitignore b/linenoise/.gitignore
new file mode 100644
index 000000000000..38bf97184f38
--- /dev/null
+++ b/linenoise/.gitignore
@@ -0,0 +1,6 @@
+*.la
+*.lo
+*.o
+.deps/
+.libs/
+linenoise_example
diff --git a/linenoise/Makefile.am b/linenoise/Makefile.am
new file mode 100644
index 000000000000..599db790414a
--- /dev/null
+++ b/linenoise/Makefile.am
@@ -0,0 +1,13 @@
+include $(top_srcdir)/Make_global.am
+
+AM_CFLAGS = -Wall -W
+
+noinst_LTLIBRARIES = liblinenoise.la
+
+liblinenoise_la_SOURCES = linenoise.c
+
+noinst_PROGRAMS = linenoise_example
+
+linenoise_example_SOURCES = example.c
+
+linenoise_example_LDADD = liblinenoise.la
diff --git a/linenoise/Makefile b/linenoise/Makefile.upstream
similarity index 100%
rename from linenoise/Makefile
rename to linenoise/Makefile.upstream
diff --git a/src/Makefile.am b/src/Makefile.am
index 740c21f2cac8..0d8a750c461f 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -13,6 +13,9 @@ endif
 if BUILD_XTABLES
 AM_CPPFLAGS += ${XTABLES_CFLAGS}
 endif
+if BUILD_CLI_LINENOISE
+AM_CPPFLAGS += -I$(top_srcdir)/linenoise
+endif
 
 AM_CFLAGS = -Wall								\
 	    -Wstrict-prototypes -Wmissing-prototypes -Wmissing-declarations	\
@@ -106,3 +109,6 @@ libnftables_la_LIBADD += ${JANSSON_LIBS}
 endif
 
 nft_LDADD = libnftables.la
+if BUILD_CLI_LINENOISE
+nft_LDADD += $(top_srcdir)/linenoise/liblinenoise.la
+endif
diff --git a/src/cli.c b/src/cli.c
index f1e89926f2bc..4c0c3e9d67c6 100644
--- a/src/cli.c
+++ b/src/cli.c
@@ -21,16 +21,36 @@
 #include <string.h>
 #include <ctype.h>
 #include <limits.h>
+#ifdef HAVE_LIBREADLINE
 #include <readline/readline.h>
 #include <readline/history.h>
+#else
+#include <linenoise.h>
+#endif
 
 #include <cli.h>
 #include <list.h>
 
 #define CMDLINE_HISTFILE	".nft.history"
+#define CMDLINE_PROMPT		"nft> "
+#define CMDLINE_QUIT		"quit"
 
-static struct nft_ctx *cli_nft;
 static char histfile[PATH_MAX];
+
+static void
+init_histfile(void)
+{
+	const char *home;
+
+	home = getenv("HOME");
+	if (home == NULL)
+		home = "";
+	snprintf(histfile, sizeof(histfile), "%s/%s", home, CMDLINE_HISTFILE);
+}
+
+#ifdef HAVE_LIBREADLINE
+
+static struct nft_ctx *cli_nft;
 static char *multiline;
 
 static char *cli_append_multiline(char *line)
@@ -100,7 +120,7 @@ static void cli_complete(char *line)
 	if (*c == '\0')
 		return;
 
-	if (!strcmp(line, "quit")) {
+	if (!strcmp(line, CMDLINE_QUIT)) {
 		cli_exit();
 		exit(0);
 	}
@@ -121,20 +141,15 @@ static char **cli_completion(const char *text, int start, int end)
 
 int cli_init(struct nft_ctx *nft)
 {
-	const char *home;
-
 	cli_nft = nft;
 	rl_readline_name = "nft";
 	rl_instream  = stdin;
 	rl_outstream = stdout;
 
-	rl_callback_handler_install("nft> ", cli_complete);
+	rl_callback_handler_install(CMDLINE_PROMPT, cli_complete);
 	rl_attempted_completion_function = cli_completion;
 
-	home = getenv("HOME");
-	if (home == NULL)
-		home = "";
-	snprintf(histfile, sizeof(histfile), "%s/%s", home, CMDLINE_HISTFILE);
+	init_histfile();
 
 	read_history(histfile);
 	history_set_pos(history_length);
@@ -150,3 +165,34 @@ void cli_exit(void)
 	rl_deprep_terminal();
 	write_history(histfile);
 }
+
+#else /* !HAVE_LIBREADLINE */
+
+int cli_init(struct nft_ctx *nft)
+{
+	int quit = 0;
+	char *line;
+
+	init_histfile();
+	linenoiseHistoryLoad(histfile);
+	linenoiseSetMultiLine(1);
+
+	while (!quit && (line = linenoise(CMDLINE_PROMPT)) != NULL) {
+		if (strcmp(line, CMDLINE_QUIT) == 0) {
+			quit = 1;
+		} else if (line[0] != '\0') {
+			linenoiseHistoryAdd(line);
+			nft_run_cmd_from_buffer(nft, line);
+		}
+		linenoiseFree(line);
+	}
+	cli_exit();
+	exit(0);
+}
+
+void cli_exit(void)
+{
+	linenoiseHistorySave(histfile);
+}
+
+#endif /* HAVE_LIBREADLINE */
-- 
2.23.0




[Index of Archives]     [Netfitler Users]     [Berkeley Packet Filter]     [LARTC]     [Bugtraq]     [Yosemite Forum]

  Powered by Linux