uci: unload package on revert
[project/rpcd.git] / file.c
diff --git a/file.c b/file.c
index 9a3dfd8b3d60b0992a1fc3a86db78462034b8356..9c1b301e023d34ff9eeaf3431425b3ca20405aea 100644 (file)
--- a/file.c
+++ b/file.c
@@ -27,6 +27,7 @@
 #include <sys/wait.h>
 #include <libubus.h>
 #include <libubox/blobmsg.h>
+#include <libubox/md5.h>
 #include <libubox/ustream.h>
 
 #include <rpcd/plugin.h>
@@ -221,7 +222,7 @@ rpc_file_write(struct ubus_context *ctx, struct ubus_object *obj,
        if (!tb[RPC_F_RW_PATH] || !tb[RPC_F_RW_DATA])
                return UBUS_STATUS_INVALID_ARGUMENT;
 
-       if ((fd = open(blobmsg_data(tb[RPC_F_RW_PATH]), O_CREAT | O_TRUNC | O_WRONLY)) < 0)
+       if ((fd = open(blobmsg_data(tb[RPC_F_RW_PATH]), O_CREAT | O_TRUNC | O_WRONLY, 0666)) < 0)
                return rpc_errno_status();
 
        if (write(fd, blobmsg_data(tb[RPC_F_RW_DATA]), blobmsg_data_len(tb[RPC_F_RW_DATA])) < 0)
@@ -236,6 +237,39 @@ rpc_file_write(struct ubus_context *ctx, struct ubus_object *obj,
        return 0;
 }
 
+static int
+rpc_file_md5(struct ubus_context *ctx, struct ubus_object *obj,
+             struct ubus_request_data *req, const char *method,
+             struct blob_attr *msg)
+{ 
+       int rv, i;
+       char *path;
+       struct stat s;
+       uint8_t md5[16];
+       char *wbuf;
+
+       if (!rpc_check_path(msg, &path, &s))
+               return rpc_errno_status();
+
+       if (!S_ISREG(s.st_mode))
+               return UBUS_STATUS_NOT_SUPPORTED;
+
+       if ((rv = md5sum(path, md5)) <= 0)
+               return rpc_errno_status();
+
+       blob_buf_init(&buf, 0);
+       wbuf = blobmsg_alloc_string_buffer(&buf, "md5", 33);
+
+       for (i = 0; i < 16; i++)
+               sprintf(wbuf + (i * 2), "%02x", (uint8_t) md5[i]);
+
+       blobmsg_add_string_buffer(&buf);
+       ubus_send_reply(ctx, req, buf.head);
+       blob_buf_free(&buf);
+
+       return UBUS_STATUS_OK;
+}
+
 static int
 rpc_file_list(struct ubus_context *ctx, struct ubus_object *obj,
               struct ubus_request_data *req, const char *method,
@@ -267,6 +301,8 @@ rpc_file_list(struct ubus_context *ctx, struct ubus_object *obj,
                blobmsg_close_table(&buf, d);
        }
 
+       closedir(fd);
+
        blobmsg_close_array(&buf, c);
        ubus_send_reply(ctx, req, buf.head);
        blob_buf_free(&buf);
@@ -609,6 +645,7 @@ rpc_file_api_init(const struct rpc_daemon_ops *o, struct ubus_context *ctx)
                UBUS_METHOD("write",   rpc_file_write, rpc_file_rw_policy),
                UBUS_METHOD("list",    rpc_file_list,  rpc_file_r_policy),
                UBUS_METHOD("stat",    rpc_file_stat,  rpc_file_r_policy),
+               UBUS_METHOD("md5",     rpc_file_md5,   rpc_file_r_policy),
                UBUS_METHOD("exec",    rpc_file_exec,  rpc_exec_policy),
        };