Branch data Line data Source code
1 : : /*-
2 : : * Copyright (c) 2015-2017, Vsevolod Stakhov
3 : : * All rights reserved.
4 : : *
5 : : * Redistribution and use in source and binary forms, with or without
6 : : * modification, are permitted provided that the following conditions are met:
7 : : * * Redistributions of source code must retain the above copyright
8 : : * notice, this list of conditions and the following disclaimer.
9 : : * * Redistributions in binary form must reproduce the above copyright
10 : : * notice, this list of conditions and the following disclaimer in the
11 : : * documentation and/or other materials provided with the distribution.
12 : : *
13 : : * THIS SOFTWARE IS PROVIDED BY AUTHOR ''AS IS'' AND ANY
14 : : * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
15 : : * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
16 : : * DISCLAIMED. IN NO EVENT SHALL AUTHOR BE LIABLE FOR ANY
17 : : * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
18 : : * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
19 : : * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
20 : : * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21 : : * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
22 : : * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23 : : */
24 : :
25 : : #ifdef HAVE_CONFIG_H
26 : : #include "pkg_config.h"
27 : : #endif
28 : :
29 : : #include "bsd_compat.h"
30 : :
31 : : #include <stddef.h>
32 : : #include <ctype.h>
33 : : #include <assert.h>
34 : : #include <string.h>
35 : : #include <stdlib.h>
36 : :
37 : : #include "pkg.h"
38 : : #include "private/event.h"
39 : : #include "private/pkg_deps.h"
40 : : #include "xmalloc.h"
41 : : #include "utlist.h"
42 : :
43 : : struct pkg_dep_formula *
44 : 12 : pkg_deps_parse_formula(const char *in)
45 : : {
46 : 12 : struct pkg_dep_formula *res = NULL, *cur = NULL;
47 : 12 : struct pkg_dep_formula_item *cur_item = NULL;
48 : 12 : struct pkg_dep_version_item *cur_ver = NULL;
49 : 12 : struct pkg_dep_option_item *cur_opt = NULL;
50 : : const char *p, *c, *end;
51 : 12 : enum pkg_dep_version_op cur_op = VERSION_ANY;
52 : : enum {
53 : : st_parse_dep_name = 0,
54 : : st_parse_after_name,
55 : : st_parse_ver_op,
56 : : st_parse_after_op,
57 : : st_parse_version_number,
58 : : st_parse_after_version,
59 : : st_parse_option_start,
60 : : st_parse_option,
61 : : st_parse_after_option,
62 : : st_parse_comma,
63 : : st_parse_or,
64 : : st_skip_spaces,
65 : : st_error
66 : 12 : } state = 0, next_state = 0;
67 : :
68 : 12 : c = in;
69 : 12 : p = in;
70 : :
71 : 12 : end = p + strlen(p);
72 : :
73 [ + + ]: 436 : while (p <= end) {
74 [ - + + + : 424 : switch (state) {
+ + + + +
+ + + +
- ]
75 : : case st_parse_dep_name:
76 [ + + + + ]: 126 : if (isspace(*p) || *p == '\0') {
77 : 20 : state = st_skip_spaces;
78 : :
79 [ + - ]: 20 : if (p == c) {
80 : : /* Spaces at the beginning */
81 : 0 : next_state = st_parse_dep_name;
82 : 0 : }
83 : : else {
84 : : /* Spaces after the name */
85 : 20 : cur_item = xcalloc(1, sizeof(*cur_item));
86 : 20 : cur_item->name = xmalloc(p - c + 1);
87 : 20 : strlcpy(cur_item->name, c, p - c + 1);
88 : 20 : next_state = st_parse_after_name;
89 : : }
90 : 20 : }
91 [ + + ]: 106 : else if (*p == ',') {
92 [ + - ]: 2 : if (p == c) {
93 : 0 : state = st_error;
94 : 0 : }
95 : : else {
96 : 2 : cur_item = xcalloc(1, sizeof(*cur_item));
97 : 2 : cur_item->name = xmalloc(p - c + 1);
98 : 2 : strlcpy(cur_item->name, c, p - c + 1);
99 : 2 : state = st_parse_after_name;
100 : : }
101 : 2 : }
102 [ + - ]: 104 : else if (!isprint(*p)) {
103 : 0 : state = st_error;
104 : 0 : }
105 : : else {
106 : 104 : p++;
107 : : }
108 : 154 : break;
109 : :
110 : : case st_parse_after_name:
111 : : case st_parse_after_version:
112 : : case st_parse_after_option: {
113 [ + - + - : 28 : switch (*p) {
+ + + + +
+ ]
114 : : case ',':
115 : : case '\0':
116 : 5 : state = st_parse_comma;
117 : 5 : break;
118 : : case '|':
119 : 5 : state = st_parse_or;
120 : 7 : break;
121 : : case '+':
122 : : case '-':
123 : 2 : c = p;
124 : 2 : state = st_parse_option_start;
125 : 18 : break;
126 : : case '>':
127 : : case '<':
128 : : case '=':
129 : : case '!':
130 : 16 : c = p;
131 : 16 : cur_op = VERSION_ANY;
132 : 16 : state = st_parse_ver_op;
133 : 16 : break;
134 : : default:
135 : 0 : state = st_error;
136 : 0 : break;
137 : : }
138 : 28 : break;
139 : : }
140 : :
141 : : case st_parse_ver_op: {
142 [ + + + + : 39 : switch (*p) {
+ ]
143 : : case '>':
144 : : case '<':
145 : : case '=':
146 : : case '!':
147 : 23 : p ++;
148 : 23 : break;
149 : : default:
150 [ + + ]: 16 : if (p - c == 2) {
151 [ + + ]: 7 : if (memcmp(c, ">=", 2) == 0) {
152 : 2 : cur_op = VERSION_GE;
153 : 2 : }
154 [ + - ]: 5 : else if (memcmp(c, "<=", 2) == 0) {
155 : 0 : cur_op = VERSION_LE;
156 : 0 : }
157 [ - + ]: 5 : else if (memcmp(c, "!=", 2) == 0) {
158 : 5 : cur_op = VERSION_NOT;
159 : 5 : }
160 [ # # ]: 0 : else if (memcmp(c, "==", 2) == 0) {
161 : 0 : cur_op = VERSION_EQ;
162 : 0 : }
163 : : else {
164 : 0 : state = st_error;
165 : : }
166 : 7 : }
167 [ + - ]: 9 : else if (p - c == 1) {
168 [ + + ]: 9 : if (*c == '>') {
169 : 2 : cur_op = VERSION_GT;
170 : 2 : }
171 [ + + ]: 7 : else if (*c == '<') {
172 : 2 : cur_op = VERSION_LT;
173 : 2 : }
174 [ - + ]: 5 : else if (*c == '!') {
175 : 0 : cur_op = VERSION_NOT;
176 : 0 : }
177 [ + - ]: 5 : else if (*c == '=') {
178 : 5 : cur_op = VERSION_EQ;
179 : 5 : }
180 : : else {
181 : 0 : state = st_error;
182 : : }
183 : 9 : }
184 : : else {
185 : 0 : state = st_error;
186 : : }
187 : :
188 [ - + ]: 16 : if (state != st_error) {
189 : 16 : state = st_skip_spaces;
190 : 16 : next_state = st_parse_after_op;
191 : 16 : }
192 : 16 : break;
193 : : }
194 : 39 : break;
195 : : }
196 : :
197 : : case st_parse_after_op:
198 [ + - ]: 16 : if (cur_op == VERSION_ANY) {
199 : 0 : state = st_error;
200 : 0 : }
201 : : else {
202 : 16 : state = st_parse_version_number;
203 : : }
204 : 16 : break;
205 : :
206 : : case st_parse_version_number:
207 [ + + - + : 73 : if (isalnum(*p) || *p == '-' || *p == '_' || *p == '.' ||
- + + + +
+ ]
208 [ + + ]: 18 : (*p == ',' && isdigit(*(p + 1)))) {
209 : 52 : p ++;
210 : 52 : }
211 : : else {
212 [ - + ]: 16 : if (p - c > 0) {
213 : 16 : cur_ver = xcalloc(1, sizeof(*cur_ver));
214 : 16 : cur_ver->ver = xmalloc(p - c + 1);
215 : 16 : strlcpy(cur_ver->ver, c, p - c + 1);
216 : 16 : cur_ver->op = cur_op;
217 [ - + ]: 16 : assert(cur_item != NULL);
218 [ + + ]: 16 : DL_APPEND(cur_item->versions, cur_ver);
219 : 16 : state = st_skip_spaces;
220 : 16 : next_state = st_parse_after_version;
221 : 16 : }
222 : : else {
223 : 0 : state = st_error;
224 : : }
225 : : }
226 : 68 : break;
227 : :
228 : : case st_parse_option_start:
229 : 2 : cur_opt = xcalloc(1, sizeof(*cur_opt));
230 [ + + ]: 2 : if (*p == '+') {
231 : 1 : cur_opt->on = true;
232 : 1 : }
233 : : else {
234 : 1 : cur_opt->on = false;
235 : : }
236 : :
237 : 2 : p ++;
238 : 2 : c = p;
239 : 2 : state = st_parse_option;
240 : 2 : break;
241 : :
242 : : case st_parse_option:
243 [ + + - + : 10 : if (isalnum(*p) || *p == '-' || *p == '_') {
- + ]
244 : 8 : p ++;
245 : 8 : }
246 : : else {
247 [ + - ]: 2 : if (p - c > 0) {
248 : 2 : cur_opt->opt = xmalloc(p - c + 1);
249 : 2 : strlcpy(cur_opt->opt, c, p - c + 1);
250 [ + - ]: 2 : assert(cur_item != NULL);
251 [ + + ]: 2 : DL_APPEND(cur_item->options, cur_opt);
252 : 2 : state = st_skip_spaces;
253 : 2 : next_state = st_parse_after_option;
254 : 2 : }
255 : : else {
256 : 0 : state = st_error;
257 : : }
258 : : }
259 : 10 : break;
260 : :
261 : : case st_parse_comma:
262 [ + - ]: 17 : assert(cur_item != NULL);
263 : :
264 [ + + ]: 17 : if (cur == NULL) {
265 : 12 : cur = xcalloc(1, sizeof(*cur));
266 : 12 : }
267 : :
268 [ + + ]: 17 : DL_APPEND(cur->items, cur_item);
269 [ + + ]: 17 : DL_APPEND(res, cur);
270 : 17 : cur_item = NULL;
271 : 17 : cur = NULL;
272 : 17 : p ++;
273 : 17 : state = st_skip_spaces;
274 : 17 : next_state = st_parse_dep_name;
275 : 17 : break;
276 : :
277 : : case st_parse_or:
278 [ + - ]: 5 : assert(cur_item != NULL);
279 : :
280 [ - + ]: 5 : if (cur == NULL) {
281 : 5 : cur = xcalloc(1, sizeof(*cur));
282 : 5 : }
283 : :
284 [ - + ]: 5 : DL_APPEND(cur->items, cur_item);
285 : 5 : cur_item = NULL;
286 : 5 : p ++;
287 : 5 : state = st_skip_spaces;
288 : 5 : next_state = st_parse_dep_name;
289 : 5 : break;
290 : :
291 : : case st_skip_spaces:
292 [ + + ]: 113 : if (isspace(*p)) {
293 : 49 : p ++;
294 : 49 : }
295 [ + + ]: 64 : else if (*p == '\0') {
296 : 12 : state = st_parse_comma;
297 : 12 : }
298 : : else {
299 : 52 : c = p;
300 : 52 : state = next_state;
301 : : }
302 : 113 : break;
303 : :
304 : 0 : case st_error:
305 : : default:
306 : 0 : pkg_emit_error("cannot parse pkg formula: %s", in);
307 : 0 : pkg_deps_formula_free(res);
308 [ # # ]: 0 : if (cur_item != NULL) {
309 : 0 : free(cur_item->name);
310 : 0 : free(cur_item);
311 : 0 : }
312 : :
313 : 0 : return (NULL);
314 : :
315 : : break;
316 : : }
317 : : }
318 : :
319 [ - + # # ]: 12 : if (state != st_skip_spaces && state != st_parse_comma) {
320 : 0 : pkg_emit_error("cannot parse pkg formula: %s", in);
321 : 0 : pkg_deps_formula_free(res);
322 [ # # ]: 0 : if (cur_item != NULL) {
323 : 0 : free(cur_item->name);
324 : 0 : free(cur_item);
325 : 0 : }
326 : :
327 : 0 : return (NULL);
328 : : }
329 : :
330 : 12 : return (res);
331 : 12 : }
332 : :
333 : : void
334 : 12 : pkg_deps_formula_free(struct pkg_dep_formula *f)
335 : : {
336 : : struct pkg_dep_formula *cf, *cftmp;
337 : : struct pkg_dep_formula_item *cit, *cittmp;
338 : : struct pkg_dep_version_item *cver, *cvertmp;
339 : : struct pkg_dep_option_item *copt, *copttmp;
340 : :
341 [ + + - + : 29 : DL_FOREACH_SAFE(f, cf, cftmp) {
+ + ]
342 [ + + - + : 39 : DL_FOREACH_SAFE(cf->items, cit, cittmp) {
+ + ]
343 : 22 : free(cit->name);
344 : :
345 [ + + - + : 38 : DL_FOREACH_SAFE(cit->versions, cver, cvertmp) {
+ + ]
346 : 16 : free(cver->ver);
347 : 16 : free(cver);
348 : 16 : }
349 : :
350 [ + + - + : 24 : DL_FOREACH_SAFE(cit->options, copt, copttmp) {
+ + ]
351 : 2 : free(copt->opt);
352 : 2 : free(copt);
353 : 2 : }
354 : :
355 : 22 : free(cit);
356 : 22 : }
357 : :
358 : 17 : free(cf);
359 : 17 : }
360 : 12 : }
361 : :
362 : : static const char*
363 : 16 : pkg_deps_op_tostring(enum pkg_dep_version_op op)
364 : : {
365 : : const char *op_str;
366 : :
367 [ + - + - : 16 : switch (op) {
- + + + ]
368 : 0 : case VERSION_ANY:
369 : : default:
370 : 0 : op_str = "?";
371 : 0 : break;
372 : : case VERSION_EQ:
373 : 5 : op_str = "=";
374 : 5 : break;
375 : : case VERSION_LE:
376 : 0 : op_str = "<=";
377 : 0 : break;
378 : : case VERSION_GE:
379 : 2 : op_str = ">=";
380 : 2 : break;
381 : : case VERSION_LT:
382 : 2 : op_str = "<";
383 : 2 : break;
384 : : case VERSION_GT:
385 : 2 : op_str = ">";
386 : 2 : break;
387 : : case VERSION_NOT:
388 : 5 : op_str = "!=";
389 : 5 : break;
390 : : }
391 : :
392 : 16 : return (op_str);
393 : : }
394 : :
395 : : char*
396 : 7 : pkg_deps_formula_tostring(struct pkg_dep_formula *f)
397 : : {
398 : : struct pkg_dep_formula *cf, *cftmp;
399 : : struct pkg_dep_formula_item *cit, *cittmp;
400 : : struct pkg_dep_version_item *cver, *cvertmp;
401 : : struct pkg_dep_option_item *copt, *copttmp;
402 : 7 : char *res = NULL, *p;
403 : :
404 : 7 : int rlen = 0, r;
405 : :
406 [ + + - + : 19 : DL_FOREACH_SAFE(f, cf, cftmp) {
+ + ]
407 [ + + - + : 27 : DL_FOREACH_SAFE(cf->items, cit, cittmp) {
+ + ]
408 : 15 : rlen += strlen(cit->name);
409 : :
410 [ + + - + : 27 : DL_FOREACH_SAFE(cit->versions, cver, cvertmp) {
+ + ]
411 : 12 : rlen += strlen(cver->ver);
412 : 12 : rlen += 4; /* <OP><SP><VER><SP> */
413 : 12 : }
414 : :
415 [ + + - + : 17 : DL_FOREACH_SAFE(cit->options, copt, copttmp) {
+ + ]
416 : 2 : rlen += strlen(copt->opt);
417 : 2 : rlen += 2; /* <+-><OPT><SP> */
418 : 2 : }
419 : :
420 : 15 : rlen += 2; /* |<SP> */
421 : 15 : }
422 : :
423 : 12 : rlen += 2; /* <,><SP> */
424 : 12 : }
425 : :
426 [ + - ]: 7 : if (rlen == 0) {
427 : 0 : return (NULL);
428 : : }
429 : :
430 : 7 : res = xmalloc(rlen + 1);
431 : :
432 : 7 : p = res;
433 : :
434 [ + + - + : 19 : DL_FOREACH_SAFE(f, cf, cftmp) {
+ + ]
435 [ + + - + : 27 : DL_FOREACH_SAFE(cf->items, cit, cittmp) {
+ + ]
436 : 15 : r = snprintf(p, rlen, "%s", cit->name);
437 : 15 : p += r;
438 : 15 : rlen -= r;
439 : :
440 [ + + - + : 27 : DL_FOREACH_SAFE(cit->versions, cver, cvertmp) {
+ + ]
441 : 24 : r = snprintf(p, rlen, " %s %s", pkg_deps_op_tostring(cver->op),
442 : 12 : cver->ver);
443 : 12 : p += r;
444 : 12 : rlen -= r;
445 : 12 : }
446 : :
447 [ + + - + : 17 : DL_FOREACH_SAFE(cit->options, copt, copttmp) {
+ + ]
448 : 2 : r = snprintf(p, rlen, " %c%s", copt->on ? '+' : '-', copt->opt);
449 : 2 : p += r;
450 : 2 : rlen -= r;
451 : 2 : }
452 : :
453 : 15 : r = snprintf(p, rlen, "%s", cit->next ? " | " : "");
454 : 15 : p += r;
455 : 15 : rlen -= r;
456 : 15 : }
457 : :
458 : 12 : r = snprintf(p, rlen, "%s", cf->next ? ", " : "");
459 : 12 : p += r;
460 : 12 : rlen -= r;
461 : 12 : }
462 : :
463 : 7 : return (res);
464 : 7 : }
465 : :
466 : : char*
467 : 5 : pkg_deps_formula_tosql(struct pkg_dep_formula_item *f)
468 : : {
469 : : struct pkg_dep_formula_item *cit, *cittmp;
470 : : struct pkg_dep_version_item *cver, *cvertmp;
471 : 5 : char *res = NULL, *p;
472 : :
473 : 5 : int rlen = 0, r;
474 : :
475 [ + + - + : 12 : DL_FOREACH_SAFE(f, cit, cittmp) {
+ + ]
476 : 7 : rlen += sizeof("AND (name='' )");
477 : 7 : rlen += strlen(cit->name);
478 : :
479 [ + + - + : 11 : DL_FOREACH_SAFE(cit->versions, cver, cvertmp) {
+ + ]
480 : 4 : rlen += sizeof(" AND vercmp(>=, version,'') ");
481 : 4 : rlen += strlen(cver->ver);
482 : 4 : }
483 : :
484 : 7 : rlen += sizeof(" OR ");
485 : 7 : }
486 : :
487 [ + - ]: 5 : if (rlen == 0) {
488 : 0 : return (NULL);
489 : : }
490 : :
491 : 5 : res = xmalloc(rlen + 1);
492 : :
493 : 5 : p = res;
494 : :
495 [ + + - + : 12 : DL_FOREACH_SAFE(f, cit, cittmp) {
+ + ]
496 : 7 : r = snprintf(p, rlen, "(name='%s'", cit->name);
497 : 7 : p += r;
498 : 7 : rlen -= r;
499 : :
500 [ + + - + : 11 : DL_FOREACH_SAFE(cit->versions, cver, cvertmp) {
+ + ]
501 : 8 : r = snprintf(p, rlen, " AND vercmp('%s',version,'%s')",
502 : 4 : pkg_deps_op_tostring(cver->op),
503 : 4 : cver->ver);
504 : 4 : p += r;
505 : 4 : rlen -= r;
506 : 4 : }
507 : 7 : r = snprintf(p, rlen, ")%s", cit->next ? " OR " : "");
508 : 7 : p += r;
509 : 7 : rlen -= r;
510 : 7 : }
511 : :
512 : 5 : return (res);
513 : 5 : }
514 : :
515 : : enum pkg_dep_version_op
516 : 11 : pkg_deps_string_toop(const char *in)
517 : : {
518 : 11 : enum pkg_dep_version_op ret = VERSION_ANY;
519 : : int len;
520 : :
521 [ + + ]: 11 : if (in != NULL) {
522 : 10 : len = strlen(in);
523 : :
524 [ + + ]: 10 : if (len == 2) {
525 [ + + ]: 5 : if (memcmp(in, ">=", 2) == 0) {
526 : 1 : ret = VERSION_GE;
527 : 1 : }
528 [ + + ]: 4 : else if (memcmp(in, "<=", 2) == 0) {
529 : 1 : ret = VERSION_LE;
530 : 1 : }
531 [ + + ]: 3 : else if (memcmp(in, "!=", 2) == 0) {
532 : 1 : ret = VERSION_NOT;
533 : 1 : }
534 [ + + ]: 2 : else if (memcmp(in, "==", 2) == 0) {
535 : 1 : ret = VERSION_EQ;
536 : 1 : }
537 : 5 : }
538 [ - + ]: 5 : else if (len == 1) {
539 [ + + ]: 5 : if (*in == '>') {
540 : 1 : ret = VERSION_GT;
541 : 1 : }
542 [ + + ]: 4 : else if (*in == '<') {
543 : 1 : ret = VERSION_LT;
544 : 1 : }
545 [ + + ]: 3 : else if (*in == '!') {
546 : 1 : ret = VERSION_NOT;
547 : 1 : }
548 [ + + ]: 2 : else if (*in == '=') {
549 : 1 : ret = VERSION_EQ;
550 : 1 : }
551 : 5 : }
552 : 10 : }
553 : :
554 : 11 : return (ret);
555 : : }
|