viruri.c 8.0 KB
Newer Older
M
Martin Kletzander 已提交
1 2 3 4 5
/*
 * viruri.c: URI parsing wrappers for libxml2 functions
 *
 * Copyright (C) 2012 Red Hat, Inc.
 *
O
Osier Yang 已提交
6 7 8 9 10 11 12 13 14 15 16
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
17
 * License along with this library.  If not, see
O
Osier Yang 已提交
18
 * <http://www.gnu.org/licenses/>.
M
Martin Kletzander 已提交
19 20 21 22 23 24
 */

#include <config.h>

#include "viruri.h"

25
#include "viralloc.h"
26
#include "virerror.h"
27
#include "virbuffer.h"
28
#include "virstring.h"
29 30 31

#define VIR_FROM_THIS VIR_FROM_URI

32 33 34 35 36 37 38 39
static int
virURIParamAppend(virURIPtr uri,
                  const char *name,
                  const char *value)
{
    char *pname = NULL;
    char *pvalue = NULL;

40 41
    if (VIR_STRDUP(pname, name) < 0 || VIR_STRDUP(pvalue, value) < 0)
        goto error;
42

43 44 45 46
    if (VIR_RESIZE_N(uri->params, uri->paramsAlloc, uri->paramsCount, 1) < 0) {
        virReportOOMError();
        goto error;
    }
47 48 49 50 51 52 53 54

    uri->params[uri->paramsCount].name = pname;
    uri->params[uri->paramsCount].value = pvalue;
    uri->params[uri->paramsCount].ignore = 0;
    uri->paramsCount++;

    return 0;

55
error:
56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
    VIR_FREE(pname);
    VIR_FREE(pvalue);
    return -1;
}


static int
virURIParseParams(virURIPtr uri)
{
    const char *end, *eq;
    const char *query = uri->query;

    if (!query || query[0] == '\0')
        return 0;

    while (*query) {
        char *name = NULL, *value = NULL;

        /* Find the next separator, or end of the string. */
75
        end = strchr(query, '&');
76
        if (!end)
77
            end = strchr(query, ';');
78
        if (!end)
79
            end = query + strlen(query);
80 81

        /* Find the first '=' character between here and end. */
82
        eq = strchr(query, '=');
83 84 85 86 87 88 89 90 91 92
        if (eq && eq >= end) eq = NULL;

        /* Empty section (eg. "&&"). */
        if (end == query)
            goto next;

        /* If there is no '=' character, then we have just "name"
         * and consistent with CGI.pm we assume value is "".
         */
        else if (!eq) {
93
            name = xmlURIUnescapeString(query, end - query, NULL);
94 95 96 97 98 99
            if (!name) goto no_memory;
        }
        /* Or if we have "name=" here (works around annoying
         * problem when calling xmlURIUnescapeString with len = 0).
         */
        else if (eq+1 == end) {
100
            name = xmlURIUnescapeString(query, eq - query, NULL);
101 102 103 104 105 106 107 108 109 110
            if (!name) goto no_memory;
        }
        /* If the '=' character is at the beginning then we have
         * "=value" and consistent with CGI.pm we _ignore_ this.
         */
        else if (query == eq)
            goto next;

        /* Otherwise it's "name=value". */
        else {
111
            name = xmlURIUnescapeString(query, eq - query, NULL);
112 113
            if (!name)
                goto no_memory;
114
            value = xmlURIUnescapeString(eq+1, end - (eq+1), NULL);
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
            if (!value) {
                VIR_FREE(name);
                goto no_memory;
            }
        }

        /* Append to the parameter set. */
        if (virURIParamAppend(uri, name, value ? value : "") < 0) {
            VIR_FREE(name);
            VIR_FREE(value);
            goto no_memory;
        }
        VIR_FREE(name);
        VIR_FREE(value);

    next:
        query = end;
        if (*query) query ++; /* skip '&' separator */
    }

    return 0;

 no_memory:
    virReportOOMError();
    return -1;
}

M
Martin Kletzander 已提交
142 143 144 145 146 147 148 149 150 151 152 153 154 155 156
/**
 * virURIParse:
 * @uri: URI to parse
 *
 * Wrapper for xmlParseURI
 *
 * Unfortunately there are few things that should be managed after
 * parsing the URI. Fortunately there is only one thing now and its
 * removing of square brackets around IPv6 addresses.
 *
 * @returns the parsed uri object with some fixes
 */
