|
krh@80
|
1 |
#include <stdio.h>
|
|
krh@80
|
2 |
#include <string.h>
|
|
krh@80
|
3 |
#include <stdarg.h>
|
|
krh@80
|
4 |
#include <unistd.h>
|
|
krh@80
|
5 |
#include <fcntl.h>
|
|
krh@80
|
6 |
#include <errno.h>
|
|
krh@80
|
7 |
#include <expat.h>
|
|
krh@80
|
8 |
|
|
krh@80
|
9 |
#include "razor.h"
|
|
krh@80
|
10 |
|
|
krh@80
|
11 |
#define XML_BUFFER_SIZE 4096
|
|
krh@80
|
12 |
|
|
krh@80
|
13 |
static void
|
|
krh@80
|
14 |
parse_xml_file(const char *filename,
|
|
krh@80
|
15 |
XML_StartElementHandler start,
|
|
krh@80
|
16 |
XML_EndElementHandler end,
|
|
krh@80
|
17 |
void *data)
|
|
krh@80
|
18 |
{
|
|
krh@80
|
19 |
XML_Parser parser;
|
|
krh@80
|
20 |
char *buffer;
|
|
krh@80
|
21 |
int fd, len, err;
|
|
krh@80
|
22 |
|
|
krh@80
|
23 |
parser = XML_ParserCreate(NULL);
|
|
krh@80
|
24 |
XML_SetElementHandler(parser, start, end);
|
|
krh@80
|
25 |
XML_SetUserData(parser, data);
|
|
krh@80
|
26 |
|
|
krh@80
|
27 |
buffer = XML_GetBuffer(parser, XML_BUFFER_SIZE);
|
|
krh@80
|
28 |
|
|
krh@80
|
29 |
fd = open(filename, O_RDONLY);
|
|
krh@80
|
30 |
if (fd < 0) {
|
|
krh@92
|
31 |
fprintf(stderr, "failed to open %s: %m\n", filename);
|
|
krh@80
|
32 |
exit(-1);
|
|
krh@80
|
33 |
}
|
|
krh@80
|
34 |
|
|
krh@80
|
35 |
while (len = read(fd, buffer, XML_BUFFER_SIZE), len > 0) {
|
|
krh@80
|
36 |
err = XML_ParseBuffer(parser, len, len == 0);
|
|
krh@80
|
37 |
if (err == XML_STATUS_ERROR) {
|
|
krh@80
|
38 |
fprintf(stderr, "parse error at line %lu:\n%s\n",
|
|
krh@80
|
39 |
XML_GetCurrentLineNumber(parser),
|
|
krh@80
|
40 |
XML_ErrorString(XML_GetErrorCode(parser)));
|
|
krh@80
|
41 |
exit(-1);
|
|
krh@80
|
42 |
}
|
|
krh@80
|
43 |
}
|
|
krh@80
|
44 |
|
|
krh@80
|
45 |
if (fd < 0) {
|
|
krh@80
|
46 |
fprintf(stderr, "read: %m\n");
|
|
krh@80
|
47 |
exit(-1);
|
|
krh@80
|
48 |
}
|
|
krh@80
|
49 |
|
|
krh@80
|
50 |
close(fd);
|
|
krh@80
|
51 |
}
|
|
krh@80
|
52 |
|
|
danw@125
|
53 |
struct test_context {
|
|
danw@125
|
54 |
struct razor_set *system_set, *repo_set, *result_set;
|
|
krh@80
|
55 |
|
|
krh@80
|
56 |
struct razor_importer *importer;
|
|
danw@125
|
57 |
struct razor_set **importer_set;
|
|
danw@125
|
58 |
|
|
danw@125
|
59 |
char *install_pkgs[3], *remove_pkgs[3];
|
|
danw@125
|
60 |
int n_install_pkgs, n_remove_pkgs;
|
|
danw@125
|
61 |
|
|
danw@125
|
62 |
int in_result, result_errors;
|
|
krh@80
|
63 |
};
|
|
krh@80
|
64 |
|
|
krh@80
|
65 |
static void
|
|
krh@80
|
66 |
get_atts(const char **atts, ...)
|
|
krh@80
|
67 |
{
|
|
krh@80
|
68 |
va_list ap;
|
|
krh@80
|
69 |
const char *name, **ptr;
|
|
krh@80
|
70 |
int i;
|
|
krh@80
|
71 |
|
|
krh@80
|
72 |
va_start(ap, atts);
|
|
krh@80
|
73 |
while (name = va_arg(ap, const char *), name != NULL) {
|
|
krh@80
|
74 |
ptr = va_arg(ap, const char **);
|
|
krh@92
|
75 |
*ptr = NULL;
|
|
krh@80
|
76 |
for (i = 0; atts[i]; i += 2) {
|
|
krh@80
|
77 |
if (strcmp(atts[i], name) == 0)
|
|
krh@80
|
78 |
*ptr = atts[i + 1];
|
|
krh@80
|
79 |
}
|
|
krh@80
|
80 |
}
|
|
krh@80
|
81 |
va_end(ap);
|
|
krh@80
|
82 |
}
|
|
krh@80
|
83 |
|
|
danw@125
|
84 |
static enum razor_version_relation
|
|
danw@125
|
85 |
parse_relation (const char *rel_str)
|
|
krh@80
|
86 |
{
|
|
danw@125
|
87 |
if (!rel_str)
|
|
danw@125
|
88 |
return -1;
|
|
danw@125
|
89 |
if (rel_str[0] == 'l')
|
|
danw@125
|
90 |
return rel_str[1] == 'e' ? RAZOR_VERSION_LESS_OR_EQUAL : RAZOR_VERSION_LESS;
|
|
danw@125
|
91 |
else if (rel_str[0] == 'g')
|
|
danw@125
|
92 |
return rel_str[1] == 'e' ? RAZOR_VERSION_GREATER_OR_EQUAL : RAZOR_VERSION_GREATER;
|
|
danw@125
|
93 |
else if (rel_str[0] == 'e' || rel_str[1] == 'q')
|
|
danw@125
|
94 |
return RAZOR_VERSION_EQUAL;
|
|
danw@125
|
95 |
else
|
|
danw@125
|
96 |
return -1;
|
|
krh@80
|
97 |
}
|
|
krh@80
|
98 |
|
|
krh@80
|
99 |
static void
|
|
danw@125
|
100 |
start_test(struct test_context *ctx, const char **atts)
|
|
krh@80
|
101 |
{
|
|
danw@125
|
102 |
const char *name = NULL;
|
|
krh@80
|
103 |
|
|
danw@125
|
104 |
get_atts(atts, "name", &name, NULL);
|
|
danw@125
|
105 |
if (!name) {
|
|
danw@125
|
106 |
fprintf(stderr, "Test with no name\n");
|
|
danw@125
|
107 |
exit(1);
|
|
danw@125
|
108 |
}
|
|
danw@125
|
109 |
printf("%s\n", name);
|
|
danw@125
|
110 |
}
|
|
danw@125
|
111 |
|
|
danw@125
|
112 |
static void
|
|
danw@125
|
113 |
end_test(struct test_context *ctx)
|
|
danw@125
|
114 |
{
|
|
danw@125
|
115 |
if (ctx->system_set) {
|
|
danw@125
|
116 |
razor_set_destroy(ctx->system_set);
|
|
danw@125
|
117 |
ctx->system_set = NULL;
|
|
danw@125
|
118 |
}
|
|
danw@125
|
119 |
if (ctx->repo_set) {
|
|
danw@125
|
120 |
razor_set_destroy(ctx->repo_set);
|
|
danw@125
|
121 |
ctx->repo_set = NULL;
|
|
danw@125
|
122 |
}
|
|
danw@125
|
123 |
if (ctx->result_set) {
|
|
danw@125
|
124 |
razor_set_destroy(ctx->result_set);
|
|
danw@125
|
125 |
ctx->result_set = NULL;
|
|
krh@80
|
126 |
}
|
|
krh@80
|
127 |
}
|
|
krh@80
|
128 |
|
|
krh@80
|
129 |
static void
|
|
danw@125
|
130 |
start_set(struct test_context *ctx, const char **atts)
|
|
krh@80
|
131 |
{
|
|
danw@125
|
132 |
const char *name = NULL;
|
|
krh@80
|
133 |
|
|
danw@125
|
134 |
ctx->importer = razor_importer_new();
|
|
danw@125
|
135 |
get_atts(atts, "name", &name, NULL);
|
|
danw@125
|
136 |
if (!name)
|
|
danw@125
|
137 |
ctx->importer_set = &ctx->result_set;
|
|
danw@125
|
138 |
else if (!strcmp(name, "system"))
|
|
danw@125
|
139 |
ctx->importer_set = &ctx->system_set;
|
|
danw@125
|
140 |
else if (!strcmp(name, "repo"))
|
|
danw@125
|
141 |
ctx->importer_set = &ctx->repo_set;
|
|
danw@125
|
142 |
else {
|
|
danw@125
|
143 |
fprintf(stderr, " bad set name '%s'\n", name);
|
|
danw@125
|
144 |
exit(1);
|
|
krh@92
|
145 |
}
|
|
krh@92
|
146 |
}
|
|
krh@92
|
147 |
|
|
krh@92
|
148 |
static void
|
|
danw@125
|
149 |
end_set(struct test_context *ctx)
|
|
krh@92
|
150 |
{
|
|
danw@125
|
151 |
*ctx->importer_set = razor_importer_finish(ctx->importer);
|
|
danw@125
|
152 |
ctx->importer = NULL;
|
|
danw@125
|
153 |
}
|
|
krh@92
|
154 |
|
|
danw@125
|
155 |
static void
|
|
danw@125
|
156 |
start_package(struct test_context *ctx, const char **atts)
|
|
danw@125
|
157 |
{
|
|
danw@125
|
158 |
const char *name = NULL, *version = NULL, *arch = NULL;
|
|
danw@125
|
159 |
|
|
danw@125
|
160 |
get_atts(atts, "name", &name, "version", &version, "arch", &arch, NULL);
|
|
danw@125
|
161 |
if (!name) {
|
|
danw@125
|
162 |
fprintf(stderr, " package with no name\n");
|
|
danw@125
|
163 |
exit(1);
|
|
krh@92
|
164 |
}
|
|
krh@92
|
165 |
|
|
danw@125
|
166 |
razor_importer_begin_package(ctx->importer, name, version);
|
|
danw@125
|
167 |
razor_importer_add_property(ctx->importer, name,
|
|
danw@125
|
168 |
RAZOR_VERSION_EQUAL, version,
|
|
danw@125
|
169 |
RAZOR_PROPERTY_PROVIDES);
|
|
danw@125
|
170 |
}
|
|
danw@125
|
171 |
|
|
danw@125
|
172 |
static void
|
|
danw@125
|
173 |
end_package(struct test_context *ctx)
|
|
danw@125
|
174 |
{
|
|
danw@125
|
175 |
razor_importer_finish_package(ctx->importer);
|
|
danw@125
|
176 |
}
|
|
danw@125
|
177 |
|
|
danw@125
|
178 |
static void
|
|
danw@125
|
179 |
start_property(struct test_context *ctx, enum razor_property_type type, const char **atts)
|
|
danw@125
|
180 |
{
|
|
danw@125
|
181 |
const char *name = NULL, *rel_str = NULL, *version = NULL;
|
|
danw@125
|
182 |
enum razor_version_relation rel;
|
|
danw@125
|
183 |
|
|
danw@125
|
184 |
get_atts(atts, "name", &name, "rel", &rel_str, "version", &version, NULL);
|
|
danw@125
|
185 |
if (name == NULL) {
|
|
danw@125
|
186 |
fprintf(stderr, " no name specified for property\n");
|
|
danw@125
|
187 |
exit(1);
|
|
danw@125
|
188 |
}
|
|
danw@125
|
189 |
if (version) {
|
|
danw@125
|
190 |
rel = parse_relation(rel_str);
|
|
danw@125
|
191 |
if (rel == -1) {
|
|
danw@125
|
192 |
fprintf(stderr, " bad or missing version relation for property %s\n", name);
|
|
danw@125
|
193 |
exit(1);
|
|
krh@92
|
194 |
}
|
|
danw@125
|
195 |
} else
|
|
danw@125
|
196 |
rel = RAZOR_VERSION_EQUAL;
|
|
danw@125
|
197 |
|
|
danw@125
|
198 |
razor_importer_add_property(ctx->importer, name,
|
|
danw@125
|
199 |
rel, version, type);
|
|
danw@125
|
200 |
}
|
|
krh@92
|
201 |
|
|
danw@125
|
202 |
static void
|
|
danw@125
|
203 |
start_transaction(struct test_context *ctx, const char **atts)
|
|
danw@125
|
204 |
{
|
|
danw@125
|
205 |
ctx->n_install_pkgs = 0;
|
|
danw@125
|
206 |
ctx->n_remove_pkgs = 0;
|
|
danw@125
|
207 |
}
|
|
danw@125
|
208 |
|
|
danw@125
|
209 |
static void
|
|
danw@125
|
210 |
end_transaction(struct test_context *ctx)
|
|
danw@125
|
211 |
{
|
|
danw@125
|
212 |
/* FIXME: removes */
|
|
danw@125
|
213 |
ctx->system_set = razor_set_update(ctx->system_set,
|
|
danw@125
|
214 |
ctx->repo_set,
|
|
danw@125
|
215 |
ctx->n_install_pkgs,
|
|
danw@125
|
216 |
(const char **)ctx->install_pkgs);
|
|
danw@125
|
217 |
|
|
danw@125
|
218 |
while (ctx->n_install_pkgs--)
|
|
danw@125
|
219 |
free(ctx->install_pkgs[ctx->n_install_pkgs]);
|
|
danw@125
|
220 |
while (ctx->n_remove_pkgs--)
|
|
danw@125
|
221 |
free(ctx->remove_pkgs[ctx->n_remove_pkgs]);
|
|
danw@125
|
222 |
}
|
|
danw@125
|
223 |
|
|
danw@125
|
224 |
static void
|
|
danw@125
|
225 |
start_install_or_update(struct test_context *ctx, const char **atts)
|
|
danw@125
|
226 |
{
|
|
danw@125
|
227 |
const char *name = NULL;
|
|
danw@125
|
228 |
|
|
danw@125
|
229 |
get_atts(atts, "name", &name, NULL);
|
|
danw@125
|
230 |
if (!name) {
|
|
danw@125
|
231 |
fprintf(stderr, " install/update with no name\n");
|
|
danw@125
|
232 |
exit(1);
|
|
danw@125
|
233 |
}
|
|
danw@125
|
234 |
|
|
danw@125
|
235 |
ctx->install_pkgs[ctx->n_install_pkgs++] = strdup(name);
|
|
danw@125
|
236 |
}
|
|
danw@125
|
237 |
|
|
danw@125
|
238 |
static void
|
|
danw@125
|
239 |
start_remove(struct test_context *ctx, const char **atts)
|
|
danw@125
|
240 |
{
|
|
danw@125
|
241 |
const char *name = NULL;
|
|
danw@125
|
242 |
|
|
danw@125
|
243 |
get_atts(atts, "name", &name, NULL);
|
|
danw@125
|
244 |
if (!name) {
|
|
danw@125
|
245 |
fprintf(stderr, " remove with no name\n");
|
|
danw@125
|
246 |
exit(1);
|
|
danw@125
|
247 |
}
|
|
danw@125
|
248 |
|
|
danw@125
|
249 |
ctx->remove_pkgs[ctx->n_remove_pkgs++] = strdup(name);
|
|
danw@125
|
250 |
}
|
|
danw@125
|
251 |
|
|
danw@125
|
252 |
static void
|
|
danw@125
|
253 |
start_result(struct test_context *ctx, const char **atts)
|
|
danw@125
|
254 |
{
|
|
danw@125
|
255 |
ctx->in_result = 1;
|
|
danw@125
|
256 |
}
|
|
danw@125
|
257 |
|
|
danw@125
|
258 |
static void
|
|
danw@125
|
259 |
diff_callback(const char *name,
|
|
danw@125
|
260 |
const char *old_version, const char *new_version,
|
|
danw@125
|
261 |
void *data)
|
|
danw@125
|
262 |
{
|
|
danw@125
|
263 |
struct test_context *ctx = data;
|
|
danw@125
|
264 |
|
|
danw@125
|
265 |
ctx->result_errors++;
|
|
danw@125
|
266 |
if (old_version) {
|
|
danw@125
|
267 |
fprintf(stderr, " result set should not contain %s %s\n",
|
|
danw@125
|
268 |
name, old_version);
|
|
danw@125
|
269 |
} else {
|
|
danw@125
|
270 |
fprintf(stderr, " result set should contain %s %s\n",
|
|
danw@125
|
271 |
name, new_version);
|
|
krh@92
|
272 |
}
|
|
krh@92
|
273 |
}
|
|
krh@92
|
274 |
|
|
krh@92
|
275 |
static void
|
|
danw@125
|
276 |
end_result(struct test_context *ctx)
|
|
krh@92
|
277 |
{
|
|
danw@125
|
278 |
ctx->in_result = 0;
|
|
krh@92
|
279 |
|
|
danw@125
|
280 |
/* FIXME */
|
|
danw@125
|
281 |
if (ctx->n_remove_pkgs) {
|
|
danw@125
|
282 |
printf (" (ignoring because of unimplemented remove)\n");
|
|
danw@125
|
283 |
return;
|
|
krh@92
|
284 |
}
|
|
krh@92
|
285 |
|
|
danw@125
|
286 |
if (ctx->system_set && ctx->result_set) {
|
|
danw@125
|
287 |
ctx->result_errors = 0;
|
|
danw@125
|
288 |
razor_set_diff(ctx->system_set, ctx->result_set,
|
|
danw@125
|
289 |
diff_callback, ctx);
|
|
danw@125
|
290 |
if (ctx->result_errors)
|
|
danw@125
|
291 |
exit(1);
|
|
krh@92
|
292 |
}
|
|
danw@125
|
293 |
|
|
krh@92
|
294 |
}
|
|
krh@92
|
295 |
|
|
krh@92
|
296 |
static void
|
|
danw@125
|
297 |
start_unsatisfied(struct test_context *ctx, const char **atts)
|
|
krh@92
|
298 |
{
|
|
danw@125
|
299 |
/* FIXME */
|
|
danw@125
|
300 |
fprintf(stderr, "Can't handle <unsatisfied>\n");
|
|
danw@125
|
301 |
exit(1);
|
|
danw@125
|
302 |
}
|
|
krh@92
|
303 |
|
|
danw@125
|
304 |
static void
|
|
danw@125
|
305 |
end_unsatisfied(struct test_context *ctx)
|
|
danw@125
|
306 |
{
|
|
krh@92
|
307 |
}
|
|
krh@92
|
308 |
|
|
krh@92
|
309 |
static void
|
|
krh@92
|
310 |
start_test_element(void *data, const char *element, const char **atts)
|
|
krh@92
|
311 |
{
|
|
krh@92
|
312 |
struct test_context *ctx = data;
|
|
krh@92
|
313 |
|
|
danw@125
|
314 |
if (strcmp(element, "tests") == 0) {
|
|
danw@125
|
315 |
;
|
|
danw@125
|
316 |
} else if (strcmp(element, "test") == 0) {
|
|
danw@125
|
317 |
start_test(ctx, atts);
|
|
danw@125
|
318 |
} else if (strcmp(element, "set") == 0) {
|
|
danw@125
|
319 |
start_set(ctx, atts);
|
|
danw@125
|
320 |
} else if (strcmp(element, "transaction") == 0) {
|
|
danw@125
|
321 |
start_transaction(ctx, atts);
|
|
danw@125
|
322 |
} else if (strcmp(element, "install") == 0) {
|
|
danw@125
|
323 |
start_install_or_update(ctx, atts);
|
|
danw@125
|
324 |
} else if (strcmp(element, "install") == 0) {
|
|
danw@125
|
325 |
start_install_or_update(ctx, atts);
|
|
danw@125
|
326 |
} else if (strcmp(element, "remove") == 0) {
|
|
danw@125
|
327 |
start_remove(ctx, atts);
|
|
danw@125
|
328 |
} else if (strcmp(element, "result") == 0) {
|
|
danw@125
|
329 |
start_result(ctx, atts);
|
|
danw@125
|
330 |
} else if (strcmp(element, "unsatisfied") == 0) {
|
|
danw@125
|
331 |
start_unsatisfied(ctx, atts);
|
|
krh@92
|
332 |
} else if (strcmp(element, "package") == 0) {
|
|
danw@125
|
333 |
start_package(ctx, atts);
|
|
krh@92
|
334 |
} else if (strcmp(element, "requires") == 0) {
|
|
danw@125
|
335 |
start_property(ctx, RAZOR_PROPERTY_REQUIRES, atts);
|
|
krh@92
|
336 |
} else if (strcmp(element, "provides") == 0) {
|
|
danw@125
|
337 |
start_property(ctx, RAZOR_PROPERTY_PROVIDES, atts);
|
|
krh@92
|
338 |
} else if (strcmp(element, "conflicts") == 0) {
|
|
danw@125
|
339 |
start_property(ctx, RAZOR_PROPERTY_CONFLICTS, atts);
|
|
krh@92
|
340 |
} else if (strcmp(element, "obsoletes") == 0) {
|
|
danw@125
|
341 |
start_property(ctx, RAZOR_PROPERTY_OBSOLETES, atts);
|
|
danw@125
|
342 |
} else {
|
|
danw@125
|
343 |
fprintf(stderr, "Unrecognized element '%s'\n", element);
|
|
danw@125
|
344 |
exit(1);
|
|
krh@92
|
345 |
}
|
|
krh@92
|
346 |
}
|
|
krh@92
|
347 |
|
|
krh@92
|
348 |
static void
|
|
krh@92
|
349 |
end_test_element (void *data, const char *element)
|
|
krh@92
|
350 |
{
|
|
krh@92
|
351 |
struct test_context *ctx = data;
|
|
krh@92
|
352 |
|
|
danw@125
|
353 |
if (strcmp(element, "test") == 0) {
|
|
danw@125
|
354 |
end_test(ctx);
|
|
danw@125
|
355 |
} else if (strcmp(element, "set") == 0) {
|
|
danw@125
|
356 |
end_set(ctx);
|
|
danw@125
|
357 |
} else if (strcmp(element, "package") == 0) {
|
|
danw@125
|
358 |
end_package(ctx);
|
|
danw@125
|
359 |
} else if (strcmp(element, "transaction") == 0) {
|
|
danw@125
|
360 |
end_transaction(ctx);
|
|
danw@125
|
361 |
} else if (strcmp(element, "result") == 0) {
|
|
danw@125
|
362 |
end_result(ctx);
|
|
danw@125
|
363 |
} else if (strcmp(element, "unsatisfied") == 0) {
|
|
danw@125
|
364 |
end_unsatisfied(ctx);
|
|
danw@125
|
365 |
}
|
|
krh@92
|
366 |
}
|
|
krh@92
|
367 |
|
|
krh@80
|
368 |
int main(int argc, char *argv[])
|
|
krh@80
|
369 |
{
|
|
krh@80
|
370 |
struct test_context ctx;
|
|
krh@80
|
371 |
|
|
krh@92
|
372 |
if (argc != 2) {
|
|
krh@92
|
373 |
fprintf(stderr, "usage: %s TESTS-FILE\n", argv[0]);
|
|
krh@80
|
374 |
exit(-1);
|
|
krh@80
|
375 |
}
|
|
krh@80
|
376 |
|
|
krh@80
|
377 |
memset(&ctx, 0, sizeof ctx);
|
|
krh@92
|
378 |
parse_xml_file(argv[1], start_test_element, end_test_element, &ctx);
|
|
krh@80
|
379 |
|
|
krh@80
|
380 |
return 0;
|
|
krh@80
|
381 |
}
|