rpcd: implement luci2.network.device_list to find names and types of all currently...
[project/luci2/ui.git] / luci2 / src / rpcd / luci2.c
index f136c56..a736dd8 100644 (file)
@@ -16,6 +16,8 @@
  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  */
 
+#define _GNU_SOURCE /* crypt() */
+
 #include <fcntl.h>
 #include <errno.h>
 #include <unistd.h>
@@ -31,6 +33,7 @@
 #include <signal.h>
 #include <glob.h>
 #include <libubox/blobmsg_json.h>
+#include <libubox/avl-cmp.h>
 #include <libubus.h>
 #include <uci.h>
 
@@ -41,7 +44,7 @@
 #define RPC_LUCI2_DEF_LOGSIZE       (16 * 1024)
 
 /* location of menu definitions */
-#define RPC_LUCI2_MENU_FILES        "/usr/share/luci2/menu.d/*.json" /* */
+#define RPC_LUCI2_MENU_FILES        "/usr/share/rpcd/menu.d/*.json" /* */
 
 
 static const struct rpc_daemon_ops *ops;
@@ -141,6 +144,15 @@ static const struct blobmsg_policy rpc_menu_policy[__RPC_MENU_MAX] = {
                                                  .type = BLOBMSG_TYPE_STRING },
 };
 
+enum {
+       RPC_SWITCH_NAME,
+       __RPC_SWITCH_MAX
+};
+
+static const struct blobmsg_policy rpc_switch_policy[__RPC_SWITCH_MAX] = {
+       [RPC_SWITCH_NAME]  = { .name = "switch",  .type = BLOBMSG_TYPE_STRING },
+};
+
 
 static int
 rpc_errno_status(void)
@@ -1737,6 +1749,528 @@ rpc_luci2_network_routes6(struct ubus_context *ctx, struct ubus_object *obj,
 }
 
 