virURIPtr
virURIParse(const char *uri)
{
157 158
    xmlURIPtr xmluri;
    virURIPtr ret = NULL;
M
Martin Kletzander 已提交
159

160 161 162
    xmluri = xmlParseURI(uri);

    if (!xmluri) {
163
        /* libxml2 does not tell us what failed. Grr :-( */
164 165
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Unable to parse URI %s"), uri);
166 167 168
        return NULL;
    }

169 170 171 172
    if (VIR_ALLOC(ret) < 0) {
        virReportOOMError();
        goto error;
    }
173

174 175 176 177
    if (VIR_STRDUP(ret->scheme, xmluri->scheme) < 0)
        goto error;
    if (VIR_STRDUP(ret->server, xmluri->server) < 0)
        goto error;
178
    ret->port = xmluri->port;
179 180
    if (VIR_STRDUP(ret->path, xmluri->path) < 0)
        goto error;
181
#ifdef HAVE_XMLURI_QUERY_RAW
182 183
    if (VIR_STRDUP(ret->query, xmluri->query_raw) < 0)
        goto error;
184
#else
185 186
    if (VIR_STRDUP(ret->query, xmluri->query) < 0)
        goto error;
187
#endif
188 189 190 191
    if (VIR_STRDUP(ret->fragment, xmluri->fragment) < 0)
        goto error;
    if (VIR_STRDUP(ret->user, xmluri->user) < 0)
        goto error;
192

M
Martin Kletzander 已提交
193
    /* First check: does it even make sense to jump inside */
194
    if (ret->server != NULL &&
M
Martin Kletzander 已提交
195 196 197 198 199 200 201 202 203 204 205 206 207 208 209
        ret->server[0] == '[') {
        size_t length = strlen(ret->server);

        /* We want to modify the server string only if there are
         * square brackets on both ends and inside there is IPv6
         * address. Otherwise we could make a mistake by modifying
         * something other than an IPv6 address. */
        if (ret->server[length - 1] == ']' && strchr(ret->server, ':')) {
            memmove(&ret->server[0], &ret->server[1], length - 2);
            ret->server[length - 2] = '\0';
        }
        /* Even after such modification, it is completely ok to free
         * the uri with xmlFreeURI() */
    }

210 211 212
    if (virURIParseParams(ret) < 0)
        goto error;

213 214
    xmlFreeURI(xmluri);

M
Martin Kletzander 已提交
215
    return ret;
216

217
error:
218 219 220
    xmlFreeURI(xmluri);
    virURIFree(ret);
    return NULL;
M
Martin Kletzander 已提交
221 222 223 224 225 226 227 228 229 230 231 232 233 234
}

/**
 * virURIFormat:
 * @uri: URI to format
 *
 * Wrapper for xmlSaveUri
 *
 * This function constructs back everything that @ref virURIParse
 * changes after parsing
 *
 * @returns the constructed uri as a string
 */
char *
235
virURIFormat(virURIPtr uri)
M
Martin Kletzander 已提交
236
{
237
    xmlURI xmluri;
M
Martin Kletzander 已提交
238 239 240
    char *tmpserver = NULL;
    char *ret;

241 242 243 244 245 246
    memset(&xmluri, 0, sizeof(xmluri));

    xmluri.scheme = uri->scheme;
    xmluri.server = uri->server;
    xmluri.port = uri->port;
    xmluri.path = uri->path;
247 248 249
#ifdef HAVE_XMLURI_QUERY_RAW
    xmluri.query_raw = uri->query;
#else
250
    xmluri.query = uri->query;
251
#endif
252
    xmluri.fragment = uri->fragment;
253
    xmluri.user = uri->user;
254

M
Martin Kletzander 已提交
255
    /* First check: does it make sense to do anything */
256 257
    if (xmluri.server != NULL &&
        strchr(xmluri.server, ':') != NULL) {
M
Martin Kletzander 已提交
258

259 260
        if (virAsprintf(&tmpserver, "[%s]", xmluri.server) < 0) {
            virReportOOMError();
M
Martin Kletzander 已提交
261
            return NULL;
262
        }
M
Martin Kletzander 已提交
263

264
        xmluri.server = tmpserver;
M
Martin Kletzander 已提交
265 266
    }

267
    ret = (char *)xmlSaveUri(&xmluri);
268 269 270 271
    if (!ret) {
        virReportOOMError();
        goto cleanup;
    }
M
Martin Kletzander 已提交
272

273
cleanup:
274
    VIR_FREE(tmpserver);
M
Martin Kletzander 已提交
275 276 277

    return ret;
}
278 279


280 281 282 283 284 285 286
char *virURIFormatParams(virURIPtr uri)
{
    virBuffer buf = VIR_BUFFER_INITIALIZER;
    int i, amp = 0;

    for (i = 0; i < uri->paramsCount; ++i) {
        if (!uri->params[i].ignore) {
287 288 289
            if (amp) virBufferAddChar(&buf, '&');
            virBufferStrcat(&buf, uri->params[i].name, "=", NULL);
            virBufferURIEncodeString(&buf, uri->params[i].value);
290 291 292 293 294 295 296 297 298 299 300 301 302
            amp = 1;
        }
    }

    if (virBufferError(&buf)) {
        virBufferFreeAndReset(&buf);
        virReportOOMError();
        return NULL;
    }

    return virBufferContentAndReset(&buf);
}

303 304 305 306 307 308 309 310
/**
 * virURIFree:
 * @uri: uri to free
 *
 * Frees the URI
 */
void virURIFree(virURIPtr uri)
{
311 312
    size_t i;

313 314 315 316 317
    if (!uri)
        return;

    VIR_FREE(uri->scheme);
    VIR_FREE(uri->server);
318
    VIR_FREE(uri->user);
319 320
    VIR_FREE(uri->path);
    VIR_FREE(uri->query);
321 322
    VIR_FREE(uri->fragment);

323
    for (i = 0; i < uri->paramsCount; i++) {
324 325 326 327 328
        VIR_FREE(uri->params[i].name);
        VIR_FREE(uri->params[i].value);
    }
    VIR_FREE(uri->params);

329
    VIR_FREE(uri);
330
}