- add repo_add_poolstr_array
[platform/upstream/libsolv.git] / tools / repo_updateinfoxml.c
1 /*
2  * Copyright (c) 2007, Novell Inc.
3  *
4  * This program is licensed under the BSD license, read LICENSE.BSD
5  * for further information
6  */
7
8 #define DO_ARRAY 1
9
10 #define _GNU_SOURCE
11 #include <sys/types.h>
12 #include <limits.h>
13 #include <fcntl.h>
14 #include <stdio.h>
15 #include <stdlib.h>
16 #include <string.h>
17 #include <expat.h>
18
19 #include "pool.h"
20 #include "repo.h"
21 #include "repo_updateinfoxml.h"
22 #include "tools_util.h"
23
24 /*
25  * <updates>
26  *   <update from="rel-eng@fedoraproject.org" status="stable" type="security" version="1.4">
27  *     <id>FEDORA-2007-4594</id>
28  *     <title>imlib-1.9.15-6.fc8</title>
29  *     <release>Fedora 8</release>
30  *     <issued date="2007-12-28 16:42:30"/>
31  *     <references>
32  *       <reference href="https://bugzilla.redhat.com/show_bug.cgi?id=426091" id="426091" title="CVE-2007-3568 imlib: infinite loop DoS using crafted BMP image" type="bugzilla"/>
33  *     </references>
34  *     <description>This update includes a fix for a denial-of-service issue (CVE-2007-3568) whereby an attacker who could get an imlib-using user to view a  specially-crafted BMP image could cause the user's CPU to go into an infinite loop.</description>
35  *     <pkglist>
36  *       <collection short="F8">
37  *         <name>Fedora 8</name>
38  *         <package arch="ppc64" name="imlib-debuginfo" release="6.fc8" src="http://download.fedoraproject.org/pub/fedora/linux/updates/8/ppc64/imlib-debuginfo-1.9.15-6.fc8.ppc64.rpm" version="1.9.15">
39  *           <filename>imlib-debuginfo-1.9.15-6.fc8.ppc64.rpm</filename>
40  *           <reboot_suggested>True</reboot_suggested>
41  *         </package>
42  *       </collection>
43  *     </pkglist>
44  *   </update>
45  * </updates>
46 */
47
48 enum state {
49   STATE_START,
50   STATE_UPDATES,      /* 1 */
51   STATE_UPDATE,       /* 2 */
52   STATE_ID,           /* 3 */
53   STATE_TITLE,        /* 4 */
54   STATE_RELEASE,      /* 5 */
55   STATE_ISSUED,       /* 6 */
56   STATE_MESSAGE,      /* 7 */
57   STATE_REFERENCES,   /* 8 */
58   STATE_REFERENCE,    /* 9 */
59   STATE_DESCRIPTION,  /* 10 */
60   STATE_PKGLIST,     /* 11 */
61   STATE_COLLECTION,  /* 12 */
62   STATE_NAME,        /* 13 */
63   STATE_PACKAGE,     /* 14 */
64   STATE_FILENAME,    /* 15 */
65   STATE_REBOOT,      /* 16 */
66   STATE_RESTART,     /* 17 */
67   STATE_RELOGIN,     /* 18 */
68   NUMSTATES
69 };
70
71 struct stateswitch {
72   enum state from;
73   char *ename;
74   enum state to;
75   int docontent;
76 };
77
78
79 /* !! must be sorted by first column !! */
80 static struct stateswitch stateswitches[] = {
81   { STATE_START,       "updates",         STATE_UPDATES,     0 },
82   { STATE_START,       "update",          STATE_UPDATE,      0 },
83   { STATE_UPDATES,     "update",          STATE_UPDATE,      0 },
84   { STATE_UPDATE,      "id",              STATE_ID,          1 },
85   { STATE_UPDATE,      "title",           STATE_TITLE,       1 },
86   { STATE_UPDATE,      "release",         STATE_RELEASE,     1 },
87   { STATE_UPDATE,      "issued",          STATE_ISSUED,      1 },
88   { STATE_UPDATE,      "description",     STATE_DESCRIPTION, 1 },
89   { STATE_UPDATE,      "message",         STATE_MESSAGE    , 1 },
90   { STATE_UPDATE,      "references",      STATE_REFERENCES,  0 },
91   { STATE_UPDATE,      "pkglist",         STATE_PKGLIST,     0 },
92   { STATE_REFERENCES,  "reference",       STATE_REFERENCE,   0 },
93   { STATE_PKGLIST,     "collection",      STATE_COLLECTION,  0 },
94   { STATE_COLLECTION,  "name",            STATE_NAME,        1 },
95   { STATE_COLLECTION,  "package",         STATE_PACKAGE,     0 },
96   { STATE_PACKAGE,     "filename",        STATE_FILENAME,    1 },
97   { STATE_PACKAGE,     "reboot_suggested",STATE_REBOOT,      1 },
98   { STATE_PACKAGE,     "restart_suggested",STATE_RESTART,    1 },
99   { STATE_PACKAGE,     "relogin_suggested",STATE_RELOGIN,    1 },
100   { NUMSTATES }
101 };
102
103 struct parsedata {
104   int depth;
105   enum state state;
106   int statedepth;
107   char *content;
108   int lcontent;
109   int acontent;
110   int docontent;
111   Pool *pool;
112   Repo *repo;
113   Repodata *data;
114   unsigned int datanum;
115   Solvable *solvable;
116   unsigned int timestamp;
117   
118
119   struct stateswitch *swtab[NUMSTATES];
120   enum state sbtab[NUMSTATES];
121   char *tempstr;
122   int ltemp;
123   int atemp;
124 };
125
126 /*
127  * if we have seen a <filename>...
128  * inside of <package>...
129  * 
130  *
131  * If not, we must insert an empty filename to UPDATE_COLLECTION_FILENAME
132  * at </package> in order to keep all UPDATE_COLLECTION_* arrays in sync
133  */
134
135 static int package_filename_seen = 0;
136 static int package_flags = 0; /* same for reboot/restart flags, to be written at </package> */
137
138 /*
139  * create evr (as Id) from 'epoch', 'version' and 'release' attributes
140  */
141
142 static Id
143 makeevr_atts(Pool *pool, struct parsedata *pd, const char **atts)
144 {
145   const char *e, *v, *r, *v2;
146   char *c;
147   int l;
148
149   e = v = r = 0;
150   for (; *atts; atts += 2)
151     {
152       if (!strcmp(*atts, "epoch"))
153         e = atts[1];
154       else if (!strcmp(*atts, "version"))
155         v = atts[1];
156       else if (!strcmp(*atts, "release"))
157         r = atts[1];
158     }
159   if (e && !strcmp(e, "0"))
160     e = 0;
161   if (v && !e)
162     {
163       for (v2 = v; *v2 >= '0' && *v2 <= '9'; v2++)
164         ;
165       if (v2 > v && *v2 == ':')
166         e = "0";
167     }
168   l = 1;
169   if (e)
170     l += strlen(e) + 1;
171   if (v)
172     l += strlen(v);
173   if (r)
174     l += strlen(r) + 1;
175   if (l > pd->acontent)
176     {
177       pd->content = realloc(pd->content, l + 256);
178       pd->acontent = l + 256;
179     }
180   c = pd->content;
181   if (e)
182     {
183       strcpy(c, e);
184       c += strlen(c);
185       *c++ = ':';
186     }
187   if (v)
188     {
189       strcpy(c, v);
190       c += strlen(c);
191     }
192   if (r)
193     {
194       *c++ = '-';
195       strcpy(c, r);
196       c += strlen(c);
197     }
198   *c = 0;
199   if (!*pd->content)
200     return 0;
201 #if 0
202   fprintf(stderr, "evr: %s\n", pd->content);
203 #endif
204   return str2id(pool, pd->content, 1);
205 }
206
207
208
209 static void XMLCALL
210 startElement(void *userData, const char *name, const char **atts)
211 {
212   struct parsedata *pd = userData;
213   Pool *pool = pd->pool;
214   Solvable *solvable = pd->solvable;
215   struct stateswitch *sw;
216   /*const char *str; */
217
218 #if 0
219       fprintf(stderr, "start: [%d]%s\n", pd->state, name);
220 #endif
221   if (pd->depth != pd->statedepth)
222     {
223       pd->depth++;
224       return;
225     }
226
227   pd->depth++;
228   for (sw = pd->swtab[pd->state]; sw->from == pd->state; sw++)  /* find name in statetable */
229     if (!strcmp(sw->ename, name))
230       break;
231   
232
233   if (sw->from != pd->state)
234     {
235 #if 1
236       fprintf(stderr, "into unknown: [%d]%s (from: %d)\n", sw->to, name, sw->from);
237       exit( 1 );
238 #endif
239       return;
240     }
241   pd->state = sw->to;
242   pd->docontent = sw->docontent;
243   pd->statedepth = pd->depth;
244   pd->lcontent = 0;
245   *pd->content = 0;
246
247   switch(pd->state)
248     {
249       case STATE_START:
250       break;
251       case STATE_UPDATES:
252       break;
253       /*
254        * <update from="rel-eng@fedoraproject.org"
255        *         status="stable"
256        *         type="bugfix" (enhancement, security)
257        *         version="1.4">
258        */
259       case STATE_UPDATE:
260       {
261         const char *from = 0, *status = 0, *type = 0, *version = 0;
262         for (; *atts; atts += 2)
263         {
264           if (!strcmp(*atts, "from"))
265             from = atts[1];
266           else if (!strcmp(*atts, "status"))
267             status = atts[1];
268           else if (!strcmp(*atts, "type"))
269             type = atts[1];
270           else if (!strcmp(*atts, "version"))
271             version = atts[1];
272         }
273         
274
275         solvable = pd->solvable = pool_id2solvable(pool, repo_add_solvable(pd->repo));
276         pd->datanum = (pd->solvable - pool->solvables) - pd->repo->start;
277         repodata_extend(pd->data, pd->solvable - pool->solvables);      
278         repodata_extend(pd->data, pd->solvable - pool->solvables);
279         pd->datanum = repodata_get_handle(pd->data, pd->datanum);
280         
281
282         solvable->vendor = str2id(pool, from, 1);
283         solvable->evr = str2id(pool, version, 1);
284         solvable->arch = ARCH_NOARCH;
285         repodata_set_str(pd->data, pd->datanum, SOLVABLE_PATCHCATEGORY, type);
286       }
287       break;
288       /* <id>FEDORA-2007-4594</id> */
289       case STATE_ID:
290       break;
291       /* <title>imlib-1.9.15-6.fc8</title> */
292       case STATE_TITLE:
293       break;
294       /* <release>Fedora 8</release> */
295       case STATE_RELEASE:
296       break;
297       /*  <issued date="2008-03-21 21:36:55"/>
298       */
299       case STATE_ISSUED:
300       {
301         const char *date = 0;
302         for (; *atts; atts += 2)
303         {
304           if (!strcmp(*atts, "date"))
305             date = atts[1];
306         }
307         repodata_set_str(pd->data, pd->datanum, SOLVABLE_BUILDTIME, date);
308       }
309       break;
310       case STATE_REFERENCES:
311       break;
312       /*  <reference href="https://bugzilla.redhat.com/show_bug.cgi?id=330471"
313        *             id="330471"
314        *             title="LDAP schema file missing for dhcpd"
315        *             type="bugzilla"/>
316        */
317       case STATE_REFERENCE:
318       {
319         const char *href = 0, *id = 0, *title = 0, *type = 0;
320         for (; *atts; atts += 2)
321         {
322           if (!strcmp(*atts, "href"))
323             href = atts[1];
324           else if (!strcmp(*atts, "id"))
325             id = atts[1];
326           else if (!strcmp(*atts, "title"))
327             title = atts[1];
328           else if (!strcmp(*atts, "type"))
329             type = atts[1];
330         }
331 #if DO_ARRAY
332         repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_REFERENCE_HREF, href);
333         repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_REFERENCE_ID, id);
334         repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_REFERENCE_TITLE, title);
335         repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_REFERENCE_TYPE, type);
336 #endif
337       }
338       break;
339       /* <description>This update ...</description> */
340       case STATE_DESCRIPTION:
341       break;
342       /* <message type="confirm">This update ...</message> */
343       case STATE_MESSAGE:
344       break;
345       case STATE_PKGLIST:
346       break;
347       /* <collection short="F8" */
348       case STATE_COLLECTION:
349       break;
350       /* <name>Fedora 8</name> */ 
351       case STATE_NAME:
352       break;
353       /*   <package arch="ppc64" name="imlib-debuginfo" release="6.fc8"
354        *            src="http://download.fedoraproject.org/pub/fedora/linux/updates/8/ppc64/imlib-debuginfo-1.9.15-6.fc8.ppc64.rpm"
355        *            version="1.9.15">
356        * 
357        *
358        * -> patch.conflicts: {name} < {version}.{release}
359        */
360       case STATE_PACKAGE:
361       {
362         const char *arch = 0, *name = 0, *src = 0;
363         Id evr = makeevr_atts(pool, pd, atts); /* parse "epoch", "version", "release" */
364         Id n, a, na;
365         Id rel_id;
366         
367
368         /* reset package_* markers, to be evaluated at </package> */
369         package_filename_seen = 0;
370         package_flags = 0;
371         
372
373         for (; *atts; atts += 2)
374         {
375           if (!strcmp(*atts, "arch"))
376             arch = atts[1];
377           else if (!strcmp(*atts, "name"))
378             name = atts[1];
379           else if (!strcmp(*atts, "src"))
380             src = atts[1];
381         }
382         /* generated Ids for name and arch */
383         n = str2id(pool, name, 1);
384         if (arch)
385           a = str2id(pool, arch, 1);
386         else
387           a = ARCH_NOARCH;
388         /*  now combine both to a single Id */
389         na = rel2id(pool, n, a, REL_ARCH, 1);
390         
391
392         rel_id = rel2id(pool, na, evr, REL_LT, 1);
393
394         solvable->conflicts = repo_addid_dep(pd->repo, solvable->conflicts, rel_id, 0);
395 #if DO_ARRAY
396         repodata_add_idarray(pd->data, pd->datanum, UPDATE_COLLECTION_NAME, n);
397         repodata_add_idarray(pd->data, pd->datanum, UPDATE_COLLECTION_EVR, evr);
398         repodata_add_idarray(pd->data, pd->datanum, UPDATE_COLLECTION_ARCH, a);
399 #else
400         /* _FILENAME and _FLAGS are written at </package> */
401         if (1) {
402           const char *evrstr = id2str(pool, evr);
403           int buflen = strlen(name) + 1 + strlen(evrstr) + 1 + strlen(arch?arch:"") + 1;
404           char *buf;
405           if (!arch) arch = "";
406           buf = (char *)malloc(buflen);
407           if (!buf) exit(1);
408           sprintf(buf, "%s %s %s", name, evrstr, arch);
409           repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_COLLECTION, buf);
410           free(buf);
411         }
412 #endif
413       }
414       break;
415       /* <filename>libntlm-0.4.2-1.fc8.x86_64.rpm</filename> */ 
416       /* <filename>libntlm-0.4.2-1.fc8.x86_64.rpm</filename> */
417       case STATE_FILENAME:
418       break;
419       /* <reboot_suggested>True</reboot_suggested> */
420       case STATE_REBOOT:
421       break;
422       /* <restart_suggested>True</restart_suggested> */
423       case STATE_RESTART:
424       break;
425       /* <relogin_suggested>True</relogin_suggested> */
426       case STATE_RELOGIN:
427       break;
428       case NUMSTATES+1:
429         split(NULL, NULL, 0); /* just to keep gcc happy about tools_util.h: static ... split() {...}  Urgs!*/
430       break;
431       default:
432       break;
433     }
434   return;
435 }
436
437
438 static void XMLCALL
439 endElement(void *userData, const char *name)
440 {
441   struct parsedata *pd = userData;
442   Pool *pool = pd->pool;
443   Solvable *s = pd->solvable;
444   Repo *repo = pd->repo;
445
446 #if 0
447       fprintf(stderr, "end: %s\n", name);
448 #endif
449   if (pd->depth != pd->statedepth)
450     {
451       pd->depth--;
452 #if 1
453       fprintf(stderr, "back from unknown %d %d %d\n", pd->state, pd->depth, pd->statedepth);
454 #endif
455       return;
456     }
457
458   pd->depth--;
459   pd->statedepth--;
460   switch (pd->state)
461     {
462       case STATE_START:
463       break;
464       case STATE_UPDATES:
465       break;
466       case STATE_UPDATE:
467       s->provides = repo_addid_dep(repo, s->provides, rel2id(pool, s->name, s->evr, REL_EQ, 1), 0);
468       break;
469       case STATE_ID:
470       {
471         if (pd->content) {
472           s->name = str2id(pool, join2("patch", ":", pd->content), 1);
473         }
474       }
475       break;
476       /* <title>imlib-1.9.15-6.fc8</title> */
477       case STATE_TITLE:
478       {
479         while (pd->lcontent > 0
480                && *(pd->content + pd->lcontent - 1) == '\n')
481         {
482           --pd->lcontent;
483           *(pd->content + pd->lcontent) = 0;
484         }
485         repodata_set_str(pd->data, pd->datanum, SOLVABLE_SUMMARY, pd->content);
486       }
487       break;
488       /*
489        * <release>Fedora 8</release>
490        */
491       case STATE_RELEASE:
492       break;
493       case STATE_ISSUED:
494       break;
495       case STATE_REFERENCES:
496       break;
497       case STATE_REFERENCE:
498       break;
499       /*
500        * <description>This update ...</description>
501        */
502       case STATE_DESCRIPTION:
503       {
504         repodata_set_str(pd->data, pd->datanum, SOLVABLE_DESCRIPTION, pd->content);
505       }
506       break;   
507       break;
508       /*
509        * <message>Warning! ...</message>
510        */
511       case STATE_MESSAGE:
512       {
513         repodata_set_str(pd->data, pd->datanum, UPDATE_MESSAGE, pd->content);
514       }
515       break;
516       case STATE_PKGLIST:
517       break;
518       case STATE_COLLECTION:
519       break;
520       case STATE_NAME:
521       break;
522       case STATE_PACKAGE:
523       {
524 #if DO_ARRAY
525         /* write _FILENAME and _FLAGS at </package>
526          * to ensure all UPDATE_COLLECTION_* arrays are filled in parallel
527          */
528         if (!package_filename_seen)
529         {
530           repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_COLLECTION_FILENAME, "");
531         }
532         repodata_add_idarray(pd->data, pd->datanum, UPDATE_COLLECTION_FLAGS, package_flags+1);
533 #endif
534       }
535       break;
536       /* <filename>libntlm-0.4.2-1.fc8.x86_64.rpm</filename> */ 
537       /* <filename>libntlm-0.4.2-1.fc8.x86_64.rpm</filename> */
538       case STATE_FILENAME:
539       {
540 #if DO_ARRAY
541         repodata_add_poolstr_array(pd->data, pd->datanum, UPDATE_COLLECTION_FILENAME, pd->content);
542         package_filename_seen = 1;
543 #endif
544       }
545       break;
546       /* <reboot_suggested>True</reboot_suggested> */
547       case STATE_REBOOT:
548       {
549         if (pd->content
550             && (pd->content[0] == 'T'
551                 || pd->content[0] == 't'|| pd->content[0] == '1'))
552         {
553           /* FIXME: this is per-package, the global flag should be computed at runtime */
554           repodata_set_void(pd->data, pd->datanum, UPDATE_REBOOT);
555           package_flags = 1;
556         }
557       }
558       break;
559       /* <restart_suggested>True</restart_suggested> */
560       case STATE_RESTART:
561       {
562         if (pd->content
563             && (pd->content[0] == 'T'
564                 || pd->content[0] == 't' || pd->content[0] == '1'))
565         {
566           /* FIXME: this is per-package, the global flag should be computed at runtime */
567           repodata_set_void(pd->data, pd->datanum, UPDATE_RESTART);
568           package_flags = 2;
569         }
570       }
571       break;
572       /* <relogin_suggested>True</relogin_suggested> */
573       case STATE_RELOGIN:
574       {
575         if (pd->content
576             && (pd->content[0] == 'T'
577                 || pd->content[0] == 't' || pd->content[0] == '1'))
578         {
579           /* FIXME: this is per-package, the global flag should be computed at runtime */
580           repodata_set_void(pd->data, pd->datanum, UPDATE_RELOGIN);
581           package_flags = 2;
582         }
583       }
584       break;
585       default:
586       break;
587     }
588
589   pd->state = pd->sbtab[pd->state];
590   pd->docontent = 0;
591   
592
593   return;
594 }
595
596
597 static void XMLCALL
598 characterData(void *userData, const XML_Char *s, int len)
599 {
600   struct parsedata *pd = userData;
601   int l;
602   char *c;
603   if (!pd->docontent) {
604 #if 0
605     char *dup = strndup( s, len );
606   fprintf(stderr, "Content: [%d]'%s'\n", pd->state, dup );
607   free( dup );
608 #endif
609     return;
610   }
611   l = pd->lcontent + len + 1;
612   if (l > pd->acontent)
613     {
614       pd->content = realloc(pd->content, l + 256);
615       pd->acontent = l + 256;
616     }
617   c = pd->content + pd->lcontent;
618   pd->lcontent += len;
619   while (len-- > 0)
620     *c++ = *s++;
621   *c = 0;
622 }
623
624
625 #define BUFF_SIZE 8192
626
627 void
628 repo_add_updateinfoxml(Repo *repo, FILE *fp, int flags)
629 {
630   Pool *pool = repo->pool;
631   struct parsedata pd;
632   char buf[BUFF_SIZE];
633   int i, l;
634   struct stateswitch *sw;
635
636   memset(&pd, 0, sizeof(pd));
637   for (i = 0, sw = stateswitches; sw->from != NUMSTATES; i++, sw++)
638     {
639       if (!pd.swtab[sw->from])
640         pd.swtab[sw->from] = sw;
641       pd.sbtab[sw->to] = sw->from;
642     }
643   pd.pool = pool;
644   pd.repo = repo;
645   pd.data = repo_add_repodata(pd.repo, 0);
646
647   pd.content = malloc(256);
648   pd.acontent = 256;
649   pd.lcontent = 0;
650   pd.tempstr = malloc(256);
651   pd.atemp = 256;
652   pd.ltemp = 0;
653   XML_Parser parser = XML_ParserCreate(NULL);
654   XML_SetUserData(parser, &pd);
655   XML_SetElementHandler(parser, startElement, endElement);
656   XML_SetCharacterDataHandler(parser, characterData);
657   for (;;)
658     {
659       l = fread(buf, 1, sizeof(buf), fp);
660       if (XML_Parse(parser, buf, l, l == 0) == XML_STATUS_ERROR)
661         {
662           fprintf(stderr, "repo_updateinfoxml: %s at line %u:%u\n", XML_ErrorString(XML_GetErrorCode(parser)), (unsigned int)XML_GetCurrentLineNumber(parser), (unsigned int)XML_GetCurrentColumnNumber(parser));
663           exit(1);
664         }
665       if (l == 0)
666         break;
667     }
668   XML_ParserFree(parser);
669
670   if (pd.data)
671     repodata_internalize(pd.data);
672
673   free(pd.content);
674   join_freemem();
675 }
676
677 /* EOF */