viruri.c 7.9 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
    if (VIR_RESIZE_N(uri->params, uri->paramsAlloc, uri->paramsCount, 1) < 0)
44
        goto error;
45 46 47 48 49 50 51 52

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

    return 0;

53
error:
54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
    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. */
73
        end = strchr(query, '&');
74
        if (!end)
75
            end = strchr(query, ';');
76
        if (!end)
77
            end = query + strlen(query);
78 79

        /* Find the first '=' character between here and end. */
80
        eq = strchr(query, '=');
81 82 83 84 85 86 87 88 89 90
        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) {
91
            name = xmlURIUnescapeString(query, end - query, NULL);
92 93 94 95 96 97
            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) {
98
            name = xmlURIUnescapeString(query, eq - query, NULL);
99 100 101 102 103 104 105 106 107 108
            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 {
109
            name = xmlURIUnescapeString(query, eq - query, NULL);
110 111
            if (!name)
                goto no_memory;
112
            value = xmlURIUnescapeString(eq+1, end - (eq+1), NULL);
113 114 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
            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 已提交
140 141 142 143 144 145 146 147 148 149 150 151 152 153 154
/**
 * 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)
{
155 156
    xmlURIPtr xmluri;
    virURIPtr ret = NULL;
M
Martin Kletzander 已提交
157

158 159 160
    xmluri = xmlParseURI(uri);

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

167
    if (VIR_ALLOC(ret) < 0)
168
        goto error;
169

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

M
Martin Kletzander 已提交
189
    /* First check: does it even make sense to jump inside */
190
    if (ret->server != NULL &&
M
Martin Kletzander 已提交
191 192 193 194 195 196 197 198 199 200 201 202 203 204 205
        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() */
    }

206 207 208
    if (virURIParseParams(ret) < 0)
        goto error;

209 210
    xmlFreeURI(xmluri);

M
Martin Kletzander 已提交
211
    return ret;
212

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

/**
 * 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 *
231
virURIFormat(virURIPtr uri)
M
Martin Kletzander 已提交
232
{
233
    xmlURI xmluri;
M
Martin Kletzander 已提交
234 235 236
    char *tmpserver = NULL;
    char *ret;

237 238 239 240 241 242
    memset(&xmluri, 0, sizeof(xmluri));

    xmluri.scheme = uri->scheme;
    xmluri.server = uri->server;
    xmluri.port = uri->port;
    xmluri.path = uri->path;
243 244 245
#ifdef HAVE_XMLURI_QUERY_RAW
    xmluri.query_raw = uri->query;
#else
246
    xmluri.query = uri->query;
247
#endif
248
    xmluri.fragment = uri->fragment;
249
    xmluri.user = uri->user;
250

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

255
        if (virAsprintf(&tmpserver, "[%s]", xmluri.server) < 0)
M
Martin Kletzander 已提交
256 257
            return NULL;

258
        xmluri.server = tmpserver;
M
Martin Kletzander 已提交
259 260
    }

261
    ret = (char *)xmlSaveUri(&xmluri);
262 263 264 265
    if (!ret) {
        virReportOOMError();
        goto cleanup;
    }
M
Martin Kletzander 已提交
266

267
cleanup:
268
    VIR_FREE(tmpserver);
M
Martin Kletzander 已提交
269 270 271

    return ret;
}
272 273


274 275 276
char *virURIFormatParams(virURIPtr uri)
{
    virBuffer buf = VIR_BUFFER_INITIALIZER;
277
    size_t i;
278
    bool amp = false;
279 280 281

    for (i = 0; i < uri->paramsCount; ++i) {
        if (!uri->params[i].ignore) {
282 283 284
            if (amp) virBufferAddChar(&buf, '&');
            virBufferStrcat(&buf, uri->params[i].name, "=", NULL);
            virBufferURIEncodeString(&buf, uri->params[i].value);
285
            amp = true;
286 287 288 289 290 291 292 293 294 295 296 297
        }
    }

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

    return virBufferContentAndReset(&buf);
}

298 299 300 301 302 303 304 305
/**
 * virURIFree:
 * @uri: uri to free
 *
 * Frees the URI
 */
void virURIFree(virURIPtr uri)
{
306 307
    size_t i;

308 309 310 311 312
    if (!uri)
        return;

    VIR_FREE(uri->scheme);
    VIR_FREE(uri->server);
313
    VIR_FREE(uri->user);
314 315
    VIR_FREE(uri->path);
    VIR_FREE(uri->query);
316 317
    VIR_FREE(uri->fragment);

318
    for (i = 0; i < uri->paramsCount; i++) {
319 320 321 322 323
        VIR_FREE(uri->params[i].name);
        VIR_FREE(uri->params[i].value);
    }
    VIR_FREE(uri->params);

324
    VIR_FREE(uri);
325
}