From 861f88aa43cebd6dbfd0ecde175a4e81b995c2a1 Mon Sep 17 00:00:00 2001 From: Michael Drake Date: Wed, 28 Sep 2011 11:26:30 +0000 Subject: [PATCH] Port llcache to nsurl. svn path=/trunk/netsurf/; revision=12904 --- content/hlcache.c | 34 ++++++- content/llcache.c | 247 ++++++++++++++++++---------------------------- content/llcache.h | 7 +- desktop/browser.c | 23 ++++- desktop/netsurf.c | 5 +- 5 files changed, 157 insertions(+), 159 deletions(-) diff --git a/content/hlcache.c b/content/hlcache.c index db25f08ad..f7da968b5 100644 --- a/content/hlcache.c +++ b/content/hlcache.c @@ -233,6 +233,8 @@ nserror hlcache_handle_retrieve(const char *url, uint32_t flags, { hlcache_retrieval_ctx *ctx; nserror error; + nsurl *nsref = NULL; + nsurl *nsurl; assert(cb != NULL); @@ -264,9 +266,7 @@ nserror hlcache_handle_retrieve(const char *url, uint32_t flags, ctx->handle->cb = cb; ctx->handle->pw = pw; - error = llcache_handle_retrieve(url, flags, referer, post, - hlcache_llcache_callback, ctx, - &ctx->llcache); + error = nsurl_create(url, &nsurl); if (error != NSERROR_OK) { free((char *) ctx->child.charset); free(ctx->handle); @@ -274,6 +274,34 @@ nserror hlcache_handle_retrieve(const char *url, uint32_t flags, return error; } + if (referer != NULL) { + error = nsurl_create(referer, &nsref); + if (error != NSERROR_OK) { + free((char *) ctx->child.charset); + free(ctx->handle); + free(ctx); + nsurl_unref(nsurl); + return error; + } + } + + error = llcache_handle_retrieve(nsurl, flags, nsref, post, + hlcache_llcache_callback, ctx, + &ctx->llcache); + if (error != NSERROR_OK) { + nsurl_unref(nsurl); + if (nsref != NULL) + nsurl_unref(nsref); + free((char *) ctx->child.charset); + free(ctx->handle); + free(ctx); + return error; + } + + nsurl_unref(nsurl); + if (nsref != NULL) + nsurl_unref(nsref); + RING_INSERT(hlcache->retrieval_ctx_ring, ctx); *result = ctx->handle; diff --git a/content/llcache.c b/content/llcache.c index afa2702b9..124f98652 100644 --- a/content/llcache.c +++ b/content/llcache.c @@ -31,7 +31,7 @@ #include "content/urldb.h" #include "utils/log.h" #include "utils/messages.h" -#include "utils/url.h" +#include "utils/nsurl.h" #include "utils/utils.h" /** Define to enable tracing of llcache operations. */ @@ -73,7 +73,7 @@ typedef struct llcache_object_user { /** Low-level cache object fetch context */ typedef struct { uint32_t flags; /**< Fetch flags */ - char *referer; /**< Referring URL, or NULL if none */ + nsurl *referer; /**< Referring URL, or NULL if none */ llcache_post_data *post; /**< POST data, or NULL for GET */ struct fetch *fetch; /**< Fetch handle for this object */ @@ -119,7 +119,7 @@ struct llcache_object { llcache_object *prev; /**< Previous in list */ llcache_object *next; /**< Next in list */ - char *url; /**< Post-redirect URL for object */ + nsurl *url; /**< Post-redirect URL for object */ bool has_query; /**< URL has a query segment */ /** \todo We need a generic dynamic buffer object */ @@ -159,15 +159,21 @@ struct llcache_s { /** low level cache state */ static struct llcache_s *llcache = NULL; +/* Static lwc_strings */ +static lwc_string *llcache_file_lwc; +static lwc_string *llcache_about_lwc; +static lwc_string *llcache_resource_lwc; + + static nserror llcache_object_user_new(llcache_handle_callback cb, void *pw, llcache_object_user **user); static nserror llcache_object_user_destroy(llcache_object_user *user); -static nserror llcache_object_retrieve(const char *url, uint32_t flags, - const char *referer, const llcache_post_data *post, +static nserror llcache_object_retrieve(nsurl *url, uint32_t flags, + nsurl *referer, const llcache_post_data *post, uint32_t redirect_count, llcache_object **result); -static nserror llcache_object_retrieve_from_cache(const char *url, - uint32_t flags, const char *referer, +static nserror llcache_object_retrieve_from_cache(nsurl *url, + uint32_t flags, nsurl *referer, const llcache_post_data *post, uint32_t redirect_count, llcache_object **result); static bool llcache_object_is_fresh(const llcache_object *object); @@ -175,11 +181,11 @@ static nserror llcache_object_cache_update(llcache_object *object); static nserror llcache_object_clone_cache_data(llcache_object *source, llcache_object *destination, bool deep); static nserror llcache_object_fetch(llcache_object *object, uint32_t flags, - const char *referer, const llcache_post_data *post, + nsurl *referer, const llcache_post_data *post, uint32_t redirect_count); static nserror llcache_object_refetch(llcache_object *object); -static nserror llcache_object_new(const char *url, llcache_object **result); +static nserror llcache_object_new(nsurl *url, llcache_object **result); static nserror llcache_object_destroy(llcache_object *object); static nserror llcache_object_add_user(llcache_object *object, llcache_object_user *user); @@ -265,6 +271,19 @@ llcache_initialise(llcache_query_callback cb, void *pw, uint32_t llcache_limit) llcache->query_cb_pw = pw; llcache->limit = llcache_limit; + /* Create static scheme strings */ + if (lwc_intern_string("file", SLEN("file"), + &llcache_file_lwc) != lwc_error_ok) + return NSERROR_NOMEM; + + if (lwc_intern_string("about", SLEN("about"), + &llcache_about_lwc) != lwc_error_ok) + return NSERROR_NOMEM; + + if (lwc_intern_string("resource", SLEN("resource"), + &llcache_resource_lwc) != lwc_error_ok) + return NSERROR_NOMEM; + LOG(("llcache initialised with a limit of %d bytes", llcache_limit)); return NSERROR_OK; @@ -317,6 +336,11 @@ void llcache_finalise(void) llcache_object_destroy(object); } + /* Unref static scheme lwc strings */ + lwc_string_unref(llcache_file_lwc); + lwc_string_unref(llcache_about_lwc); + lwc_string_unref(llcache_resource_lwc); + free(llcache); llcache = NULL; } @@ -343,28 +367,19 @@ nserror llcache_poll(void) } /* See llcache.h for documentation */ -nserror llcache_handle_retrieve(const char *url, uint32_t flags, - const char *referer, const llcache_post_data *post, +nserror llcache_handle_retrieve(nsurl *url, uint32_t flags, + nsurl *referer, const llcache_post_data *post, llcache_handle_callback cb, void *pw, llcache_handle **result) { nserror error; llcache_object_user *user; llcache_object *object; - nsurl *nsurl; - - /* make a nsurl */ - error = nsurl_create(url, &nsurl); - if (error != NSERROR_OK) - return error; /* Can we fetch this URL at all? */ - if (fetch_can_fetch(nsurl) == false) + if (fetch_can_fetch(url) == false) return NSERROR_NO_FETCH_HANDLER; - /* done with nsurl */ - nsurl_unref(nsurl); - /* Create a new object user */ error = llcache_object_user_new(cb, pw, &user); if (error != NSERROR_OK) @@ -535,7 +550,9 @@ nserror llcache_handle_invalidate_cache_data(llcache_handle *handle) /* See llcache.h for documentation */ const char *llcache_handle_get_url(const llcache_handle *handle) { - return handle->object != NULL ? handle->object->url : NULL; + /* TODO: return a nsurl? */ + return handle->object != NULL ? + nsurl_access(handle->object->url) : NULL; } /* See llcache.h for documentation */ @@ -689,16 +706,14 @@ static nserror llcache_send_event_to_users(llcache_object *object, * \param result Pointer to location to recieve retrieved object * \return NSERROR_OK on success, appropriate error otherwise */ -nserror llcache_object_retrieve(const char *url, uint32_t flags, - const char *referer, const llcache_post_data *post, +nserror llcache_object_retrieve(nsurl *url, uint32_t flags, + nsurl *referer, const llcache_post_data *post, uint32_t redirect_count, llcache_object **result) { nserror error; llcache_object *obj; bool has_query; - url_func_result res; - struct url_components components; - char *defragmented_url; + nsurl *defragmented_url; #ifdef LLCACHE_TRACE LOG(("Retrieve %s (%x, %s, %p)", url, flags, referer, post)); @@ -712,26 +727,17 @@ nserror llcache_object_retrieve(const char *url, uint32_t flags, */ /* Look for a query segment */ - res = url_get_components(url, &components); - if (res == URL_FUNC_NOMEM) - return NSERROR_NOMEM; + has_query = nsurl_enquire(url, NSURL_QUERY); - has_query = (components.query != NULL); - - components.fragment = NULL; - - defragmented_url = url_reform_components(&components); - - url_destroy_components(&components); - - if (defragmented_url == NULL) - return NSERROR_NOMEM; + error = nsurl_defragment(url, &defragmented_url); + if (error != NSERROR_OK) + return error; if (flags & LLCACHE_RETRIEVE_FORCE_FETCH || post != NULL) { /* Create new object */ error = llcache_object_new(defragmented_url, &obj); if (error != NSERROR_OK) { - free(defragmented_url); + nsurl_unref(defragmented_url); return error; } @@ -740,7 +746,7 @@ nserror llcache_object_retrieve(const char *url, uint32_t flags, redirect_count); if (error != NSERROR_OK) { llcache_object_destroy(obj); - free(defragmented_url); + nsurl_unref(defragmented_url); return error; } @@ -750,7 +756,7 @@ nserror llcache_object_retrieve(const char *url, uint32_t flags, error = llcache_object_retrieve_from_cache(defragmented_url, flags, referer, post, redirect_count, &obj); if (error != NSERROR_OK) { - free(defragmented_url); + nsurl_unref(defragmented_url); return error; } @@ -765,7 +771,7 @@ nserror llcache_object_retrieve(const char *url, uint32_t flags, *result = obj; - free(defragmented_url); + nsurl_unref(defragmented_url); return NSERROR_OK; } @@ -781,8 +787,8 @@ nserror llcache_object_retrieve(const char *url, uint32_t flags, * \param result Pointer to location to recieve retrieved object * \return NSERROR_OK on success, appropriate error otherwise */ -nserror llcache_object_retrieve_from_cache(const char *url, uint32_t flags, - const char *referer, const llcache_post_data *post, +nserror llcache_object_retrieve_from_cache(nsurl *url, uint32_t flags, + nsurl *referer, const llcache_post_data *post, uint32_t redirect_count, llcache_object **result) { nserror error; @@ -794,13 +800,11 @@ nserror llcache_object_retrieve_from_cache(const char *url, uint32_t flags, /* Search for the most recently fetched matching object */ for (obj = llcache->cached_objects; obj != NULL; obj = obj->next) { - bool match; if ((newest == NULL || obj->cache.req_time > newest->cache.req_time) && - url_compare(obj->url, url, true, - &match) == URL_FUNC_OK && - match == true) { + nsurl_compare(obj->url, url, + NSURL_COMPLETE) == true) { newest = obj; } } @@ -1013,31 +1017,26 @@ nserror llcache_object_clone_cache_data(llcache_object *source, * \pre There must not be a fetch in progress for \a object */ nserror llcache_object_fetch(llcache_object *object, uint32_t flags, - const char *referer, const llcache_post_data *post, + nsurl *referer, const llcache_post_data *post, uint32_t redirect_count) { nserror error; - char *referer_clone = NULL; + nsurl *referer_clone = NULL; llcache_post_data *post_clone = NULL; #ifdef LLCACHE_TRACE LOG(("Starting fetch for %p", object)); #endif - if (referer != NULL) { - referer_clone = strdup(referer); - if (referer_clone == NULL) - return NSERROR_NOMEM; - } - if (post != NULL) { error = llcache_post_data_clone(post, &post_clone); - if (error != NSERROR_OK) { - free(referer_clone); + if (error != NSERROR_OK) return error; - } } + if (referer != NULL) + referer_clone = nsurl_ref(referer); + object->fetch.flags = flags; object->fetch.referer = referer_clone; object->fetch.post = post_clone; @@ -1060,8 +1059,6 @@ nserror llcache_object_refetch(llcache_object *object) struct fetch_multipart_data *multipart = NULL; char **headers = NULL; int header_idx = 0; - nsurl*nsurl, *nsurl_ref; - nserror error; if (object->fetch.post != NULL) { if (object->fetch.post->type == LLCACHE_POST_URL_ENCODED) @@ -1120,35 +1117,14 @@ nserror llcache_object_refetch(llcache_object *object) LOG(("Refetching %p", object)); #endif - /* make nsurls */ - error = nsurl_create(object->url, &nsurl); - if (error != NSERROR_OK) { - free(headers); - return error; - } - if (object->fetch.referer != NULL) { - error = nsurl_create(object->fetch.referer, &nsurl_ref); - if (error != NSERROR_OK) { - nsurl_unref(nsurl); - free(headers); - return error; - } - } else { - nsurl_ref = NULL; - } - /* Kick off fetch */ - object->fetch.fetch = fetch_start(nsurl, nsurl_ref, + object->fetch.fetch = fetch_start(object->url, object->fetch.referer, llcache_fetch_callback, object, object->fetch.flags & LLCACHE_RETRIEVE_NO_ERROR_PAGES, urlenc, multipart, object->fetch.flags & LLCACHE_RETRIEVE_VERIFIABLE, (const char **) headers); - nsurl_unref(nsurl); - if (nsurl_ref != NULL) - nsurl_unref(nsurl_ref); - /* Clean up cache-control headers */ while (--header_idx >= 0) free(headers[header_idx]); @@ -1168,21 +1144,17 @@ nserror llcache_object_refetch(llcache_object *object) * \param result Pointer to location to receive result * \return NSERROR_OK on success, appropriate error otherwise */ -nserror llcache_object_new(const char *url, llcache_object **result) +nserror llcache_object_new(nsurl *url, llcache_object **result) { llcache_object *obj = calloc(1, sizeof(llcache_object)); if (obj == NULL) return NSERROR_NOMEM; #ifdef LLCACHE_TRACE - LOG(("Created object %p (%s)", obj, url)); + LOG(("Created object %p (%s)", obj, nsurl_access(url))); #endif - obj->url = strdup(url); - if (obj->url == NULL) { - free(obj); - return NSERROR_NOMEM; - } + obj->url = nsurl_ref(url); *result = obj; @@ -1207,7 +1179,7 @@ nserror llcache_object_destroy(llcache_object *object) LOG(("Destroying object %p", object)); #endif - free(object->url); + nsurl_unref(object->url); free(object->source_data); if (object->fetch.fetch != NULL) { @@ -1215,7 +1187,8 @@ nserror llcache_object_destroy(llcache_object *object) object->fetch.fetch = NULL; } - free(object->fetch.referer); + if (object->fetch.referer != NULL) + nsurl_unref(object->fetch.referer); if (object->fetch.post != NULL) { if (object->fetch.post->type == LLCACHE_POST_URL_ENCODED) { @@ -2026,13 +1999,12 @@ nserror llcache_fetch_redirect(llcache_object *object, const char *target, llcache_object *dest; llcache_object_user *user, *next; const llcache_post_data *post = object->fetch.post; - char *url, *absurl; - char *scheme; - char *object_scheme; - url_func_result result; + nsurl *url; + lwc_string *scheme; + lwc_string *object_scheme; + bool match; /* Extract HTTP response code from the fetch object */ long http_code = fetch_http_code(object->fetch.fetch); - nsurl *nsurl; /* Abort fetch for this object */ fetch_abort(object->fetch.fetch); @@ -2059,75 +2031,50 @@ nserror llcache_fetch_redirect(llcache_object *object, const char *target, #undef REDIRECT_LIMIT /* Make target absolute */ - result = url_join(target, object->url, &absurl); - if (result != URL_FUNC_OK) { - return NSERROR_NOMEM; - } - - /* Ensure target is normalised */ - result = url_normalize(absurl, &url); - - /* No longer require absolute url */ - free(absurl); - - if (result != URL_FUNC_OK) { - return NSERROR_NOMEM; - } + error = nsurl_join(object->url, target, &url); + if (error != NSERROR_OK) + return error; /* Reject attempts to redirect from unvalidated to validated schemes * A "validated" scheme is one over which we have some guarantee that * the source is trustworthy. */ - result = url_scheme(object->url, &object_scheme); - if (result != URL_FUNC_OK) { - free(url); - return NSERROR_NOMEM; - } - - result = url_scheme(url, &scheme); - if (result != URL_FUNC_OK) { - free(object_scheme); - free(url); - return NSERROR_NOMEM; - } + object_scheme = nsurl_get_component(object->url, NSURL_SCHEME); + scheme = nsurl_get_component(url, NSURL_SCHEME); /* resource: and about: are allowed to redirect anywhere */ - if ((strcasecmp(object_scheme, "resource") != 0) && - (strcasecmp(object_scheme, "about") != 0)) { + if ((lwc_string_isequal(object_scheme, llcache_resource_lwc, + &match) == lwc_error_ok && match == false) && + (lwc_string_isequal(object_scheme, llcache_about_lwc, + &match) == lwc_error_ok && match == false)) { /* file, about and resource are not valid redirect targets */ - if ((strcasecmp(scheme, "file") == 0) || - (strcasecmp(scheme, "about") == 0) || - (strcasecmp(scheme, "resource") == 0)) { - free(object_scheme); - free(scheme); - free(url); + if ((lwc_string_isequal(object_scheme, llcache_file_lwc, + &match) == lwc_error_ok && match == true) || + (lwc_string_isequal(object_scheme, llcache_about_lwc, + &match) == lwc_error_ok && match == true) || + (lwc_string_isequal(object_scheme, llcache_resource_lwc, + &match) == lwc_error_ok && match == true)) { + lwc_string_unref(object_scheme); + lwc_string_unref(scheme); + nsurl_unref(url); return NSERROR_OK; } } - free(scheme); - free(object_scheme); - - /* make a nsurl */ - error = nsurl_create(url, &nsurl); - if (error != NSERROR_OK) { - free(url); - return error; - } + lwc_string_unref(scheme); + lwc_string_unref(object_scheme); /* Bail out if we've no way of handling this URL */ - if (fetch_can_fetch(nsurl) == false) { - free(url); - nsurl_unref(nsurl); + if (fetch_can_fetch(url) == false) { + nsurl_unref(url); return NSERROR_OK; } - nsurl_unref(nsurl); if (http_code == 301 || http_code == 302 || http_code == 303) { /* 301, 302, 303 redirects are all unconditional GET requests */ post = NULL; } else if (http_code != 307 || post != NULL) { /** \todo 300, 305, 307 with POST */ - free(url); + nsurl_unref(url); return NSERROR_OK; } @@ -2137,7 +2084,7 @@ nserror llcache_fetch_redirect(llcache_object *object, const char *target, object->fetch.redirect_count + 1, &dest); /* No longer require url */ - free(url); + nsurl_unref(url); if (error != NSERROR_OK) return error; @@ -2512,9 +2459,9 @@ nserror llcache_fetch_auth(llcache_object *object, const char *realm) /* If there was no realm, then default to the URL */ /** \todo If there was no WWW-Authenticate header, use response body */ if (realm == NULL) - realm = object->url; + realm = nsurl_access(object->url); - auth = urldb_get_auth_details(object->url, realm); + auth = urldb_get_auth_details(nsurl_access(object->url), realm); if (auth == NULL || object->fetch.tried_with_auth == true) { /* No authentication details, or tried what we had, so ask */ diff --git a/content/llcache.h b/content/llcache.h index 8070db90d..337408167 100644 --- a/content/llcache.h +++ b/content/llcache.h @@ -28,6 +28,7 @@ #include #include "utils/errors.h" +#include "utils/nsurl.h" struct ssl_cert_info; struct fetch_multipart_data; @@ -106,7 +107,7 @@ typedef enum { typedef struct { llcache_query_type type; /**< Type of query */ - const char *url; /**< URL being fetched */ + nsurl *url; /**< URL being fetched */ union { struct { @@ -189,8 +190,8 @@ void llcache_clean(void); * \param result Pointer to location to recieve cache handle * \return NSERROR_OK on success, appropriate error otherwise */ -nserror llcache_handle_retrieve(const char *url, uint32_t flags, - const char *referer, const llcache_post_data *post, +nserror llcache_handle_retrieve(nsurl *url, uint32_t flags, + nsurl *referer, const llcache_post_data *post, llcache_handle_callback cb, void *pw, llcache_handle **result); diff --git a/desktop/browser.c b/desktop/browser.c index b2a9f17b6..4cf79fba6 100644 --- a/desktop/browser.c +++ b/desktop/browser.c @@ -60,6 +60,7 @@ #include "render/textplain.h" #include "utils/log.h" #include "utils/messages.h" +#include "utils/nsurl.h" #include "utils/schedule.h" #include "utils/url.h" #include "utils/utils.h" @@ -752,11 +753,28 @@ void browser_window_go_post(struct browser_window *bw, const char *url, /* Get download out of the way */ if (download) { llcache_handle *l; + nsurl *nsref = NULL; + nsurl *nsurl; + + error = nsurl_create(url2, &nsurl); + if (error != NSERROR_OK) { + free(url2); + return; + } + + if (referer != NULL) { + error = nsurl_create(referer, &nsref); + if (error != NSERROR_OK) { + free(url2); + nsurl_unref(nsurl); + return; + } + } fetch_flags |= LLCACHE_RETRIEVE_FORCE_FETCH; fetch_flags |= LLCACHE_RETRIEVE_STREAM_DATA; - error = llcache_handle_retrieve(url2, fetch_flags, referer, + error = llcache_handle_retrieve(nsurl, fetch_flags, nsref, fetch_is_post ? &post : NULL, NULL, NULL, &l); if (error == NSERROR_NO_FETCH_HANDLER) { @@ -774,6 +792,9 @@ void browser_window_go_post(struct browser_window *bw, const char *url, } free(url2); + nsurl_unref(nsurl); + if (nsref != NULL) + nsurl_unref(nsref); return; } diff --git a/desktop/netsurf.c b/desktop/netsurf.c index 91f0fac9b..8fa02255f 100644 --- a/desktop/netsurf.c +++ b/desktop/netsurf.c @@ -94,14 +94,15 @@ static nserror netsurf_llcache_query_handler(const llcache_query *query, { switch (query->type) { case LLCACHE_QUERY_AUTH: - gui_401login_open(query->url, query->data.auth.realm, cb, cbpw); + gui_401login_open(nsurl_access(query->url), + query->data.auth.realm, cb, cbpw); break; case LLCACHE_QUERY_REDIRECT: /** \todo Need redirect query dialog */ /* For now, do nothing, as this query type isn't emitted yet */ break; case LLCACHE_QUERY_SSL: - gui_cert_verify(query->url, query->data.ssl.certs, + gui_cert_verify(nsurl_access(query->url), query->data.ssl.certs, query->data.ssl.num, cb, cbpw); break; }