blob: 28810c7c91dbeb70366ae959e2cb7b8a238ea7e9 [file] [log] [blame]
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001/**
2 * uri.c: set of generic URI related routines
3 *
4 * Reference: RFC 2396
5 *
6 * See Copyright for the status of this software.
7 *
8 * Daniel.Veillard@w3.org
9 */
10
11#ifdef WIN32
12#define INCLUDE_WINSOCK
13#include "win32config.h"
14#else
15#include "config.h"
16#endif
17
18#include <stdio.h>
19#include <string.h>
20
Daniel Veillard361d8452000-04-03 19:48:13 +000021#include <libxml/xmlmemory.h>
22#include <libxml/uri.h>
Daniel Veillard3dd82e72000-03-20 11:48:04 +000023
Daniel Veillard06047432000-04-24 11:33:38 +000024/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000025 * alpha = lowalpha | upalpha
26 */
27#define IS_ALPHA(x) (IS_LOWALPHA(x) || IS_UPALPHA(x))
28
29
Daniel Veillard06047432000-04-24 11:33:38 +000030/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000031 * lowalpha = "a" | "b" | "c" | "d" | "e" | "f" | "g" | "h" | "i" | "j" |
32 * "k" | "l" | "m" | "n" | "o" | "p" | "q" | "r" | "s" | "t" |
33 * "u" | "v" | "w" | "x" | "y" | "z"
34 */
35
36#define IS_LOWALPHA(x) (((x) >= 'a') && ((x) <= 'z'))
37
Daniel Veillard06047432000-04-24 11:33:38 +000038/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000039 * upalpha = "A" | "B" | "C" | "D" | "E" | "F" | "G" | "H" | "I" | "J" |
40 * "K" | "L" | "M" | "N" | "O" | "P" | "Q" | "R" | "S" | "T" |
41 * "U" | "V" | "W" | "X" | "Y" | "Z"
42 */
43#define IS_UPALPHA(x) (((x) >= 'A') && ((x) <= 'Z'))
44
Daniel Veillard06047432000-04-24 11:33:38 +000045/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000046 * digit = "0" | "1" | "2" | "3" | "4" | "5" | "6" | "7" | "8" | "9"
47 */
48
49#define IS_DIGIT(x) (((x) >= '0') && ((x) <= '9'))
50
Daniel Veillard06047432000-04-24 11:33:38 +000051/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000052 * alphanum = alpha | digit
53 */
54
55#define IS_ALPHANUM(x) (IS_ALPHA(x) || IS_DIGIT(x))
56
Daniel Veillard06047432000-04-24 11:33:38 +000057/*
58 * hex = digit | "A" | "B" | "C" | "D" | "E" | "F" |
Daniel Veillard3dd82e72000-03-20 11:48:04 +000059 * "a" | "b" | "c" | "d" | "e" | "f"
60 */
61
62#define IS_HEX(x) ((IS_DIGIT(x)) || (((x) >= 'a') && ((x) <= 'f')) || \
63 (((x) >= 'A') && ((x) <= 'F')))
64
Daniel Veillard06047432000-04-24 11:33:38 +000065/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000066 * mark = "-" | "_" | "." | "!" | "~" | "*" | "'" | "(" | ")"
67 */
68
69#define IS_MARK(x) (((x) == '-') || ((x) == '_') || ((x) == '.') || \
70 ((x) == '!') || ((x) == '~') || ((x) == '*') || ((x) == '\'') || \
71 ((x) == '(') || ((x) == ')'))
72
73
Daniel Veillard06047432000-04-24 11:33:38 +000074/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000075 * reserved = ";" | "/" | "?" | ":" | "@" | "&" | "=" | "+" | "$" | ","
76 */
77
78#define IS_RESERVED(x) (((x) == ';') || ((x) == '/') || ((x) == '?') || \
79 ((x) == ':') || ((x) == '@') || ((x) == '&') || ((x) == '=') || \
80 ((x) == '+') || ((x) == '$') || ((x) == ','))
81
Daniel Veillard06047432000-04-24 11:33:38 +000082/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000083 * unreserved = alphanum | mark
84 */
85
86#define IS_UNRESERVED(x) (IS_ALPHANUM(x) || IS_MARK(x))
87
Daniel Veillard06047432000-04-24 11:33:38 +000088/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000089 * escaped = "%" hex hex
90 */
91
92#define IS_ESCAPED(p) ((*(p) == '%') && (IS_HEX((p)[1])) && \
93 (IS_HEX((p)[2])))
94
Daniel Veillard06047432000-04-24 11:33:38 +000095/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +000096 * uric_no_slash = unreserved | escaped | ";" | "?" | ":" | "@" |
97 * "&" | "=" | "+" | "$" | ","
98 */
99#define IS_URIC_NO_SLASH(p) ((IS_UNRESERVED(*(p))) || (IS_ESCAPED(p)) ||\
100 ((*(p) == ';')) || ((*(p) == '?')) || ((*(p) == ':')) ||\
101 ((*(p) == '@')) || ((*(p) == '&')) || ((*(p) == '=')) ||\
102 ((*(p) == '+')) || ((*(p) == '$')) || ((*(p) == ',')))
103
Daniel Veillard06047432000-04-24 11:33:38 +0000104/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000105 * pchar = unreserved | escaped | ":" | "@" | "&" | "=" | "+" | "$" | ","
106 */
107#define IS_PCHAR(p) ((IS_UNRESERVED(*(p))) || (IS_ESCAPED(p)) || \
108 ((*(p) == ':')) || ((*(p) == '@')) || ((*(p) == '&')) ||\
109 ((*(p) == '=')) || ((*(p) == '+')) || ((*(p) == '$')) ||\
110 ((*(p) == ',')))
111
Daniel Veillard06047432000-04-24 11:33:38 +0000112/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000113 * rel_segment = 1*( unreserved | escaped |
114 * ";" | "@" | "&" | "=" | "+" | "$" | "," )
115 */
116
117#define IS_SEGMENT(p) ((IS_UNRESERVED(*(p))) || (IS_ESCAPED(p)) || \
118 ((*(p) == ';')) || ((*(p) == '@')) || ((*(p) == '&')) || \
119 ((*(p) == '=')) || ((*(p) == '+')) || ((*(p) == '$')) || \
120 ((*(p) == ',')))
121
Daniel Veillard06047432000-04-24 11:33:38 +0000122/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000123 * scheme = alpha *( alpha | digit | "+" | "-" | "." )
124 */
125
126#define IS_SCHEME(x) ((IS_ALPHA(x)) || (IS_DIGIT(x)) || \
127 ((x) == '+') || ((x) == '-') || ((x) == '.'))
128
Daniel Veillard06047432000-04-24 11:33:38 +0000129/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000130 * reg_name = 1*( unreserved | escaped | "$" | "," |
131 * ";" | ":" | "@" | "&" | "=" | "+" )
132 */
133
134#define IS_REG_NAME(p) ((IS_UNRESERVED(*(p))) || (IS_ESCAPED(p)) || \
135 ((*(p) == '$')) || ((*(p) == ',')) || ((*(p) == ';')) || \
136 ((*(p) == ':')) || ((*(p) == '@')) || ((*(p) == '&')) || \
137 ((*(p) == '=')) || ((*(p) == '+')))
138
Daniel Veillard06047432000-04-24 11:33:38 +0000139/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000140 * userinfo = *( unreserved | escaped | ";" | ":" | "&" | "=" |
141 * "+" | "$" | "," )
142 */
143#define IS_USERINFO(p) ((IS_UNRESERVED(*(p))) || (IS_ESCAPED(p)) || \
144 ((*(p) == ';')) || ((*(p) == ':')) || ((*(p) == '&')) || \
145 ((*(p) == '=')) || ((*(p) == '+')) || ((*(p) == '$')) || \
146 ((*(p) == ',')))
147
Daniel Veillard06047432000-04-24 11:33:38 +0000148/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000149 * uric = reserved | unreserved | escaped
150 */
151
152#define IS_URIC(p) ((IS_UNRESERVED(*(p))) || (IS_ESCAPED(p)) || \
153 (IS_RESERVED(*(p))))
154
Daniel Veillard06047432000-04-24 11:33:38 +0000155/*
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000156 * Skip to next pointer char, handle escaped sequences
157 */
158
159#define NEXT(p) ((*p == '%')? p += 3 : p++)
160
Daniel Veillard06047432000-04-24 11:33:38 +0000161/*
Daniel Veillard361d8452000-04-03 19:48:13 +0000162 * Productions from the spec.
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000163 *
Daniel Veillard361d8452000-04-03 19:48:13 +0000164 * authority = server | reg_name
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000165 * reg_name = 1*( unreserved | escaped | "$" | "," |
166 * ";" | ":" | "@" | "&" | "=" | "+" )
Daniel Veillard361d8452000-04-03 19:48:13 +0000167 *
168 * path = [ abs_path | opaque_part ]
169 */
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000170
171/**
172 * xmlCreateURI:
173 *
174 * Simply creates an empty xmlURI
175 *
176 * Returns the new structure or NULL in case of error
177 */
178xmlURIPtr
179xmlCreateURI(void) {
180 xmlURIPtr ret;
181
182 ret = (xmlURIPtr) xmlMalloc(sizeof(xmlURI));
183 if (ret == NULL) {
184 fprintf(stderr, "xmlCreateURI: out of memory\n");
185 return(NULL);
186 }
187 memset(ret, 0, sizeof(xmlURI));
188 return(ret);
189}
190
191/**
Daniel Veillardec303412000-03-24 13:41:54 +0000192 * xmlSaveUri:
193 * @uri: pointer to an xmlURI
194 *
195 * Save the URI as an escaped string
196 *
197 * Returns a new string (to be deallocated by caller)
198 */
199xmlChar *
200xmlSaveUri(xmlURIPtr uri) {
201 xmlChar *ret = NULL;
202 const char *p;
203 int len;
204 int max;
205
206 if (uri == NULL) return(NULL);
207
208
209 max = 80;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000210 ret = (xmlChar *) xmlMalloc((max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000211 if (ret == NULL) {
212 fprintf(stderr, "xmlSaveUri: out of memory\n");
213 return(NULL);
214 }
215 len = 0;
216
217 if (uri->scheme != NULL) {
218 p = uri->scheme;
219 while (*p != 0) {
220 if (len >= max) {
221 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000222 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000223 if (ret == NULL) {
224 fprintf(stderr, "xmlSaveUri: out of memory\n");
225 return(NULL);
226 }
227 }
228 ret[len++] = *p++;
229 }
230 if (len >= max) {
231 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000232 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000233 if (ret == NULL) {
234 fprintf(stderr, "xmlSaveUri: out of memory\n");
235 return(NULL);
236 }
237 }
238 ret[len++] = ':';
239 }
240 if (uri->opaque != NULL) {
241 p = uri->opaque;
242 while (*p != 0) {
243 if (len + 3 >= max) {
244 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000245 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000246 if (ret == NULL) {
247 fprintf(stderr, "xmlSaveUri: out of memory\n");
248 return(NULL);
249 }
250 }
251 if ((IS_UNRESERVED(*(p))) ||
252 ((*(p) == ';')) || ((*(p) == '?')) || ((*(p) == ':')) ||
253 ((*(p) == '@')) || ((*(p) == '&')) || ((*(p) == '=')) ||
254 ((*(p) == '+')) || ((*(p) == '$')) || ((*(p) == ',')))
255 ret[len++] = *p++;
256 else {
257 int val = *p++;
258 ret[len++] = '%';
259 switch (val / 0x10) {
260 case 0xF: ret[len++] = 'F'; break;
261 case 0xE: ret[len++] = 'E'; break;
262 case 0xD: ret[len++] = 'D'; break;
263 case 0xC: ret[len++] = 'C'; break;
264 case 0xB: ret[len++] = 'B'; break;
265 case 0xA: ret[len++] = 'A'; break;
266 default: ret[len++] = '0' + (val / 0x10);
267 }
268 switch (val % 0x10) {
269 case 0xF: ret[len++] = 'F'; break;
270 case 0xE: ret[len++] = 'E'; break;
271 case 0xD: ret[len++] = 'D'; break;
272 case 0xC: ret[len++] = 'C'; break;
273 case 0xB: ret[len++] = 'B'; break;
274 case 0xA: ret[len++] = 'A'; break;
275 default: ret[len++] = '0' + (val % 0x10);
276 }
277 }
278 }
279 if (len >= max) {
280 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000281 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000282 if (ret == NULL) {
283 fprintf(stderr, "xmlSaveUri: out of memory\n");
284 return(NULL);
285 }
286 }
287 ret[len++] = 0;
288 } else {
Daniel Veillard361d8452000-04-03 19:48:13 +0000289 if (uri->server != NULL) {
290 if (len + 3 >= max) {
291 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000292 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillard361d8452000-04-03 19:48:13 +0000293 if (ret == NULL) {
294 fprintf(stderr, "xmlSaveUri: out of memory\n");
295 return(NULL);
296 }
297 }
298 ret[len++] = '/';
299 ret[len++] = '/';
300 if (uri->user != NULL) {
301 p = uri->user;
302 while (*p != 0) {
303 if (len + 3 >= max) {
304 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000305 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillard361d8452000-04-03 19:48:13 +0000306 if (ret == NULL) {
307 fprintf(stderr, "xmlSaveUri: out of memory\n");
308 return(NULL);
309 }
310 }
311 if ((IS_UNRESERVED(*(p))) ||
312 ((*(p) == ';')) || ((*(p) == ':')) || ((*(p) == '&')) ||
313 ((*(p) == '=')) || ((*(p) == '+')) || ((*(p) == '$')) ||
314 ((*(p) == ',')))
315 ret[len++] = *p++;
316 else {
317 int val = *p++;
318 ret[len++] = '%';
319 switch (val / 0x10) {
320 case 0xF: ret[len++] = 'F'; break;
321 case 0xE: ret[len++] = 'E'; break;
322 case 0xD: ret[len++] = 'D'; break;
323 case 0xC: ret[len++] = 'C'; break;
324 case 0xB: ret[len++] = 'B'; break;
325 case 0xA: ret[len++] = 'A'; break;
326 default: ret[len++] = '0' + (val / 0x10);
327 }
328 switch (val % 0x10) {
329 case 0xF: ret[len++] = 'F'; break;
330 case 0xE: ret[len++] = 'E'; break;
331 case 0xD: ret[len++] = 'D'; break;
332 case 0xC: ret[len++] = 'C'; break;
333 case 0xB: ret[len++] = 'B'; break;
334 case 0xA: ret[len++] = 'A'; break;
335 default: ret[len++] = '0' + (val % 0x10);
336 }
337 }
338 }
339 if (len + 3 >= max) {
340 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000341 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillard361d8452000-04-03 19:48:13 +0000342 if (ret == NULL) {
343 fprintf(stderr, "xmlSaveUri: out of memory\n");
344 return(NULL);
345 }
346 }
347 ret[len++] = '@';
348 }
349 p = uri->server;
350 while (*p != 0) {
351 if (len >= max) {
352 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000353 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillard361d8452000-04-03 19:48:13 +0000354 if (ret == NULL) {
355 fprintf(stderr, "xmlSaveUri: out of memory\n");
356 return(NULL);
357 }
358 }
359 ret[len++] = *p++;
360 }
361 if (uri->port > 0) {
362 if (len + 10 >= max) {
363 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000364 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillard361d8452000-04-03 19:48:13 +0000365 if (ret == NULL) {
366 fprintf(stderr, "xmlSaveUri: out of memory\n");
367 return(NULL);
368 }
369 }
370 len += sprintf((char *) &ret[len], ":%d", uri->port);
371 }
372 } else if (uri->authority != NULL) {
Daniel Veillardec303412000-03-24 13:41:54 +0000373 if (len + 3 >= max) {
374 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000375 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000376 if (ret == NULL) {
377 fprintf(stderr, "xmlSaveUri: out of memory\n");
378 return(NULL);
379 }
380 }
381 ret[len++] = '/';
382 ret[len++] = '/';
383 p = uri->authority;
384 while (*p != 0) {
385 if (len + 3 >= max) {
386 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000387 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000388 if (ret == NULL) {
389 fprintf(stderr, "xmlSaveUri: out of memory\n");
390 return(NULL);
391 }
392 }
393 if ((IS_UNRESERVED(*(p))) ||
394 ((*(p) == '$')) || ((*(p) == ',')) || ((*(p) == ';')) ||
395 ((*(p) == ':')) || ((*(p) == '@')) || ((*(p) == '&')) ||
396 ((*(p) == '=')) || ((*(p) == '+')))
397 ret[len++] = *p++;
398 else {
399 int val = *p++;
400 ret[len++] = '%';
401 switch (val / 0x10) {
402 case 0xF: ret[len++] = 'F'; break;
403 case 0xE: ret[len++] = 'E'; break;
404 case 0xD: ret[len++] = 'D'; break;
405 case 0xC: ret[len++] = 'C'; break;
406 case 0xB: ret[len++] = 'B'; break;
407 case 0xA: ret[len++] = 'A'; break;
408 default: ret[len++] = '0' + (val / 0x10);
409 }
410 switch (val % 0x10) {
411 case 0xF: ret[len++] = 'F'; break;
412 case 0xE: ret[len++] = 'E'; break;
413 case 0xD: ret[len++] = 'D'; break;
414 case 0xC: ret[len++] = 'C'; break;
415 case 0xB: ret[len++] = 'B'; break;
416 case 0xA: ret[len++] = 'A'; break;
417 default: ret[len++] = '0' + (val % 0x10);
418 }
419 }
420 }
421 }
422 if (uri->path != NULL) {
423 p = uri->path;
424 while (*p != 0) {
425 if (len + 3 >= max) {
426 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000427 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000428 if (ret == NULL) {
429 fprintf(stderr, "xmlSaveUri: out of memory\n");
430 return(NULL);
431 }
432 }
433 if ((IS_UNRESERVED(*(p))) || ((*(p) == '/')) ||
434 ((*(p) == ';')) || ((*(p) == '@')) || ((*(p) == '&')) ||
435 ((*(p) == '=')) || ((*(p) == '+')) || ((*(p) == '$')) ||
436 ((*(p) == ',')))
437 ret[len++] = *p++;
438 else {
439 int val = *p++;
440 ret[len++] = '%';
441 switch (val / 0x10) {
442 case 0xF: ret[len++] = 'F'; break;
443 case 0xE: ret[len++] = 'E'; break;
444 case 0xD: ret[len++] = 'D'; break;
445 case 0xC: ret[len++] = 'C'; break;
446 case 0xB: ret[len++] = 'B'; break;
447 case 0xA: ret[len++] = 'A'; break;
448 default: ret[len++] = '0' + (val / 0x10);
449 }
450 switch (val % 0x10) {
451 case 0xF: ret[len++] = 'F'; break;
452 case 0xE: ret[len++] = 'E'; break;
453 case 0xD: ret[len++] = 'D'; break;
454 case 0xC: ret[len++] = 'C'; break;
455 case 0xB: ret[len++] = 'B'; break;
456 case 0xA: ret[len++] = 'A'; break;
457 default: ret[len++] = '0' + (val % 0x10);
458 }
459 }
460 }
461 }
462 if (uri->query != NULL) {
463 if (len + 3 >= max) {
464 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000465 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000466 if (ret == NULL) {
467 fprintf(stderr, "xmlSaveUri: out of memory\n");
468 return(NULL);
469 }
470 }
471 ret[len++] = '?';
472 p = uri->query;
473 while (*p != 0) {
474 if (len + 3 >= max) {
475 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000476 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000477 if (ret == NULL) {
478 fprintf(stderr, "xmlSaveUri: out of memory\n");
479 return(NULL);
480 }
481 }
482 if ((IS_UNRESERVED(*(p))) || (IS_RESERVED(*(p))))
483 ret[len++] = *p++;
484 else {
485 int val = *p++;
486 ret[len++] = '%';
487 switch (val / 0x10) {
488 case 0xF: ret[len++] = 'F'; break;
489 case 0xE: ret[len++] = 'E'; break;
490 case 0xD: ret[len++] = 'D'; break;
491 case 0xC: ret[len++] = 'C'; break;
492 case 0xB: ret[len++] = 'B'; break;
493 case 0xA: ret[len++] = 'A'; break;
494 default: ret[len++] = '0' + (val / 0x10);
495 }
496 switch (val % 0x10) {
497 case 0xF: ret[len++] = 'F'; break;
498 case 0xE: ret[len++] = 'E'; break;
499 case 0xD: ret[len++] = 'D'; break;
500 case 0xC: ret[len++] = 'C'; break;
501 case 0xB: ret[len++] = 'B'; break;
502 case 0xA: ret[len++] = 'A'; break;
503 default: ret[len++] = '0' + (val % 0x10);
504 }
505 }
506 }
507 }
508 if (uri->fragment != NULL) {
509 if (len + 3 >= max) {
510 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000511 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000512 if (ret == NULL) {
513 fprintf(stderr, "xmlSaveUri: out of memory\n");
514 return(NULL);
515 }
516 }
517 ret[len++] = '#';
518 p = uri->fragment;
519 while (*p != 0) {
520 if (len + 3 >= max) {
521 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000522 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000523 if (ret == NULL) {
524 fprintf(stderr, "xmlSaveUri: out of memory\n");
525 return(NULL);
526 }
527 }
528 if ((IS_UNRESERVED(*(p))) || (IS_RESERVED(*(p))))
529 ret[len++] = *p++;
530 else {
531 int val = *p++;
532 ret[len++] = '%';
533 switch (val / 0x10) {
534 case 0xF: ret[len++] = 'F'; break;
535 case 0xE: ret[len++] = 'E'; break;
536 case 0xD: ret[len++] = 'D'; break;
537 case 0xC: ret[len++] = 'C'; break;
538 case 0xB: ret[len++] = 'B'; break;
539 case 0xA: ret[len++] = 'A'; break;
540 default: ret[len++] = '0' + (val / 0x10);
541 }
542 switch (val % 0x10) {
543 case 0xF: ret[len++] = 'F'; break;
544 case 0xE: ret[len++] = 'E'; break;
545 case 0xD: ret[len++] = 'D'; break;
546 case 0xC: ret[len++] = 'C'; break;
547 case 0xB: ret[len++] = 'B'; break;
548 case 0xA: ret[len++] = 'A'; break;
549 default: ret[len++] = '0' + (val % 0x10);
550 }
551 }
552 }
553 }
554 if (len >= max) {
555 max *= 2;
Daniel Veillard32bc74e2000-07-14 14:49:25 +0000556 ret = (xmlChar *) xmlRealloc(ret, (max + 1) * sizeof(xmlChar));
Daniel Veillardec303412000-03-24 13:41:54 +0000557 if (ret == NULL) {
558 fprintf(stderr, "xmlSaveUri: out of memory\n");
559 return(NULL);
560 }
561 }
562 ret[len++] = 0;
563 }
564 return(ret);
565}
566
567/**
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000568 * xmlPrintURI:
569 * @stream: a FILE* for the output
570 * @uri: pointer to an xmlURI
571 *
572 * Prints the URI in the stream @steam.
573 */
574void
575xmlPrintURI(FILE *stream, xmlURIPtr uri) {
Daniel Veillardec303412000-03-24 13:41:54 +0000576 xmlChar *out;
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000577
Daniel Veillardec303412000-03-24 13:41:54 +0000578 out = xmlSaveUri(uri);
579 if (out != NULL) {
580 fprintf(stream, "%s", out);
581 xmlFree(out);
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000582 }
583}
584
585/**
586 * xmlCleanURI:
587 * @uri: pointer to an xmlURI
588 *
589 * Make sure the xmlURI struct is free of content
590 */
591void
592xmlCleanURI(xmlURIPtr uri) {
593 if (uri == NULL) return;
594
595 if (uri->scheme != NULL) xmlFree(uri->scheme);
596 uri->scheme = NULL;
597 if (uri->server != NULL) xmlFree(uri->server);
598 uri->server = NULL;
Daniel Veillard361d8452000-04-03 19:48:13 +0000599 if (uri->user != NULL) xmlFree(uri->user);
600 uri->user = NULL;
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000601 if (uri->path != NULL) xmlFree(uri->path);
602 uri->path = NULL;
603 if (uri->fragment != NULL) xmlFree(uri->fragment);
604 uri->fragment = NULL;
605 if (uri->opaque != NULL) xmlFree(uri->opaque);
606 uri->opaque = NULL;
607 if (uri->authority != NULL) xmlFree(uri->authority);
608 uri->authority = NULL;
609 if (uri->query != NULL) xmlFree(uri->query);
610 uri->query = NULL;
611}
612
613/**
614 * xmlFreeURI:
615 * @uri: pointer to an xmlURI
616 *
617 * Free up the xmlURI struct
618 */
619void
620xmlFreeURI(xmlURIPtr uri) {
621 if (uri == NULL) return;
622
623 if (uri->scheme != NULL) xmlFree(uri->scheme);
624 if (uri->server != NULL) xmlFree(uri->server);
Daniel Veillard361d8452000-04-03 19:48:13 +0000625 if (uri->user != NULL) xmlFree(uri->user);
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000626 if (uri->path != NULL) xmlFree(uri->path);
627 if (uri->fragment != NULL) xmlFree(uri->fragment);
628 if (uri->opaque != NULL) xmlFree(uri->opaque);
629 if (uri->authority != NULL) xmlFree(uri->authority);
630 if (uri->query != NULL) xmlFree(uri->query);
631 memset(uri, -1, sizeof(xmlURI));
632 xmlFree(uri);
633}
634
635/**
Daniel Veillard361d8452000-04-03 19:48:13 +0000636 * xmlURIUnescapeString:
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000637 * @str: the string to unescape
638 * @len: the lenght in bytes to unescape (or <= 0 to indicate full string)
639 * @target: optionnal destination buffer
640 *
641 * Unescaping routine, does not do validity checks !
Daniel Veillardec303412000-03-24 13:41:54 +0000642 * Output is direct unsigned char translation of %XX values (no encoding)
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000643 *
644 * Returns an copy of the string, but unescaped
645 */
646char *
Daniel Veillard361d8452000-04-03 19:48:13 +0000647xmlURIUnescapeString(const char *str, int len, char *target) {
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000648 char *ret, *out;
649 const char *in;
650
651 if (str == NULL)
652 return(NULL);
653 if (len <= 0) len = strlen(str);
654 if (len <= 0) return(NULL);
655
656 if (target == NULL) {
657 ret = (char *) xmlMalloc(len + 1);
658 if (ret == NULL) {
Daniel Veillard361d8452000-04-03 19:48:13 +0000659 fprintf(stderr, "xmlURIUnescapeString: out of memory\n");
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000660 return(NULL);
661 }
662 } else
663 ret = target;
664 in = str;
665 out = ret;
666 while(len > 0) {
667 if (*in == '%') {
668 in++;
669 if ((*in >= '0') && (*in <= '9'))
670 *out = (*in - '0');
671 else if ((*in >= 'a') && (*in <= 'f'))
672 *out = (*in - 'a') + 10;
673 else if ((*in >= 'A') && (*in <= 'F'))
674 *out = (*in - 'A') + 10;
675 in++;
676 if ((*in >= '0') && (*in <= '9'))
677 *out = *out * 16 + (*in - '0');
678 else if ((*in >= 'a') && (*in <= 'f'))
679 *out = *out * 16 + (*in - 'a') + 10;
680 else if ((*in >= 'A') && (*in <= 'F'))
681 *out = *out * 16 + (*in - 'A') + 10;
682 in++;
683 len -= 3;
Daniel Veillardec303412000-03-24 13:41:54 +0000684 out++;
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000685 } else {
686 *out++ = *in++;
687 len--;
688 }
689 }
690 *out = 0;
691 return(ret);
692}
693
694
695/**
696 * xmlParseURIFragment:
697 * @uri: pointer to an URI structure
698 * @str: pointer to the string to analyze
699 *
700 * Parse an URI fragment string and fills in the appropriate fields
701 * of the @uri structure.
702 *
703 * fragment = *uric
704 *
705 * Returns 0 or the error code
706 */
707int
708xmlParseURIFragment(xmlURIPtr uri, const char **str) {
709 const char *cur = *str;
710
711 if (str == NULL) return(-1);
712
713 while (IS_URIC(cur)) NEXT(cur);
714 if (uri != NULL) {
715 if (uri->fragment != NULL) xmlFree(uri->fragment);
Daniel Veillard361d8452000-04-03 19:48:13 +0000716 uri->fragment = xmlURIUnescapeString(*str, cur - *str, NULL);
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000717 }
718 *str = cur;
719 return(0);
720}
721
722/**
723 * xmlParseURIQuery:
724 * @uri: pointer to an URI structure
725 * @str: pointer to the string to analyze
726 *
727 * Parse the query part of an URI
728 *
729 * query = *uric
730 *
731 * Returns 0 or the error code
732 */
733int
734xmlParseURIQuery(xmlURIPtr uri, const char **str) {
735 const char *cur = *str;
736
737 if (str == NULL) return(-1);
738
739 while (IS_URIC(cur)) NEXT(cur);
740 if (uri != NULL) {
741 if (uri->query != NULL) xmlFree(uri->query);
Daniel Veillard361d8452000-04-03 19:48:13 +0000742 uri->query = xmlURIUnescapeString(*str, cur - *str, NULL);
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000743 }
744 *str = cur;
745 return(0);
746}
747
748/**
749 * xmlParseURIScheme:
750 * @uri: pointer to an URI structure
751 * @str: pointer to the string to analyze
752 *
753 * Parse an URI scheme
754 *
755 * scheme = alpha *( alpha | digit | "+" | "-" | "." )
756 *
757 * Returns 0 or the error code
758 */
759int
760xmlParseURIScheme(xmlURIPtr uri, const char **str) {
761 const char *cur;
762
763 if (str == NULL)
764 return(-1);
765
766 cur = *str;
767 if (!IS_ALPHA(*cur))
768 return(2);
769 cur++;
770 while (IS_SCHEME(*cur)) cur++;
771 if (uri != NULL) {
772 if (uri->scheme != NULL) xmlFree(uri->scheme);
Daniel Veillard361d8452000-04-03 19:48:13 +0000773 uri->scheme = xmlURIUnescapeString(*str, cur - *str, NULL); /* !!! strndup */
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000774 }
775 *str = cur;
776 return(0);
777}
778
779/**
780 * xmlParseURIOpaquePart:
781 * @uri: pointer to an URI structure
782 * @str: pointer to the string to analyze
783 *
784 * Parse an URI opaque part
785 *
786 * opaque_part = uric_no_slash *uric
787 *
788 * Returns 0 or the error code
789 */
790int
791xmlParseURIOpaquePart(xmlURIPtr uri, const char **str) {
792 const char *cur;
793
794 if (str == NULL)
795 return(-1);
796
797 cur = *str;
798 if (!IS_URIC_NO_SLASH(cur)) {
799 return(3);
800 }
801 NEXT(cur);
802 while (IS_URIC(cur)) NEXT(cur);
803 if (uri != NULL) {
804 if (uri->opaque != NULL) xmlFree(uri->opaque);
Daniel Veillard361d8452000-04-03 19:48:13 +0000805 uri->opaque = xmlURIUnescapeString(*str, cur - *str, NULL);
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000806 }
807 *str = cur;
808 return(0);
809}
810
811/**
Daniel Veillard361d8452000-04-03 19:48:13 +0000812 * xmlParseURIServer:
813 * @uri: pointer to an URI structure
814 * @str: pointer to the string to analyze
815 *
816 * Parse a server subpart of an URI, it's a finer grain analysis
817 * of the authority part.
818 *
819 * server = [ [ userinfo "@" ] hostport ]
820 * userinfo = *( unreserved | escaped |
821 * ";" | ":" | "&" | "=" | "+" | "$" | "," )
822 * hostport = host [ ":" port ]
823 * host = hostname | IPv4address
824 * hostname = *( domainlabel "." ) toplabel [ "." ]
825 * domainlabel = alphanum | alphanum *( alphanum | "-" ) alphanum
826 * toplabel = alpha | alpha *( alphanum | "-" ) alphanum
827 * IPv4address = 1*digit "." 1*digit "." 1*digit "." 1*digit
828 * port = *digit
829 *
830 * Returns 0 or the error code
831 */
832int
833xmlParseURIServer(xmlURIPtr uri, const char **str) {
834 const char *cur;
835 const char *host, *tmp;
836
837 if (str == NULL)
838 return(-1);
839
840 cur = *str;
841
842 /*
843 * is there an userinfo ?
844 */
845 while (IS_USERINFO(cur)) NEXT(cur);
846 if (*cur == '@') {
847 if (uri != NULL) {
848 if (uri->user != NULL) xmlFree(uri->user);
849 uri->user = xmlURIUnescapeString(*str, cur - *str, NULL);
850 }
851 cur++;
852 } else {
853 if (uri != NULL) {
854 if (uri->user != NULL) xmlFree(uri->user);
855 uri->user = NULL;
856 }
857 cur = *str;
858 }
859 /*
860 * host part of hostport can derive either an IPV4 address
861 * or an unresolved name. Check the IP first, it easier to detect
862 * errors if wrong one
863 */
864 host = cur;
865 if (IS_DIGIT(*cur)) {
866 while(IS_DIGIT(*cur)) cur++;
867 if (*cur != '.')
868 goto host_name;
869 cur++;
870 if (!IS_DIGIT(*cur))
871 goto host_name;
872 while(IS_DIGIT(*cur)) cur++;
873 if (*cur != '.')
874 goto host_name;
875 cur++;
876 if (!IS_DIGIT(*cur))
877 goto host_name;
878 while(IS_DIGIT(*cur)) cur++;
879 if (*cur != '.')
880 goto host_name;
881 cur++;
882 if (!IS_DIGIT(*cur))
883 goto host_name;
884 while(IS_DIGIT(*cur)) cur++;
885 if (uri != NULL) {
886 if (uri->authority != NULL) xmlFree(uri->authority);
887 uri->authority = NULL;
888 if (uri->server != NULL) xmlFree(uri->server);
889 uri->server = xmlURIUnescapeString(host, cur - host, NULL);
890 }
891 goto host_done;
892 }
893host_name:
894 /*
895 * the hostname production as-is is a parser nightmare.
896 * simplify it to
897 * hostname = *( domainlabel "." ) domainlabel [ "." ]
898 * and just make sure the last label starts with a non numeric char.
899 */
900 if (!IS_ALPHANUM(*cur))
901 return(6);
902 while (IS_ALPHANUM(*cur)) {
903 while ((IS_ALPHANUM(*cur)) || (*cur == '-')) cur++;
904 if (*cur == '.')
905 cur++;
906 }
907 tmp = cur;
908 tmp--;
909 while (IS_ALPHANUM(*tmp) && (*tmp != '.') && (tmp >= host)) tmp--;
910 tmp++;
911 if (!IS_ALPHA(*tmp))
912 return(7);
913 if (uri != NULL) {
914 if (uri->authority != NULL) xmlFree(uri->authority);
915 uri->authority = NULL;
916 if (uri->server != NULL) xmlFree(uri->server);
917 uri->server = xmlURIUnescapeString(host, cur - host, NULL);
918 }
919
920host_done:
921
922 /*
923 * finish by checking for a port presence.
924 */
925 if (*cur == ':') {
926 cur++;
927 if (IS_DIGIT(*cur)) {
928 if (uri != NULL)
929 uri->port = 0;
930 while (IS_DIGIT(*cur)) {
931 if (uri != NULL)
932 uri->port = uri->port * 10 + (*cur - '0');
933 cur++;
934 }
935 }
936 }
937 *str = cur;
938 return(0);
939}
940
941/**
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000942 * xmlParseURIRelSegment:
943 * @uri: pointer to an URI structure
944 * @str: pointer to the string to analyze
945 *
946 * Parse an URI relative segment
947 *
948 * rel_segment = 1*( unreserved | escaped | ";" | "@" | "&" | "=" |
949 * "+" | "$" | "," )
950 *
951 * Returns 0 or the error code
952 */
953int
954xmlParseURIRelSegment(xmlURIPtr uri, const char **str) {
955 const char *cur;
956
957 if (str == NULL)
958 return(-1);
959
960 cur = *str;
961 if (!IS_SEGMENT(cur)) {
962 return(3);
963 }
964 NEXT(cur);
965 while (IS_SEGMENT(cur)) NEXT(cur);
966 if (uri != NULL) {
967 if (uri->path != NULL) xmlFree(uri->path);
Daniel Veillard361d8452000-04-03 19:48:13 +0000968 uri->path = xmlURIUnescapeString(*str, cur - *str, NULL);
Daniel Veillard3dd82e72000-03-20 11:48:04 +0000969 }
970 *str = cur;
971 return(0);
972}
973
974/**
975 * xmlParseURIPathSegments:
976 * @uri: pointer to an URI structure
977 * @str: pointer to the string to analyze
978 * @slash: should we add a leading slash
979 *
980 * Parse an URI set of path segments
981 *
982 * path_segments = segment *( "/" segment )
983 * segment = *pchar *( ";" param )
984 * param = *pchar
985 *
986 * Returns 0 or the error code
987 */
988int
989xmlParseURIPathSegments(xmlURIPtr uri, const char **str, int slash) {
990 const char *cur;
991
992 if (str == NULL)
993 return(-1);
994
995 cur = *str;
996
997 do {
998 while (IS_PCHAR(cur)) NEXT(cur);
999 if (*cur == ';') {
1000 cur++;
1001 while (IS_PCHAR(cur)) NEXT(cur);
1002 }
1003 if (*cur != '/') break;
1004 cur++;
1005 } while (1);
1006 if (uri != NULL) {
1007 int len, len2 = 0;
1008 char *path;
1009
1010 /*
1011 * Concat the set of path segments to the current path
1012 */
1013 len = cur - *str;
1014 if (slash)
1015 len++;
1016
1017 if (uri->path != NULL) {
1018 len2 = strlen(uri->path);
1019 len += len2;
1020 }
1021 path = (char *) xmlMalloc(len + 1);
1022 if (path == NULL) {
1023 fprintf(stderr, "xmlParseURIPathSegments: out of memory\n");
1024 *str = cur;
1025 return(-1);
1026 }
Daniel Veillard52402ce2000-08-22 23:36:12 +00001027 path[len] = '\0';
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001028 if (uri->path != NULL)
1029 memcpy(path, uri->path, len2);
1030 if (slash) {
1031 path[len2] = '/';
1032 len2++;
1033 }
Daniel Veillard361d8452000-04-03 19:48:13 +00001034 xmlURIUnescapeString(*str, cur - *str, &path[len2]);
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001035 if (uri->path != NULL)
1036 xmlFree(uri->path);
1037 uri->path = path;
1038 }
1039 *str = cur;
1040 return(0);
1041}
1042
1043/**
1044 * xmlParseURIAuthority:
1045 * @uri: pointer to an URI structure
1046 * @str: pointer to the string to analyze
1047 *
1048 * Parse the authority part of an URI.
1049 *
1050 * authority = server | reg_name
1051 * server = [ [ userinfo "@" ] hostport ]
1052 * reg_name = 1*( unreserved | escaped | "$" | "," | ";" | ":" |
1053 * "@" | "&" | "=" | "+" )
1054 *
1055 * Note : this is completely ambiguous since reg_name is allowed to
1056 * use the full set of chars in use by server:
1057 *
1058 * 3.2.1. Registry-based Naming Authority
1059 *
1060 * The structure of a registry-based naming authority is specific
1061 * to the URI scheme, but constrained to the allowed characters
1062 * for an authority component.
1063 *
1064 * Returns 0 or the error code
1065 */
1066int
1067xmlParseURIAuthority(xmlURIPtr uri, const char **str) {
1068 const char *cur;
Daniel Veillard361d8452000-04-03 19:48:13 +00001069 int ret;
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001070
1071 if (str == NULL)
1072 return(-1);
1073
1074 cur = *str;
Daniel Veillard361d8452000-04-03 19:48:13 +00001075
1076 /*
1077 * try first to parse it as a server string.
1078 */
1079 ret = xmlParseURIServer(uri, str);
1080 if (ret == 0)
1081 return(0);
1082
1083 /*
1084 * failed, fallback to reg_name
1085 */
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001086 if (!IS_REG_NAME(cur)) {
1087 return(5);
1088 }
1089 NEXT(cur);
1090 while (IS_REG_NAME(cur)) NEXT(cur);
1091 if (uri != NULL) {
Daniel Veillard361d8452000-04-03 19:48:13 +00001092 if (uri->server != NULL) xmlFree(uri->server);
1093 uri->server = NULL;
1094 if (uri->user != NULL) xmlFree(uri->user);
1095 uri->user = NULL;
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001096 if (uri->authority != NULL) xmlFree(uri->authority);
Daniel Veillard361d8452000-04-03 19:48:13 +00001097 uri->authority = xmlURIUnescapeString(*str, cur - *str, NULL);
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001098 }
1099 *str = cur;
1100 return(0);
1101}
1102
1103/**
1104 * xmlParseURIHierPart:
1105 * @uri: pointer to an URI structure
1106 * @str: pointer to the string to analyze
1107 *
1108 * Parse an URI hirarchical part
1109 *
1110 * hier_part = ( net_path | abs_path ) [ "?" query ]
1111 * abs_path = "/" path_segments
1112 * net_path = "//" authority [ abs_path ]
1113 *
1114 * Returns 0 or the error code
1115 */
1116int
1117xmlParseURIHierPart(xmlURIPtr uri, const char **str) {
1118 int ret;
1119 const char *cur;
1120
1121 if (str == NULL)
1122 return(-1);
1123
1124 cur = *str;
1125
1126 if ((cur[0] == '/') && (cur[1] == '/')) {
1127 cur += 2;
1128 ret = xmlParseURIAuthority(uri, &cur);
1129 if (ret != 0)
1130 return(ret);
1131 if (cur[0] == '/') {
1132 cur++;
1133 ret = xmlParseURIPathSegments(uri, &cur, 1);
1134 }
1135 } else if (cur[0] == '/') {
1136 cur++;
1137 ret = xmlParseURIPathSegments(uri, &cur, 1);
1138 } else {
1139 return(4);
1140 }
1141 if (ret != 0)
1142 return(ret);
1143 if (*cur == '?') {
1144 cur++;
1145 ret = xmlParseURIQuery(uri, &cur);
1146 if (ret != 0)
1147 return(ret);
1148 }
1149 *str = cur;
1150 return(0);
1151}
1152
1153/**
1154 * xmlParseAbsoluteURI:
1155 * @uri: pointer to an URI structure
1156 * @str: pointer to the string to analyze
1157 *
1158 * Parse an URI reference string and fills in the appropriate fields
1159 * of the @uri structure
1160 *
1161 * absoluteURI = scheme ":" ( hier_part | opaque_part )
1162 *
1163 * Returns 0 or the error code
1164 */
1165int
1166xmlParseAbsoluteURI(xmlURIPtr uri, const char **str) {
1167 int ret;
1168
1169 if (str == NULL)
1170 return(-1);
1171
1172 ret = xmlParseURIScheme(uri, str);
1173 if (ret != 0) return(ret);
1174 if (**str != ':')
1175 return(1);
1176 (*str)++;
1177 if (**str == '/')
1178 return(xmlParseURIHierPart(uri, str));
1179 return(xmlParseURIOpaquePart(uri, str));
1180}
1181
1182/**
1183 * xmlParseRelativeURI:
1184 * @uri: pointer to an URI structure
1185 * @str: pointer to the string to analyze
1186 *
1187 * Parse an relative URI string and fills in the appropriate fields
1188 * of the @uri structure
1189 *
1190 * relativeURI = ( net_path | abs_path | rel_path ) [ "?" query ]
1191 * abs_path = "/" path_segments
1192 * net_path = "//" authority [ abs_path ]
1193 * rel_path = rel_segment [ abs_path ]
1194 *
1195 * Returns 0 or the error code
1196 */
1197int
1198xmlParseRelativeURI(xmlURIPtr uri, const char **str) {
1199 int ret = 0;
1200 const char *cur;
1201
1202 if (str == NULL)
1203 return(-1);
1204
1205 cur = *str;
1206 if ((cur[0] == '/') && (cur[1] == '/')) {
1207 cur += 2;
1208 ret = xmlParseURIAuthority(uri, &cur);
1209 if (ret != 0)
1210 return(ret);
1211 if (cur[0] == '/') {
1212 cur++;
1213 ret = xmlParseURIPathSegments(uri, &cur, 1);
1214 }
1215 } else if (cur[0] == '/') {
1216 cur++;
1217 ret = xmlParseURIPathSegments(uri, &cur, 1);
1218 } else {
1219 ret = xmlParseURIRelSegment(uri, &cur);
1220 if (ret != 0)
1221 return(ret);
1222 if (cur[0] == '/') {
1223 cur++;
1224 ret = xmlParseURIPathSegments(uri, &cur, 1);
1225 }
1226 }
1227 if (ret != 0)
1228 return(ret);
1229 if (*cur == '?') {
1230 cur++;
1231 ret = xmlParseURIQuery(uri, &cur);
1232 if (ret != 0)
1233 return(ret);
1234 }
1235 *str = cur;
1236 return(ret);
1237}
1238
1239/**
1240 * xmlParseURIReference:
1241 * @uri: pointer to an URI structure
1242 * @str: the string to analyze
1243 *
1244 * Parse an URI reference string and fills in the appropriate fields
1245 * of the @uri structure
1246 *
1247 * URI-reference = [ absoluteURI | relativeURI ] [ "#" fragment ]
1248 *
1249 * Returns 0 or the error code
1250 */
1251int
1252xmlParseURIReference(xmlURIPtr uri, const char *str) {
1253 int ret;
1254 const char *tmp = str;
1255
1256 if (str == NULL)
1257 return(-1);
1258 xmlCleanURI(uri);
1259
1260 /*
1261 * Try first to parse aboslute refs, then fallback to relative if
1262 * it fails.
1263 */
1264 ret = xmlParseAbsoluteURI(uri, &str);
1265 if (ret != 0) {
1266 xmlCleanURI(uri);
1267 str = tmp;
1268 ret = xmlParseRelativeURI(uri, &str);
1269 }
1270 if (ret != 0) {
1271 xmlCleanURI(uri);
1272 return(ret);
1273 }
1274
1275 if (*str == '#') {
1276 str++;
1277 ret = xmlParseURIFragment(uri, &str);
1278 if (ret != 0) return(ret);
1279 }
1280 if (*str != 0) {
1281 xmlCleanURI(uri);
1282 return(1);
1283 }
1284 return(0);
1285}
1286
1287/**
Daniel Veillard496a1cf2000-05-03 14:20:55 +00001288 * xmlParseURI:
1289 * @str: the URI string to analyze
1290 *
1291 * Parse an URI
1292 *
1293 * URI-reference = [ absoluteURI | relativeURI ] [ "#" fragment ]
1294 *
1295 * Returns a newly build xmlURIPtr or NULL in case of error
1296 */
1297xmlURIPtr
1298xmlParseURI(const char *str) {
1299 xmlURIPtr uri;
1300 int ret;
1301
1302 if (str == NULL)
1303 return(NULL);
1304 uri = xmlCreateURI();
1305 if (uri != NULL) {
1306 ret = xmlParseURIReference(uri, str);
1307 if (ret) {
1308 xmlFreeURI(uri);
1309 return(NULL);
1310 }
1311 }
1312 return(uri);
1313}
1314
1315/**
Daniel Veillardec303412000-03-24 13:41:54 +00001316 * xmlNormalizeURIPath:
1317 * @path: pointer to the path string
1318 *
1319 * applies the 5 normalization steps to a path string
1320 * Normalization occurs directly on the string, no new allocation is done
1321 *
1322 * Returns 0 or an error code
1323 */
1324int
1325xmlNormalizeURIPath(char *path) {
1326 int cur, out;
1327
1328 if (path == NULL)
1329 return(-1);
1330 cur = 0;
1331 out = 0;
1332 while ((path[cur] != 0) && (path[cur] != '/')) cur++;
1333 if (path[cur] == 0)
1334 return(0);
1335
1336 /* we are positionned at the beginning of the first segment */
1337 cur++;
1338 out = cur;
1339
1340 /*
1341 * Analyze each segment in sequence.
1342 */
1343 while (path[cur] != 0) {
1344 /*
1345 * c) All occurrences of "./", where "." is a complete path segment,
1346 * are removed from the buffer string.
1347 */
1348 if ((path[cur] == '.') && (path[cur + 1] == '/')) {
1349 cur += 2;
1350 continue;
1351 }
1352
1353 /*
1354 * d) If the buffer string ends with "." as a complete path segment,
1355 * that "." is removed.
1356 */
1357 if ((path[cur] == '.') && (path[cur + 1] == 0)) {
1358 path[out] = 0;
1359 break;
1360 }
1361
1362 /* read the segment */
1363 while ((path[cur] != 0) && (path[cur] != '/')) {
1364 path[out++] = path[cur++];
1365 }
1366 path[out++] = path[cur];
1367 if (path[cur] != 0) {
1368 cur++;
1369 }
1370 }
1371
1372 cur = 0;
1373 out = 0;
1374 while ((path[cur] != 0) && (path[cur] != '/')) cur++;
1375 if (path[cur] == 0)
1376 return(0);
1377 /* we are positionned at the beginning of the first segment */
1378 cur++;
1379 out = cur;
1380 /*
1381 * Analyze each segment in sequence.
1382 */
1383 while (path[cur] != 0) {
1384 /*
1385 * e) All occurrences of "<segment>/../", where <segment> is a
1386 * complete path segment not equal to "..", are removed from the
1387 * buffer string. Removal of these path segments is performed
1388 * iteratively, removing the leftmost matching pattern on each
1389 * iteration, until no matching pattern remains.
1390 */
1391 if ((cur > 1) && (out > 1) &&
1392 (path[cur] == '/') && (path[cur + 1] == '.') &&
1393 (path[cur + 2] == '.') && (path[cur + 3] == '/') &&
1394 ((path[out] != '.') || (path[out - 1] != '.') ||
1395 (path[out - 2] != '/'))) {
1396 cur += 3;
1397 out --;
1398 while ((out > 0) && (path[out] != '/')) { out --; }
1399 path[out] = 0;
1400 continue;
1401 }
1402
1403 /*
1404 * f) If the buffer string ends with "<segment>/..", where <segment>
1405 * is a complete path segment not equal to "..", that
1406 * "<segment>/.." is removed.
1407 */
1408 if ((path[cur] == '/') && (path[cur + 1] == '.') &&
1409 (path[cur + 2] == '.') && (path[cur + 3] == 0) &&
1410 ((path[out] != '.') || (path[out - 1] != '.') ||
1411 (path[out - 2] != '/'))) {
1412 cur += 4;
1413 out --;
1414 while ((out > 0) && (path[out - 1] != '/')) { out --; }
1415 path[out] = 0;
1416 continue;
1417 }
1418
1419 path[out++] = path[cur++]; /* / or 0 */
1420 }
1421 path[out] = 0;
1422
1423 /*
1424 * g) If the resulting buffer string still begins with one or more
1425 * complete path segments of "..", then the reference is
1426 * considered to be in error. Implementations may handle this
1427 * error by retaining these components in the resolved path (i.e.,
1428 * treating them as part of the final URI), by removing them from
1429 * the resolved path (i.e., discarding relative levels above the
1430 * root), or by avoiding traversal of the reference.
1431 *
1432 * We discard them from the final path.
1433 */
1434 cur = 0;
1435 while ((path[cur] == '/') && (path[cur + 1] == '.') &&
1436 (path[cur + 2] == '.'))
1437 cur += 3;
1438 if (cur != 0) {
1439 out = 0;
1440 while (path[cur] != 0) path[out++] = path[cur++];
1441 path[out] = 0;
1442 }
1443 return(0);
1444}
1445
1446/**
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001447 * xmlBuildURI:
1448 * @URI: the URI instance found in the document
1449 * @base: the base value
1450 *
1451 * Computes he final URI of the reference done by checking that
1452 * the given URI is valid, and building the final URI using the
1453 * base URI. This is processed according to section 5.2 of the
1454 * RFC 2396
1455 *
1456 * 5.2. Resolving Relative References to Absolute Form
1457 *
Daniel Veillardec303412000-03-24 13:41:54 +00001458 * Returns a new URI string (to be freed by the caller) or NULL in case
1459 * of error.
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001460 */
1461xmlChar *
1462xmlBuildURI(const xmlChar *URI, const xmlChar *base) {
Daniel Veillardec303412000-03-24 13:41:54 +00001463 xmlChar *val = NULL;
1464 int ret, len, index, cur, out;
1465 xmlURIPtr ref = NULL;
1466 xmlURIPtr bas = NULL;
1467 xmlURIPtr res = NULL;
1468
1469
1470 /*
1471 * 1) The URI reference is parsed into the potential four components and
1472 * fragment identifier, as described in Section 4.3.
1473 */
1474 ref = xmlCreateURI();
1475 if (ref == NULL)
1476 goto done;
1477 ret = xmlParseURIReference(ref, (const char *) URI);
1478 if (ret != 0)
1479 goto done;
1480 bas = xmlCreateURI();
1481 if (bas == NULL)
1482 goto done;
1483 ret = xmlParseURIReference(bas, (const char *) base);
1484 if (ret != 0)
1485 goto done;
1486
1487 /*
1488 * 2) If the path component is empty and the scheme, authority, and
1489 * query components are undefined, then it is a reference to the
1490 * current document and we are done. Otherwise, the reference URI's
1491 * query and fragment components are defined as found (or not found)
1492 * within the URI reference and not inherited from the base URI.
1493 */
1494 res = xmlCreateURI();
1495 if (res == NULL)
1496 goto done;
1497 if ((ref->scheme == NULL) && (ref->path == NULL) &&
Daniel Veillard361d8452000-04-03 19:48:13 +00001498 ((ref->authority == NULL) && (ref->server == NULL)) &&
1499 (ref->query == NULL)) {
Daniel Veillardec303412000-03-24 13:41:54 +00001500 if (ref->fragment == NULL)
1501 goto done;
1502 res->fragment = xmlMemStrdup(ref->fragment);
1503 val = xmlSaveUri(res);
1504 goto done;
1505 }
1506
1507 /*
1508 * 3) If the scheme component is defined, indicating that the reference
1509 * starts with a scheme name, then the reference is interpreted as an
1510 * absolute URI and we are done. Otherwise, the reference URI's
1511 * scheme is inherited from the base URI's scheme component.
1512 */
1513 if (ref->scheme != NULL) {
1514 val = xmlSaveUri(ref);
1515 goto done;
1516 }
Daniel Veillardbe803962000-06-28 23:40:59 +00001517 if (bas->scheme != NULL)
1518 res->scheme = xmlMemStrdup(bas->scheme);
1519 else
1520 res->scheme = NULL;
Daniel Veillardec303412000-03-24 13:41:54 +00001521
1522 /*
1523 * 4) If the authority component is defined, then the reference is a
1524 * network-path and we skip to step 7. Otherwise, the reference
1525 * URI's authority is inherited from the base URI's authority
1526 * component, which will also be undefined if the URI scheme does not
1527 * use an authority component.
1528 */
Daniel Veillard361d8452000-04-03 19:48:13 +00001529 if ((ref->authority != NULL) || (ref->server != NULL)) {
1530 if (ref->authority != NULL)
1531 res->authority = xmlMemStrdup(ref->authority);
1532 else {
1533 res->server = xmlMemStrdup(ref->server);
1534 if (ref->user != NULL)
1535 res->user = xmlMemStrdup(ref->user);
1536 res->port = ref->port;
1537 }
Daniel Veillardec303412000-03-24 13:41:54 +00001538 if (ref->path != NULL)
1539 res->path = xmlMemStrdup(ref->path);
1540 if (ref->query != NULL)
1541 res->query = xmlMemStrdup(ref->query);
1542 if (ref->fragment != NULL)
1543 res->fragment = xmlMemStrdup(ref->fragment);
1544 goto step_7;
1545 }
1546 if (bas->authority != NULL)
1547 res->authority = xmlMemStrdup(bas->authority);
Daniel Veillard361d8452000-04-03 19:48:13 +00001548 else if (bas->server != NULL) {
1549 res->server = xmlMemStrdup(bas->server);
1550 if (bas->user != NULL)
1551 res->user = xmlMemStrdup(bas->user);
1552 res->port = bas->port;
1553 }
Daniel Veillardec303412000-03-24 13:41:54 +00001554
1555 /*
1556 * 5) If the path component begins with a slash character ("/"), then
1557 * the reference is an absolute-path and we skip to step 7.
1558 */
1559 if ((ref->path != NULL) && (ref->path[0] == '/')) {
1560 res->path = xmlMemStrdup(ref->path);
1561 if (ref->query != NULL)
1562 res->query = xmlMemStrdup(ref->query);
1563 if (ref->fragment != NULL)
1564 res->fragment = xmlMemStrdup(ref->fragment);
1565 goto step_7;
1566 }
1567
1568
1569 /*
1570 * 6) If this step is reached, then we are resolving a relative-path
1571 * reference. The relative path needs to be merged with the base
1572 * URI's path. Although there are many ways to do this, we will
1573 * describe a simple method using a separate string buffer.
1574 *
1575 * Allocate a buffer large enough for the result string.
1576 */
1577 len = 2; /* extra / and 0 */
1578 if (ref->path != NULL)
1579 len += strlen(ref->path);
1580 if (bas->path != NULL)
1581 len += strlen(bas->path);
1582 res->path = (char *) xmlMalloc(len);
1583 if (res->path == NULL) {
1584 fprintf(stderr, "xmlBuildURI: out of memory\n");
1585 goto done;
1586 }
1587 res->path[0] = 0;
1588
1589 /*
1590 * a) All but the last segment of the base URI's path component is
1591 * copied to the buffer. In other words, any characters after the
1592 * last (right-most) slash character, if any, are excluded.
1593 */
1594 cur = 0;
1595 out = 0;
1596 if (bas->path != NULL) {
1597 while (bas->path[cur] != 0) {
1598 while ((bas->path[cur] != 0) && (bas->path[cur] != '/'))
1599 cur++;
1600 if (bas->path[cur] == 0)
1601 break;
1602
1603 cur++;
1604 while (out < cur) {
1605 res->path[out] = bas->path[out];
1606 out++;
1607 }
1608 }
1609 }
1610 res->path[out] = 0;
1611
1612 /*
1613 * b) The reference's path component is appended to the buffer
1614 * string.
1615 */
1616 if (ref->path != NULL) {
1617 index = 0;
Daniel Veillard52402ce2000-08-22 23:36:12 +00001618 /*
1619 * Ensure the path includes a '/'
1620 */
1621 if (res->path[0] != '/' && ref->path[0] != 0 &&
1622 ref->path[index] != '/') {
1623 res->path[out++] = '/';
1624 }
Daniel Veillardec303412000-03-24 13:41:54 +00001625 while (ref->path[index] != 0) {
1626 res->path[out++] = ref->path[index++];
1627 }
1628 }
1629 res->path[out] = 0;
1630
1631 /*
1632 * Steps c) to h) are really path normalization steps
1633 */
1634 xmlNormalizeURIPath(res->path);
1635
1636step_7:
1637
1638 /*
1639 * 7) The resulting URI components, including any inherited from the
1640 * base URI, are recombined to give the absolute form of the URI
1641 * reference.
1642 */
1643 val = xmlSaveUri(res);
1644
1645done:
1646 if (ref != NULL)
1647 xmlFreeURI(ref);
1648 if (base != NULL)
1649 xmlFreeURI(bas);
1650 if (res != NULL)
1651 xmlFreeURI(res);
1652 return(val);
Daniel Veillard3dd82e72000-03-20 11:48:04 +00001653}
1654
1655