4bc023445b2c1d59114aaaa4b8745e6e981019a3
[project/uci.git] / cli.c
1 /*
2 * Copyright (C) 2008 Felix Fietkau <nbd@openwrt.org>
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2
6 * as published by the Free Software Foundation
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 */
13 #include <strings.h>
14 #include <stdlib.h>
15 #include "uci.h"
16
17 static struct uci_context *ctx;
18 static char *buf = NULL;
19 static int buflen = 256;
20
21 static void uci_usage(int argc, char **argv)
22 {
23 fprintf(stderr,
24 "Usage: %s [<options>] <command> [<arguments>]\n\n"
25 "Commands:\n"
26 "\tshow [<config>[.<section>[.<option>]]]\n"
27 "\texport [<config>]\n"
28 "\n",
29 argv[0]
30 );
31 exit(255);
32 }
33
34 static char *uci_escape(char *str)
35 {
36 char *s, *p, *t;
37 int pos = 0;
38
39 if (!buf)
40 buf = malloc(buflen);
41
42 s = str;
43 p = strchr(str, '\'');
44 if (!p)
45 return str;
46
47 do {
48 int len = p - s;
49 if (len > 0) {
50 if (p + 3 - str >= buflen) {
51 buflen *= 2;
52 buf = realloc(buf, buflen);
53 if (!buf) {
54 fprintf(stderr, "Out of memory\n");
55 exit(255);
56 }
57 }
58 memcpy(&buf[pos], s, len);
59 pos += len;
60 }
61 strcpy(&buf[pos], "'\\''");
62 pos += 3;
63 s = p + 1;
64 } while ((p = strchr(s, '\'')));
65
66 return buf;
67 }
68
69 static void uci_show_section(struct uci_section *p)
70 {
71 struct uci_option *o;
72 const char *cname, *sname;
73
74 cname = p->config->name;
75 sname = p->name;
76 printf("%s.%s=%s\n", cname, sname, p->type);
77 uci_foreach_entry(option, &p->options, o) {
78 printf("%s.%s.%s=%s\n", cname, sname, o->name, o->value);
79 }
80 }
81
82 static void uci_export_section(struct uci_section *p)
83 {
84 struct uci_option *o;
85 const char *name;
86
87 printf("\nconfig '%s'", uci_escape(p->type));
88 printf(" '%s'\n", uci_escape(p->name));
89 uci_foreach_entry(option, &p->options, o) {
90 printf("\toption '%s'", uci_escape(o->name));
91 printf(" '%s'\n", uci_escape(o->value));
92 }
93 }
94
95 static void foreach_section(const char *configname, const char *section, void (*callback)(struct uci_section *))
96 {
97 struct uci_config *cfg;
98 struct uci_section *p;
99
100 if (uci_load(ctx, configname, &cfg) != UCI_OK) {
101 uci_perror(ctx, "uci_load");
102 return;
103 }
104
105 uci_list_empty(&cfg->sections);
106 uci_foreach_entry(section, &cfg->sections, p) {
107 if (!section || !strcmp(p->name, section))
108 callback(p);
109 }
110 uci_unload(ctx, configname);
111 }
112
113 static int uci_show(int argc, char **argv)
114 {
115 char **configs = uci_list_configs();
116 char **p;
117
118 if (!configs)
119 return 0;
120
121 for (p = configs; *p; p++) {
122 if ((argc < 2) || !strcmp(argv[1], *p))
123 foreach_section(*p, (argc > 2 ? argv[2] : NULL), uci_show_section);
124 }
125
126 return 0;
127 }
128
129 static int uci_export(int argc, char **argv)
130 {
131 char **configs = uci_list_configs();
132 char **p;
133
134 if (!configs)
135 return 0;
136
137 for (p = configs; *p; p++) {
138 if ((argc < 2) || !strcmp(argv[1], *p)) {
139 printf("package '%s'\n", uci_escape(*p));
140 foreach_section(*p, NULL, uci_export_section);
141 printf("\n");
142 }
143 }
144 return 0;
145 }
146
147 static int uci_cmd(int argc, char **argv)
148 {
149 if (!strcasecmp(argv[0], "show"))
150 return uci_show(argc, argv);
151 if (!strcasecmp(argv[0], "export"))
152 return uci_export(argc, argv);
153 return 255;
154 }
155
156 int main(int argc, char **argv)
157 {
158 int ret;
159
160 ctx = uci_alloc();
161 if (argc < 2)
162 uci_usage(argc, argv);
163 ret = uci_cmd(argc - 1, argv + 1);
164 if (ret == 255)
165 uci_usage(argc, argv);
166 uci_free(ctx);
167
168 return ret;
169 }