Imported Upstream version 0.7.4
[platform/upstream/libsolv.git] / ext / 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 _GNU_SOURCE
9 #define _XOPEN_SOURCE /* glibc2 needs this */
10 #include <sys/types.h>
11 #include <stdio.h>
12 #include <stdlib.h>
13 #include <string.h>
14 #include <time.h>
15
16 #include "pool.h"
17 #include "repo.h"
18 #include "solv_xmlparser.h"
19 #include "repo_updateinfoxml.h"
20 #define DISABLE_SPLIT
21 #include "tools_util.h"
22
23 /*
24  * <updates>
25  *   <update from="rel-eng@fedoraproject.org" status="stable" type="security" version="1.4">
26  *     <id>FEDORA-2007-4594</id>
27  *     <title>imlib-1.9.15-6.fc8</title>
28  *     <severity>Important</severity>
29  *     <release>Fedora 8</release>
30  *     <rights>Copyright 2007 Company Inc</rights>
31  *     <issued date="2007-12-28 16:42:30"/>
32  *     <updated date="2008-03-14 12:00:00"/>
33  *     <references>
34  *       <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"/>
35  *     </references>
36  *     <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>
37  *     <pkglist>
38  *       <collection short="F8">
39  *         <name>Fedora 8</name>
40  *         <module name="pki-deps" stream="10.6" version="20181019123559" context="9edba152" arch="x86_64"/>
41  *         <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">
42  *           <filename>imlib-debuginfo-1.9.15-6.fc8.ppc64.rpm</filename>
43  *           <reboot_suggested>True</reboot_suggested>
44  *         </package>
45  *       </collection>
46  *     </pkglist>
47  *   </update>
48  * </updates>
49 */
50
51 enum state {
52   STATE_START,
53   STATE_UPDATES,
54   STATE_UPDATE,
55   STATE_ID,
56   STATE_TITLE,
57   STATE_RELEASE,
58   STATE_ISSUED,
59   STATE_UPDATED,
60   STATE_MESSAGE,
61   STATE_REFERENCES,
62   STATE_REFERENCE,
63   STATE_DESCRIPTION,
64   STATE_PKGLIST,
65   STATE_COLLECTION,
66   STATE_NAME,
67   STATE_PACKAGE,
68   STATE_FILENAME,
69   STATE_REBOOT,
70   STATE_RESTART,
71   STATE_RELOGIN,
72   STATE_RIGHTS,
73   STATE_SEVERITY,
74   STATE_MODULE,
75   NUMSTATES
76 };
77
78 static struct solv_xmlparser_element stateswitches[] = {
79   { STATE_START,       "updates",         STATE_UPDATES,     0 },
80   { STATE_START,       "update",          STATE_UPDATE,      0 },
81   { STATE_UPDATES,     "update",          STATE_UPDATE,      0 },
82   { STATE_UPDATE,      "id",              STATE_ID,          1 },
83   { STATE_UPDATE,      "title",           STATE_TITLE,       1 },
84   { STATE_UPDATE,      "severity",        STATE_SEVERITY,    1 },
85   { STATE_UPDATE,      "rights",          STATE_RIGHTS,      1 },
86   { STATE_UPDATE,      "release",         STATE_RELEASE,     1 },
87   { STATE_UPDATE,      "issued",          STATE_ISSUED,      0 },
88   { STATE_UPDATE,      "updated",         STATE_UPDATED,     0 },
89   { STATE_UPDATE,      "description",     STATE_DESCRIPTION, 1 },
90   { STATE_UPDATE,      "message",         STATE_MESSAGE    , 1 },
91   { STATE_UPDATE,      "references",      STATE_REFERENCES,  0 },
92   { STATE_UPDATE,      "pkglist",         STATE_PKGLIST,     0 },
93   { STATE_REFERENCES,  "reference",       STATE_REFERENCE,   0 },
94   { STATE_PKGLIST,     "collection",      STATE_COLLECTION,  0 },
95   { STATE_COLLECTION,  "name",            STATE_NAME,        1 },
96   { STATE_COLLECTION,  "package",         STATE_PACKAGE,     0 },
97   { STATE_COLLECTION,  "module",          STATE_MODULE,      0 },
98   { STATE_PACKAGE,     "filename",        STATE_FILENAME,    1 },
99   { STATE_PACKAGE,     "reboot_suggested",STATE_REBOOT,      1 },
100   { STATE_PACKAGE,     "restart_suggested",STATE_RESTART,    1 },
101   { STATE_PACKAGE,     "relogin_suggested",STATE_RELOGIN,    1 },
102   { NUMSTATES }
103 };
104
105 struct parsedata {
106   int ret;
107   Pool *pool;
108   Repo *repo;
109   Repodata *data;
110   Id handle;
111   Solvable *solvable;
112   time_t buildtime;
113   Id collhandle;
114   struct solv_xmlparser xmlp;
115   struct joindata jd;
116 };
117
118 /*
119  * Convert date strings ("1287746075" or "2010-10-22 13:14:35")
120  * to timestamp.
121  */
122 static time_t
123 datestr2timestamp(const char *date)
124 {
125   const char *p;
126   struct tm tm;
127
128   if (!date || !*date)
129     return 0;
130   for (p = date; *p >= '0' && *p <= '9'; p++)
131     ;
132   if (!*p)
133     return atoi(date);
134   memset(&tm, 0, sizeof(tm));
135   if (!strptime(date, "%F%T", &tm))
136     return 0;
137   return timegm(&tm);
138 }
139
140 /*
141  * create evr (as Id) from 'epoch', 'version' and 'release' attributes
142  */
143 static Id
144 makeevr_atts(Pool *pool, struct parsedata *pd, const char **atts)
145 {
146   const char *e, *v, *r, *v2;
147   char *c, *space;
148   int l;
149
150   e = v = r = 0;
151   for (; *atts; atts += 2)
152     {
153       if (!strcmp(*atts, "epoch"))
154         e = atts[1];
155       else if (!strcmp(*atts, "version"))
156         v = atts[1];
157       else if (!strcmp(*atts, "release"))
158         r = atts[1];
159     }
160   if (e && (!*e || !strcmp(e, "0")))
161     e = 0;
162   if (v && !e)
163     {
164       for (v2 = v; *v2 >= '0' && *v2 <= '9'; v2++)
165         ;
166       if (v2 > v && *v2 == ':')
167         e = "0";
168     }
169   l = 1;
170   if (e)
171     l += strlen(e) + 1;
172   if (v)
173     l += strlen(v);
174   if (r)
175     l += strlen(r) + 1;
176   
177   c = space = solv_xmlparser_contentspace(&pd->xmlp, l);
178   if (e)
179     {
180       strcpy(c, e);
181       c += strlen(c);
182       *c++ = ':';
183     }
184   if (v)
185     {
186       strcpy(c, v);
187       c += strlen(c);
188     }
189   if (r)
190     {
191       *c++ = '-';
192       strcpy(c, r);
193       c += strlen(c);
194     }
195   *c = 0;
196   if (!*space)
197     return 0;
198 #if 0
199   fprintf(stderr, "evr: %s\n", space);
200 #endif
201   return pool_str2id(pool, space, 1);
202 }
203
204
205
206 static void
207 startElement(struct solv_xmlparser *xmlp, int state, const char *name, const char **atts)
208 {
209   struct parsedata *pd = xmlp->userdata;
210   Pool *pool = pd->pool;
211   Solvable *solvable = pd->solvable;
212
213   switch(state)
214     {
215       /*
216        * <update from="rel-eng@fedoraproject.org"
217        *         status="stable"
218        *         type="bugfix" (enhancement, security)
219        *         version="1.4">
220        */
221     case STATE_UPDATE:
222       {
223         const char *from = 0, *type = 0, *version = 0;
224         for (; *atts; atts += 2)
225           {
226             if (!strcmp(*atts, "from"))
227               from = atts[1];
228             else if (!strcmp(*atts, "type"))
229               type = atts[1];
230             else if (!strcmp(*atts, "version"))
231               version = atts[1];
232           }
233         solvable = pd->solvable = pool_id2solvable(pool, repo_add_solvable(pd->repo));
234         pd->handle = pd->solvable - pool->solvables;
235
236         solvable->vendor = pool_str2id(pool, from, 1);
237         solvable->evr = pool_str2id(pool, version, 1);
238         solvable->arch = ARCH_NOARCH;
239         if (type)
240           repodata_set_str(pd->data, pd->handle, SOLVABLE_PATCHCATEGORY, type);
241         pd->buildtime = (time_t)0;
242       }
243       break;
244
245     case STATE_ISSUED:
246     case STATE_UPDATED:
247       {
248         const char *date = solv_xmlparser_find_attr("date", atts);
249         if (date)
250           {
251             time_t t = datestr2timestamp(date);
252             if (t && t > pd->buildtime)
253               pd->buildtime = t;
254           }
255       }
256       break;
257
258     case STATE_REFERENCE:
259       {
260         const char *href = 0, *id = 0, *title = 0, *type = 0;
261         Id refhandle;
262         for (; *atts; atts += 2)
263           {
264             if (!strcmp(*atts, "href"))
265               href = atts[1];
266             else if (!strcmp(*atts, "id"))
267               id = atts[1];
268             else if (!strcmp(*atts, "title"))
269               title = atts[1];
270             else if (!strcmp(*atts, "type"))
271               type = atts[1];
272           }
273         refhandle = repodata_new_handle(pd->data);
274         if (href)
275           repodata_set_str(pd->data, refhandle, UPDATE_REFERENCE_HREF, href);
276         if (id)
277           repodata_set_str(pd->data, refhandle, UPDATE_REFERENCE_ID, id);
278         if (title)
279           repodata_set_str(pd->data, refhandle, UPDATE_REFERENCE_TITLE, title);
280         if (type)
281           repodata_set_poolstr(pd->data, refhandle, UPDATE_REFERENCE_TYPE, type);
282         repodata_add_flexarray(pd->data, pd->handle, UPDATE_REFERENCE, refhandle);
283       }
284       break;
285
286       /*   <package arch="ppc64" name="imlib-debuginfo" release="6.fc8"
287        *            src="http://download.fedoraproject.org/pub/fedora/linux/updates/8/ppc64/imlib-debuginfo-1.9.15-6.fc8.ppc64.rpm"
288        *            version="1.9.15">
289        *
290        *
291        * -> patch.conflicts: {name} < {version}.{release}
292        */
293     case STATE_PACKAGE:
294       {
295         const char *arch = 0, *name = 0;
296         Id evr = makeevr_atts(pool, pd, atts); /* parse "epoch", "version", "release" */
297         Id n, a = 0;
298         Id rel_id;
299
300         for (; *atts; atts += 2)
301           {
302             if (!strcmp(*atts, "arch"))
303               arch = atts[1];
304             else if (!strcmp(*atts, "name"))
305               name = atts[1];
306           }
307         /* generated Id for name */
308         n = pool_str2id(pool, name, 1);
309         rel_id = n;
310         if (arch)
311           {
312             /*  generate Id for arch and combine with name */
313             a = pool_str2id(pool, arch, 1);
314             rel_id = pool_rel2id(pool, n, a, REL_ARCH, 1);
315           }
316         rel_id = pool_rel2id(pool, rel_id, evr, REL_LT, 1);
317         solvable->conflicts = repo_addid_dep(pd->repo, solvable->conflicts, rel_id, 0);
318
319         /* who needs the collection anyway? */
320         pd->collhandle = repodata_new_handle(pd->data);
321         repodata_set_id(pd->data, pd->collhandle, UPDATE_COLLECTION_NAME, n);
322         repodata_set_id(pd->data, pd->collhandle, UPDATE_COLLECTION_EVR, evr);
323         if (a)
324           repodata_set_id(pd->data, pd->collhandle, UPDATE_COLLECTION_ARCH, a);
325         break;
326       }
327     case STATE_MODULE:
328       {
329         const char *name = 0, *stream = 0, *version = 0, *context = 0, *arch = 0;
330         Id module_handle;
331
332         for (; *atts; atts += 2)
333           {
334             if (!strcmp(*atts, "arch"))
335               arch = atts[1];
336             else if (!strcmp(*atts, "name"))
337               name = atts[1];
338             else if (!strcmp(*atts, "stream"))
339               stream = atts[1];
340             else if (!strcmp(*atts, "version"))
341               version = atts[1];
342             else if (!strcmp(*atts, "context"))
343               context = atts[1];
344           }
345         module_handle = repodata_new_handle(pd->data);
346         if (name)
347           repodata_set_poolstr(pd->data, module_handle, UPDATE_MODULE_NAME, name);
348         if (stream)
349           repodata_set_poolstr(pd->data, module_handle, UPDATE_MODULE_STREAM, stream);
350         if (version)
351           repodata_set_poolstr(pd->data, module_handle, UPDATE_MODULE_VERSION, version);
352         if (context)
353           repodata_set_poolstr(pd->data, module_handle, UPDATE_MODULE_CONTEXT, context);
354         if (arch)
355           repodata_set_poolstr(pd->data, module_handle, UPDATE_MODULE_ARCH, arch);
356         repodata_add_flexarray(pd->data, pd->handle, UPDATE_MODULE, module_handle);
357         break;
358       }
359
360     default:
361       break;
362     }
363   return;
364 }
365
366
367 static void
368 endElement(struct solv_xmlparser *xmlp, int state, char *content)
369 {
370   struct parsedata *pd = xmlp->userdata;
371   Pool *pool = pd->pool;
372   Solvable *s = pd->solvable;
373   Repo *repo = pd->repo;
374
375   switch (state)
376     {
377     case STATE_UPDATE:
378       s->provides = repo_addid_dep(repo, s->provides, pool_rel2id(pool, s->name, s->evr, REL_EQ, 1), 0);
379       if (pd->buildtime)
380         {
381           repodata_set_num(pd->data, pd->handle, SOLVABLE_BUILDTIME, pd->buildtime);
382           pd->buildtime = (time_t)0;
383         }
384       break;
385
386     case STATE_ID:
387       s->name = pool_str2id(pool, join2(&pd->jd, "patch", ":", content), 1);
388       break;
389
390       /* <title>imlib-1.9.15-6.fc8</title> */
391     case STATE_TITLE:
392       /* strip trailing newlines */
393       while (pd->xmlp.lcontent > 0 && content[pd->xmlp.lcontent - 1] == '\n')
394         content[--pd->xmlp.lcontent] = 0;
395       repodata_set_str(pd->data, pd->handle, SOLVABLE_SUMMARY, content);
396       break;
397
398     case STATE_SEVERITY:
399       repodata_set_poolstr(pd->data, pd->handle, UPDATE_SEVERITY, content);
400       break;
401
402     case STATE_RIGHTS:
403       repodata_set_poolstr(pd->data, pd->handle, UPDATE_RIGHTS, content);
404       break;
405
406       /*
407        * <description>This update ...</description>
408        */
409     case STATE_DESCRIPTION:
410       repodata_set_str(pd->data, pd->handle, SOLVABLE_DESCRIPTION, content);
411       break;
412
413       /*
414        * <message>Warning! ...</message>
415        */
416     case STATE_MESSAGE:
417       repodata_set_str(pd->data, pd->handle, UPDATE_MESSAGE, content);
418       break;
419
420     case STATE_PACKAGE:
421       repodata_add_flexarray(pd->data, pd->handle, UPDATE_COLLECTION, pd->collhandle);
422       pd->collhandle = 0;
423       break;
424
425       /* <filename>libntlm-0.4.2-1.fc8.x86_64.rpm</filename> */
426       /* <filename>libntlm-0.4.2-1.fc8.x86_64.rpm</filename> */
427     case STATE_FILENAME:
428       repodata_set_str(pd->data, pd->collhandle, UPDATE_COLLECTION_FILENAME, content);
429       break;
430
431       /* <reboot_suggested>True</reboot_suggested> */
432     case STATE_REBOOT:
433       if (content[0] == 'T' || content[0] == 't'|| content[0] == '1')
434         {
435           /* FIXME: this is per-package, the global flag should be computed at runtime */
436           repodata_set_void(pd->data, pd->handle, UPDATE_REBOOT);
437           repodata_set_void(pd->data, pd->collhandle, UPDATE_REBOOT);
438         }
439       break;
440
441       /* <restart_suggested>True</restart_suggested> */
442     case STATE_RESTART:
443       if (content[0] == 'T' || content[0] == 't'|| content[0] == '1')
444         {
445           /* FIXME: this is per-package, the global flag should be computed at runtime */
446           repodata_set_void(pd->data, pd->handle, UPDATE_RESTART);
447           repodata_set_void(pd->data, pd->collhandle, UPDATE_RESTART);
448         }
449       break;
450
451       /* <relogin_suggested>True</relogin_suggested> */
452     case STATE_RELOGIN:
453       if (content[0] == 'T' || content[0] == 't'|| content[0] == '1')
454         {
455           /* FIXME: this is per-package, the global flag should be computed at runtime */
456           repodata_set_void(pd->data, pd->handle, UPDATE_RELOGIN);
457           repodata_set_void(pd->data, pd->collhandle, UPDATE_RELOGIN);
458         }
459       break;
460     default:
461       break;
462     }
463 }
464
465 int
466 repo_add_updateinfoxml(Repo *repo, FILE *fp, int flags)
467 {
468   Pool *pool = repo->pool;
469   Repodata *data;
470   struct parsedata pd;
471
472   data = repo_add_repodata(repo, flags);
473
474   memset(&pd, 0, sizeof(pd));
475   pd.pool = pool;
476   pd.repo = repo;
477   pd.data = data;
478   solv_xmlparser_init(&pd.xmlp, stateswitches, &pd, startElement, endElement);
479   if (solv_xmlparser_parse(&pd.xmlp, fp) != SOLV_XMLPARSER_OK)
480     pd.ret = pool_error(pool, -1, "repo_updateinfoxml: %s at line %u:%u", pd.xmlp.errstr, pd.xmlp.line, pd.xmlp.column);
481   solv_xmlparser_free(&pd.xmlp);
482   join_freemem(&pd.jd);
483
484   if (!(flags & REPO_NO_INTERNALIZE))
485     repodata_internalize(data);
486   return pd.ret;
487 }
488