- Add startup-config(5) manpage.
[l2tpns.git] / snoopctl.c
diff --git a/snoopctl.c b/snoopctl.c
new file mode 100644 (file)
index 0000000..3d7e120
--- /dev/null
@@ -0,0 +1,138 @@
+#include <string.h>
+#include "l2tpns.h"
+#include "plugin.h"
+#include "control.h"
+
+/* snoop control */
+
+char const *cvs_id = "$Id: snoopctl.c,v 1.1 2004-11-17 15:08:19 bodea Exp $";
+
+int plugin_api_version = PLUGIN_API_VERSION;
+static struct pluginfuncs *p = 0;
+
+char *plugin_control_help[] = {
+       "  snoop USER|SID IP PORT                      Intercept user traffic",
+       "  unsnoop USER|SID                            Stop intercepting user",
+       0
+};
+
+static int iam_master = 0;
+
+int plugin_init(struct pluginfuncs *funcs)
+{
+       if (!funcs)
+               return 0;
+
+       p = funcs;
+       return 1;
+}
+
+int plugin_become_master(void)
+{
+       iam_master = 1;
+       return PLUGIN_RET_OK;
+}
+
+int plugin_control(struct param_control *data)
+{
+       sessionidt session;
+       sessiont *s = 0;
+       int flag;
+       char *end;
+
+       if (data->argc < 1)
+               return PLUGIN_RET_OK;
+
+       if (strcmp(data->argv[0], "snoop") && strcmp(data->argv[0], "unsnoop"))
+               return PLUGIN_RET_OK; // not for us
+
+       flag = data->argv[0][0] != 'u';
+
+       if (!iam_master)
+       {
+               data->response = NSCTL_RES_ERR;
+               data->additional = "must be run on the cluster master";
+               return PLUGIN_RET_STOP;
+       }
+
+       if (flag)
+       {
+               if (data->argc != 4)
+               {
+                       data->response = NSCTL_RES_ERR;
+                       data->additional = "requires username or session id and host, port";
+                       return PLUGIN_RET_STOP;
+               }
+       }
+       else
+       {
+               if (data->argc != 2)
+               {
+                       data->response = NSCTL_RES_ERR;
+                       data->additional = "requires username or session id";
+                       return PLUGIN_RET_STOP;
+               }
+       }
+
+       if (!(session = strtol(data->argv[1], &end, 10)) || *end)
+               session = p->get_session_by_username(data->argv[1]);
+
+       if (session)
+               s = p->get_session_by_id(session);
+
+       if (!s || !s->ip)
+       {
+               data->response = NSCTL_RES_ERR;
+               data->additional = "session not found";
+               return PLUGIN_RET_STOP;
+       }
+
+       if (flag)
+       {
+               ipt ip = inet_addr(data->argv[2]);
+               u16 port = atoi(data->argv[3]);
+
+               if (!ip || ip == INADDR_NONE)
+               {
+                       data->response = NSCTL_RES_ERR;
+                       data->additional = "invalid ip address";
+                       return PLUGIN_RET_STOP;
+               }
+
+               if (!port)
+               {
+                       data->response = NSCTL_RES_ERR;
+                       data->additional = "invalid port";
+                       return PLUGIN_RET_STOP;
+               }
+
+               if (ip == s->snoop_ip && port == s->snoop_port)
+               {
+                       data->response = NSCTL_RES_ERR;
+                       data->additional = "already intercepted";
+                       return PLUGIN_RET_STOP;
+               }
+
+               s->snoop_ip = ip;
+               s->snoop_port = port;
+       }
+       else
+       {
+               if (!s->snoop_ip)
+               {
+                       data->response = NSCTL_RES_ERR;
+                       data->additional = "not intercepted";
+                       return PLUGIN_RET_STOP;
+               }
+
+               s->snoop_ip = 0;
+               s->snoop_port = 0;
+       }
+
+       p->sesssion_changed(session);
+
+       data->response = NSCTL_RES_OK;
+       data->additional = 0;
+
+       return PLUGIN_RET_STOP;
+}