+struct swconfig_state {
+       bool open;
+       void *array;
+       bool open2;
+       void *array2;
+       int port;
+};
+
+static int
+swconfig_parse_list(struct blob_buf *blob, char *buf, int len, void *priv)
+{
+       char *p;
+       char *nl = strchr(buf, '\n');
+       struct swconfig_state *s = priv;
+
+       if (!nl)
+               return 0;
+
+       if (!s->open)
+       {
+               s->open = true;
+               s->array = blobmsg_open_array(blob, "switches");
+       }
+
+       strtok(buf, "-");
+       p = strtok(NULL, " \n");
+
+       if (p)
+               blobmsg_add_string(blob, NULL, p);
+
+       return (nl - buf + 1);
+}
+
+static int
+swconfig_finish_list(struct blob_buf *blob, int status, void *priv)
+{
+       struct swconfig_state *s = priv;
+
+       if (!s->open)
+               return UBUS_STATUS_NO_DATA;
+
+       blobmsg_close_array(blob, s->array);
+
+       return UBUS_STATUS_OK;
+}
+
+static int
+rpc_luci2_network_sw_list(struct ubus_context *ctx, struct ubus_object *obj,
+                          struct ubus_request_data *req, const char *method,
+                          struct blob_attr *msg)
+{
+       struct swconfig_state *state = NULL;
+       const char *cmd[3] = { "swconfig", "list", NULL };
+
+       state = malloc(sizeof(*state));
+
+       if (!state)
+               return UBUS_STATUS_UNKNOWN_ERROR;
+
+       memset(state, 0, sizeof(*state));
+
+       return ops->exec(cmd, NULL, swconfig_parse_list, NULL, swconfig_finish_list,
+                        state, ctx, req);
+}
+
+
+static int
+swconfig_parse_help(struct blob_buf *blob, char *buf, int len, void *priv)
+{
+       void *c;
+       char *p;
+       char *nl = strchr(buf, '\n');
+       struct swconfig_state *s = priv;
+
+       if (!nl)
+               return 0;
+
+       if (!s->open)
+       {
+               s->open = true;
+               s->array = blobmsg_open_table(blob, "info");
+       }
+
+       switch (*buf)
+       {
+       case ' ':
+               strtok(buf, "-");
+               p = strtok(NULL, "-\n");
+
+               if (p)
+               {
+                       if (s->open2)
+                               blobmsg_close_array(blob, s->array2);
+
+                       s->array2 = blobmsg_open_array(blob, p);
+                       s->open2 = true;
+               }
+
+               break;
+
+       case '\t':
+               c = blobmsg_open_table(blob, NULL);
+
+               strtok(buf, "(");
+               p = strtok(NULL, ")");
+
+               if (p)
+                       blobmsg_add_string(blob, "type", p);
+
+               p = strtok(NULL, ":( ");
+
+               if (p)
+                       blobmsg_add_string(blob, "name", p);
+
+               p = strtok(NULL, "\n");
+               *(nl - 1) = 0;
+
+               if (p)
+                       blobmsg_add_string(blob, "description", p + 1);
+
+               blobmsg_close_table(blob, c);
+               break;
+
+       default:
+               strtok(buf, "(");
+               p = strtok(NULL, ")");
+
+               if (p)
+                       blobmsg_add_string(blob, "model", p);
+
+               strtok(NULL, ":");
+               p = strtok(NULL, "(");
+
+               if (p)
+                       blobmsg_add_u32(blob, "num_ports", atoi(p));
+
+               strtok(NULL, "@");
+               p = strtok(NULL, ")");
+
+               if (p)
+                       blobmsg_add_u32(blob, "cpu_port", atoi(p));
+
+               strtok(NULL, ":");
+               p = strtok(NULL, "\n");
+
+               if (p)
+                       blobmsg_add_u32(blob, "num_vlans", atoi(p));
+
+               break;
+       }
+
+       return (nl - buf + 1);
+}
+
+static int
+swconfig_finish_help(struct blob_buf *blob, int status, void *priv)
+{
+       struct swconfig_state *s = priv;
+
+       if (!s->open)
+               return UBUS_STATUS_NO_DATA;
+
+       if (s->open2)
+               blobmsg_close_array(blob, s->array2);
+
+       blobmsg_close_table(blob, s->array);
+
+       return UBUS_STATUS_OK;
+}
+
+static int
+rpc_luci2_network_sw_info(struct ubus_context *ctx, struct ubus_object *obj,
+                          struct ubus_request_data *req, const char *method,
+                          struct blob_attr *msg)
+{
+       struct swconfig_state *state = NULL;
+       struct blob_attr *tb[__RPC_SWITCH_MAX];
+       const char *cmd[5] = { "swconfig", "dev", NULL, "help", NULL };
+
+       blobmsg_parse(rpc_switch_policy, __RPC_SWITCH_MAX, tb,
+                     blob_data(msg), blob_len(msg));
+
+       if (!tb[RPC_SWITCH_NAME])
+               return UBUS_STATUS_INVALID_ARGUMENT;
+
+       state = malloc(sizeof(*state));
+
+       if (!state)
+               return UBUS_STATUS_UNKNOWN_ERROR;
+
+       memset(state, 0, sizeof(*state));
+
+       cmd[2] = blobmsg_get_string(tb[RPC_SWITCH_NAME]);
+
+       return ops->exec(cmd, NULL, swconfig_parse_help, NULL, swconfig_finish_help,
+                        state, ctx, req);
+}
+
+
+static void
+swconfig_parse_link(struct blob_buf *blob, char *val)
+{
+       char *p;
+
+       int speed = 0;
+
+       bool rxflow = false;
+       bool txflow = false;
+       bool duplex = false;
+       bool aneg = false;
+       bool up = false;
+
+       for (p = strtok(val, " "); p; p = strtok(NULL, " "))
+       {
+               if (!strncmp(p, "speed:", 6))
+                       speed = atoi(p + 6);
+               else if (!strcmp(p, "link:up"))
+                       up = true;
+               else if (!strcmp(p, "txflow"))
+                       txflow = true;
+               else if (!strcmp(p, "rxflow"))
+                       rxflow = true;
+               else if (!strcmp(p, "full-duplex"))
+                       duplex = true;
+               else if (!strcmp(p, "auto"))
+                       aneg = true;
+       }
+
+       blobmsg_add_u8(blob, "link",             up);
+       blobmsg_add_u8(blob, "rx_flow_control",  rxflow);
+       blobmsg_add_u8(blob, "tx_flow_control",  txflow);
+       blobmsg_add_u8(blob, "full_duplex",      duplex);
+       blobmsg_add_u8(blob, "auto_negotiation", aneg);
+       blobmsg_add_u32(blob, "speed",           speed);
+}
+
+static int
+swconfig_parse_stat(struct blob_buf *blob, char *buf, int len, void *priv)
+{
+       char *p, *v;
+       char *nl = strchr(buf, '\n');
+       struct swconfig_state *s = priv;
+
+       if (!nl)
+               return 0;
+
+       if (nl == buf)
+               return 1;
+
+       if (!s->open)
+       {
+               s->open = true;
+               s->array = blobmsg_open_array(blob, "ports");
+       }
+
+       p = strtok(buf, " :\t");
+
+       if (p)
+       {
+               if (!strcmp(p, "Port"))
+               {
+                       if (s->open2)
+                               blobmsg_close_table(blob, s->array2);
+
+                       s->array2 = blobmsg_open_table(blob, NULL);
+                       s->open2 = true;
+               }
+               else if (s->open2)
+               {
+                       v = strtok(NULL, "\n");
+
+                       if (v)
+                       {
+                               if (!strcmp(p, "link"))
+                                       swconfig_parse_link(blob, v);
+                       }
+               }
+       }
+
+       return (nl - buf + 1);
+}
+
+static int
+swconfig_finish_stat(struct blob_buf *blob, int status, void *priv)
+{
+       struct swconfig_state *s = priv;
+
+       if (!s->open)
+               return UBUS_STATUS_NO_DATA;
+
+       if (s->open2)
+               blobmsg_close_table(blob, s->array2);
+
+       blobmsg_close_array(blob, s->array);
+
+       return UBUS_STATUS_OK;
+}
+
+static int
+rpc_luci2_network_sw_status(struct ubus_context *ctx, struct ubus_object *obj,
+                            struct ubus_request_data *req, const char *method,
+                            struct blob_attr *msg)
+{
+       struct swconfig_state *state = NULL;
+       struct blob_attr *tb[__RPC_SWITCH_MAX];
+       const char *cmd[5] = { "swconfig", "dev", NULL, "show", NULL };
+
+       blobmsg_parse(rpc_switch_policy, __RPC_SWITCH_MAX, tb,
+                     blob_data(msg), blob_len(msg));
+
+       if (!tb[RPC_SWITCH_NAME])
+               return UBUS_STATUS_INVALID_ARGUMENT;
+
+       state = malloc(sizeof(*state));
+
+       if (!state)
+               return UBUS_STATUS_UNKNOWN_ERROR;
+
+       memset(state, 0, sizeof(*state));
+
+       cmd[2] = blobmsg_get_string(tb[RPC_SWITCH_NAME]);
+
+       return ops->exec(cmd, NULL, swconfig_parse_stat, NULL, swconfig_finish_stat,
+                        state, ctx, req);
+}
+
+enum {
+       NETWORK_CMD_PING,
+       NETWORK_CMD_PING6,
+       NETWORK_CMD_TRACEROUTE,
+       NETWORK_CMD_TRACEROUTE6,
+       NETWORK_CMD_NSLOOKUP
+};
+
+static int
+network_cmd(struct ubus_context *ctx, struct ubus_request_data *req,
+            struct blob_attr *msg, int which)
+{
+       char *arg;
+       struct blob_attr *tb[__RPC_D_MAX];
+
+       blobmsg_parse(rpc_data_policy, __RPC_D_MAX, tb,
+                     blob_data(msg), blob_len(msg));
+
+       if (!tb[RPC_D_DATA])
+               return UBUS_STATUS_INVALID_ARGUMENT;
+
+       arg = blobmsg_get_string(tb[RPC_D_DATA]);
+
+       const char *cmds[][8] = {
+               [NETWORK_CMD_PING] = {
+                       "ping", "-c", "5", "-W", "1", arg
+               },
+               [NETWORK_CMD_PING6] = {
+                       "ping6", "-c", "5", "-W", "1", arg
+               },
+               [NETWORK_CMD_TRACEROUTE] = {
+                       "traceroute", "-q", "1", "-w", "1", "-n", arg
+               },
+               [NETWORK_CMD_TRACEROUTE6] = {
+                       "traceroute6", "-q", "1", "-w", "2", "-n", arg
+               },
+               [NETWORK_CMD_NSLOOKUP] = {
+                       "nslookup", arg
+               }
+       };
+
+       return ops->exec(cmds[which], NULL, NULL, NULL, NULL, NULL, ctx, req);
+}
+
+static int
+rpc_luci2_network_ping(struct ubus_context *ctx, struct ubus_object *obj,
+                       struct ubus_request_data *req, const char *method,
+                       struct blob_attr *msg)
+{
+       return network_cmd(ctx, req, msg, NETWORK_CMD_PING);
+}
+
+static int
+rpc_luci2_network_ping6(struct ubus_context *ctx, struct ubus_object *obj,
+                        struct ubus_request_data *req, const char *method,
+                        struct blob_attr *msg)
+{
+       return network_cmd(ctx, req, msg, NETWORK_CMD_PING6);
+}
+
+static int
+rpc_luci2_network_traceroute(struct ubus_context *ctx, struct ubus_object *obj,
+                             struct ubus_request_data *req, const char *method,
+                             struct blob_attr *msg)
+{
+       return network_cmd(ctx, req, msg, NETWORK_CMD_TRACEROUTE);
+}
+
+static int
+rpc_luci2_network_traceroute6(struct ubus_context *ctx, struct ubus_object *obj,
+                              struct ubus_request_data *req, const char *method,
+                              struct blob_attr *msg)
+{
+       return network_cmd(ctx, req, msg, NETWORK_CMD_TRACEROUTE6);
+}
+
+static int
+rpc_luci2_network_nslookup(struct ubus_context *ctx, struct ubus_object *obj,
+                           struct ubus_request_data *req, const char *method,
+                           struct blob_attr *msg)
+{
+       return network_cmd(ctx, req, msg, NETWORK_CMD_NSLOOKUP);
+}
+
+
+static int
+network_ifupdown(struct ubus_context *ctx, struct ubus_request_data *req,
+                 struct blob_attr *msg, bool up)
+{
+       const char *cmd[3] = { NULL };
+       struct blob_attr *tb[__RPC_D_MAX];
+
+       blobmsg_parse(rpc_data_policy, __RPC_D_MAX, tb,
+                     blob_data(msg), blob_len(msg));
+
+       if (!tb[RPC_D_DATA])
+               return UBUS_STATUS_INVALID_ARGUMENT;
+
+       cmd[0] = up ? "/sbin/ifup" : "/sbin/ifdown";
+       cmd[1] = blobmsg_get_string(tb[RPC_D_DATA]);
+
+       return ops->exec(cmd, NULL, NULL, NULL, NULL, NULL, ctx, req);
+}
+
+static int
+rpc_luci2_network_ifup(struct ubus_context *ctx, struct ubus_object *obj,
+                       struct ubus_request_data *req, const char *method,
+                       struct blob_attr *msg)
+{
+       return network_ifupdown(ctx, req, msg, true);
+}
+
+static int
+rpc_luci2_network_ifdown(struct ubus_context *ctx, struct ubus_object *obj,
+                         struct ubus_request_data *req, const char *method,
+                         struct blob_attr *msg)
+{
+       return network_ifupdown(ctx, req, msg, false);
+}
+
+static int
+rpc_luci2_network_dev_list(struct ubus_context *ctx, struct ubus_object *obj,
+                           struct ubus_request_data *req, const char *method,
+                           struct blob_attr *msg)
+{
+       char path[PATH_MAX];
+       struct dirent *e;
+       struct stat s;
+       void *c, *t;
+       bool wireless, bridge, tuntap;
+       int type;
+       DIR *d;
+       FILE *f;
+
+       if (!(d = opendir("/sys/class/net")))
+               return rpc_errno_status();
+
+       blob_buf_init(&buf, 0);
+       c = blobmsg_open_array(&buf, "devices");
+
+       while ((e = readdir(d)) != NULL)
+       {
+               snprintf(path, sizeof(path) - 1, "/sys/class/net/%s/type", e->d_name);
+
+               if (stat(path, &s) || !S_ISREG(s.st_mode) || !(f = fopen(path, "r")))
+                       continue;
+
+               type = 1;
+               memset(path, 0, sizeof(path));
+
+               if (fread(path, 1, sizeof(path) - 1, f) > 0)
+                       type = atoi(path);
+
+               fclose(f);
+
+
+               snprintf(path, sizeof(path) - 1,
+                        "/sys/class/net/%s/wireless", e->d_name);
+
+               wireless = (!stat(path, &s) && S_ISDIR(s.st_mode));
+
+               snprintf(path, sizeof(path) - 1,
+                        "/sys/class/net/%s/phy80211", e->d_name);
+
+               wireless = (wireless || (!stat(path, &s) && S_ISLNK(s.st_mode)));
+
+               snprintf(path, sizeof(path) - 1,
+                        "/sys/class/net/%s/bridge", e->d_name);
+
+               bridge = (!stat(path, &s) && S_ISDIR(s.st_mode));
+
+               snprintf(path, sizeof(path) - 1,
+                        "/sys/class/net/%s/tun_flags", e->d_name);
+
+               tuntap = (!stat(path, &s) && S_ISREG(s.st_mode));
+
+               t = blobmsg_open_table(&buf, NULL);
+
+               blobmsg_add_string(&buf, "device", e->d_name);
+               blobmsg_add_u32(&buf, "type", type);
+               blobmsg_add_u8(&buf, "is_bridge", bridge);
+               blobmsg_add_u8(&buf, "is_tuntap", tuntap);
+               blobmsg_add_u8(&buf, "is_wireless", wireless);
+
+               blobmsg_close_table(&buf, t);
+       }
+
+       blobmsg_close_array(&buf, c);
+
+       closedir(d);
+
+       ubus_send_reply(ctx, req, buf.head);
+       return 0;
+}
+
+
 struct opkg_state {
        int cur_offset;
        int cur_count;
@@ -1998,7 +2532,7 @@ menu_access(struct blob_attr *sid, struct blob_attr *acls, struct blob_buf *e)
 
        blobmsg_for_each_attr(acl, acls, rem)
        {
-               if (!ops->session_access(blobmsg_data(sid), "luci-ui",
+               if (!ops->session_access(blobmsg_data(sid), "access-group",
                                         blobmsg_data(acl), "read"))
                {
                        rv = false;
@@ -2006,7 +2540,7 @@ menu_access(struct blob_attr *sid, struct blob_attr *acls, struct blob_buf *e)
                }
 
                blobmsg_add_u8(e, blobmsg_data(acl),
-                              ops->session_access(blobmsg_data(sid), "luci-ui",
+                              ops->session_access(blobmsg_data(sid), "access-group",
                                                   blobmsg_data(acl), "write"));
        }
 
@@ -2087,6 +2621,78 @@ skip:
 }
 
 
+static void
+parse_acl_file(struct blob_buf *acls, const char *path)
+{
+       struct blob_buf acl = { 0 };
+       struct blob_attr *cur;
+       void *c;
+       int rem;
+
+       blob_buf_init(&acl, 0);
+
+       if (blobmsg_add_json_from_file(&acl, path))
+       {
+               c = blobmsg_open_table(acls, NULL);
+
+               blob_for_each_attr(cur, acl.head, rem)
+                       blobmsg_add_blob(acls, cur);
+
+               blobmsg_close_table(acls, c);
+       }
+
+       blob_buf_free(&acl);
+}
+
+static int
+rpc_luci2_ui_acls(struct ubus_context *ctx, struct ubus_object *obj,
+                  struct ubus_request_data *req, const char *method,
+                  struct blob_attr *msg)
+{
+       int i;
+       void *c;
+       glob_t gl;
+
+       if (glob(RPC_SESSION_ACL_DIR "/*.json", 0, NULL, &gl))
+               return rpc_errno_status();
+
+       blob_buf_init(&buf, 0);
+       c = blobmsg_open_array(&buf, "acls");
+
+       for (i = 0; i < gl.gl_pathc; i++)
+               parse_acl_file(&buf, gl.gl_pathv[i]);
+
+       globfree(&gl);
+       blobmsg_close_array(&buf, c);
+
+       ubus_send_reply(ctx, req, buf.head);
+       return 0;
+}
+
+static int
+rpc_luci2_ui_crypt(struct ubus_context *ctx, struct ubus_object *obj,
+                   struct ubus_request_data *req, const char *method,
+                   struct blob_attr *msg)
+{
+       char *hash;
+       struct blob_attr *tb[__RPC_D_MAX];
+
+       blobmsg_parse(rpc_data_policy, __RPC_D_MAX, tb,
+                     blob_data(msg), blob_len(msg));
+
+       if (!tb[RPC_D_DATA] || blobmsg_data_len(tb[RPC_D_DATA]) >= 128)
+               return UBUS_STATUS_INVALID_ARGUMENT;
+
+       hash = crypt(blobmsg_get_string(tb[RPC_D_DATA]), "$1$");
+
+       blob_buf_init(&buf, 0);
+       blobmsg_add_string(&buf, "crypt", hash);
+
+       ubus_send_reply(ctx, req, buf.head);
+       return 0;
+}
+
+
 static int
 rpc_luci2_api_init(const struct rpc_daemon_ops *o, struct ubus_context *ctx)
 {
@@ -2149,6 +2755,26 @@ rpc_luci2_api_init(const struct rpc_daemon_ops *o, struct ubus_context *ctx)
                UBUS_METHOD_NOARG("dhcp6_leases",    rpc_luci2_network_leases6),
                UBUS_METHOD_NOARG("routes",          rpc_luci2_network_routes),
                UBUS_METHOD_NOARG("routes6",         rpc_luci2_network_routes6),
+               UBUS_METHOD_NOARG("switch_list",     rpc_luci2_network_sw_list),
+               UBUS_METHOD("switch_info",           rpc_luci2_network_sw_info,
+                                                    rpc_switch_policy),
+               UBUS_METHOD("switch_status",         rpc_luci2_network_sw_status,
+                                                    rpc_switch_policy),
+               UBUS_METHOD("ping",                  rpc_luci2_network_ping,
+                                                    rpc_data_policy),
+               UBUS_METHOD("ping6",                 rpc_luci2_network_ping6,
+                                                    rpc_data_policy),
+               UBUS_METHOD("traceroute",            rpc_luci2_network_traceroute,
+                                                    rpc_data_policy),
+               UBUS_METHOD("traceroute6",           rpc_luci2_network_traceroute6,
+                                                    rpc_data_policy),
+               UBUS_METHOD("nslookup",              rpc_luci2_network_nslookup,
+                                                    rpc_data_policy),
+               UBUS_METHOD("ifup",                  rpc_luci2_network_ifup,
+                                                    rpc_data_policy),
+               UBUS_METHOD("ifdown",                rpc_luci2_network_ifdown,
+                                                    rpc_data_policy),
+               UBUS_METHOD_NOARG("device_list",     rpc_luci2_network_dev_list)
        };
 
        static struct ubus_object_type luci2_network_type =
@@ -2191,7 +2817,10 @@ rpc_luci2_api_init(const struct rpc_daemon_ops *o, struct ubus_context *ctx)
 
 
        static const struct ubus_method luci2_ui_methods[] = {
-               UBUS_METHOD_NOARG("menu",            rpc_luci2_ui_menu)
+               UBUS_METHOD_NOARG("menu",            rpc_luci2_ui_menu),
+               UBUS_METHOD_NOARG("acls",            rpc_luci2_ui_acls),
+               UBUS_METHOD("crypt",                 rpc_luci2_ui_crypt,
+                                                    rpc_data_policy)
        };
 
        static struct ubus_object_type luci2_ui_type =