mirror of
https://github.com/ostreedev/ostree.git
synced 2025-01-21 22:04:15 +03:00
libcurl backend
For rpm-ostree, we already link to libcurl indirectly via librepo, and only having one HTTP library in process makes sense. Further, libcurl is (I think) more popular in the embedded space. It also supports HTTP/2.0 today, which is a *very* nice to have for OSTree. This seems to be working fairly well for me in my local testing, but it's obviously brand new nontrivial code, so it's going to need some soak time. The ugliest part of this is having to vendor in the soup-url code. With Oxidation we could follow the path of Firefox and use the [Servo URL parser](https://github.com/servo/rust-url). Having to redo cookie parsing also sucked, and that would also be a good oxidation target. But that's for the future. Closes: #641 Approved by: jlebon
This commit is contained in:
parent
425ccc0a33
commit
361aa449fb
@ -71,5 +71,27 @@ env:
|
||||
tests:
|
||||
- make check TESTS=tests/test-rollsum
|
||||
|
||||
---
|
||||
|
||||
inherit: true
|
||||
required: true
|
||||
|
||||
context: curl
|
||||
|
||||
packages:
|
||||
- pkgconfig(libcurl)
|
||||
|
||||
build:
|
||||
config-opts: >
|
||||
--prefix=/usr
|
||||
--libdir=/usr/lib64
|
||||
--enable-installed-tests
|
||||
--enable-gtk-doc
|
||||
--with-curl
|
||||
|
||||
tests:
|
||||
- make check
|
||||
- gnome-desktop-testing-runner -p 0 ostree
|
||||
|
||||
artifacts:
|
||||
- test-suite.log
|
||||
|
@ -5,6 +5,7 @@ sudo: required
|
||||
env:
|
||||
- ci_distro=ubuntu ci_suite=trusty ci_test=no # TODO: use libcurl on this
|
||||
- ci_docker=debian:jessie-slim ci_distro=debian ci_suite=jessie
|
||||
- ci_docker=debian:jessie-slim ci_distro=debian ci_suite=jessie ci_configopts="--with-curl"
|
||||
- ci_docker=debian:stretch-slim ci_distro=debian ci_suite=stretch ci_test=no # TODO gpgme flake https://github.com/ostreedev/ostree/pull/664#issuecomment-276033383
|
||||
- ci_docker=ubuntu:xenial ci_distro=ubuntu ci_suite=xenial
|
||||
|
||||
|
@ -167,18 +167,31 @@ libostree_1_la_CFLAGS += $(LIBSYSTEMD_CFLAGS)
|
||||
libostree_1_la_LIBADD += $(LIBSYSTEMD_LIBS)
|
||||
endif
|
||||
|
||||
if USE_LIBSOUP
|
||||
if USE_CURL_OR_SOUP
|
||||
libostree_1_la_SOURCES += \
|
||||
src/libostree/ostree-fetcher.h \
|
||||
src/libostree/ostree-fetcher.c \
|
||||
src/libostree/ostree-fetcher-util.h \
|
||||
src/libostree/ostree-fetcher-util.c \
|
||||
src/libostree/ostree-fetcher-uri.c \
|
||||
src/libostree/ostree-metalink.h \
|
||||
src/libostree/ostree-metalink.c \
|
||||
$(NULL)
|
||||
endif
|
||||
|
||||
if USE_CURL
|
||||
libostree_1_la_SOURCES += src/libostree/ostree-fetcher-curl.c \
|
||||
src/libostree/ostree-soup-uri.h src/libostree/ostree-soup-uri.c \
|
||||
src/libostree/ostree-soup-form.c \
|
||||
$(NULL)
|
||||
libostree_1_la_CFLAGS += $(OT_DEP_CURL_CFLAGS)
|
||||
libostree_1_la_LIBADD += $(OT_DEP_CURL_LIBS)
|
||||
else
|
||||
if USE_LIBSOUP
|
||||
libostree_1_la_SOURCES += src/libostree/ostree-fetcher-soup.c
|
||||
libostree_1_la_CFLAGS += $(OT_INTERNAL_SOUP_CFLAGS)
|
||||
libostree_1_la_LIBADD += $(OT_INTERNAL_SOUP_LIBS)
|
||||
endif
|
||||
endif
|
||||
|
||||
if USE_LIBMOUNT
|
||||
libostree_1_la_CFLAGS += $(OT_DEP_LIBMOUNT_CFLAGS)
|
||||
|
@ -88,12 +88,14 @@ ostree_SOURCES += \
|
||||
src/ostree/ot-remote-builtin-summary.c \
|
||||
$(NULL)
|
||||
|
||||
if USE_LIBSOUP
|
||||
ostree_SOURCES += \
|
||||
src/ostree/ot-remote-builtin-add-cookie.c \
|
||||
src/ostree/ot-remote-builtin-delete-cookie.c \
|
||||
src/ostree/ot-remote-builtin-list-cookies.c \
|
||||
$(NULL)
|
||||
|
||||
if USE_CURL_OR_SOUP
|
||||
ostree_SOURCES += src/ostree/ot-remote-builtin-add-cookie.c \
|
||||
src/ostree/ot-remote-builtin-delete-cookie.c \
|
||||
src/ostree/ot-remote-builtin-list-cookies.c \
|
||||
src/ostree/ot-remote-cookie-util.h \
|
||||
src/ostree/ot-remote-cookie-util.c \
|
||||
$(NULL)
|
||||
endif
|
||||
|
||||
src/ostree/parse-datetime.c: src/ostree/parse-datetime.y Makefile
|
||||
@ -112,15 +114,23 @@ ostree_CFLAGS = $(ostree_bin_shared_cflags)
|
||||
ostree_LDADD = $(ostree_bin_shared_ldadd) libbsdiff.la libostree-kernel-args.la $(LIBSYSTEMD_LIBS)
|
||||
|
||||
|
||||
if USE_LIBSOUP
|
||||
ostree_SOURCES += src/ostree/ot-builtin-pull.c src/ostree/ot-builtin-trivial-httpd.c
|
||||
ostree_CFLAGS += $(OT_INTERNAL_SOUP_CFLAGS)
|
||||
ostree_LDADD += $(OT_INTERNAL_SOUP_LIBS)
|
||||
if USE_CURL_OR_SOUP
|
||||
ostree_SOURCES += src/ostree/ot-builtin-pull.c
|
||||
endif
|
||||
|
||||
if USE_LIBSOUP
|
||||
# Eventually once we stop things from using this, we should support disabling this
|
||||
ostree_SOURCES += src/ostree/ot-builtin-trivial-httpd.c
|
||||
pkglibexec_PROGRAMS += ostree-trivial-httpd
|
||||
ostree_trivial_httpd_SOURCES = src/ostree/ostree-trivial-httpd.c
|
||||
ostree_trivial_httpd_CFLAGS = $(ostree_bin_shared_cflags) $(OT_INTERNAL_SOUP_CFLAGS)
|
||||
ostree_trivial_httpd_LDADD = $(ostree_bin_shared_ldadd) $(OT_INTERNAL_SOUP_LIBS)
|
||||
|
||||
if !USE_CURL
|
||||
# This is necessary for the cookie jar bits
|
||||
ostree_CFLAGS += $(OT_INTERNAL_SOUP_CFLAGS)
|
||||
ostree_LDADD += $(OT_INTERNAL_SOUP_LIBS)
|
||||
endif
|
||||
endif
|
||||
|
||||
if USE_LIBARCHIVE
|
||||
|
30
configure.ac
30
configure.ac
@ -79,14 +79,29 @@ PKG_CHECK_MODULES(OT_DEP_ZLIB, zlib)
|
||||
dnl We're not actually linking to this, just using the header
|
||||
PKG_CHECK_MODULES(OT_DEP_E2P, e2p)
|
||||
|
||||
dnl Arbitrary version that's in CentOS7.2 now
|
||||
CURL_DEPENDENCY=7.29.0
|
||||
AC_ARG_WITH(curl,
|
||||
AS_HELP_STRING([--with-curl], [Use libcurl @<:@default=no@:>@]),
|
||||
[], [with_curl=no])
|
||||
AS_IF([test x$with_curl != xno ], [
|
||||
PKG_CHECK_MODULES(OT_DEP_CURL, libcurl >= $CURL_DEPENDENCY)
|
||||
with_curl=yes
|
||||
AC_DEFINE([HAVE_LIBCURL], 1, [Define if we have libcurl.pc])
|
||||
dnl Currently using libcurl requires soup for trivial-httpd for tests
|
||||
with_soup_default=yes
|
||||
], [with_soup_default=check])
|
||||
AM_CONDITIONAL(USE_CURL, test x$with_curl != xno)
|
||||
if test x$with_curl = xyes; then OSTREE_FEATURES="$OSTREE_FEATURES +libcurl"; fi
|
||||
|
||||
dnl When bumping the libsoup-2.4 dependency, remember to bump
|
||||
dnl SOUP_VERSION_MIN_REQUIRED and SOUP_VERSION_MAX_ALLOWED in
|
||||
dnl Makefile.am
|
||||
SOUP_DEPENDENCY="libsoup-2.4 >= 2.39.1"
|
||||
AC_ARG_WITH(soup,
|
||||
AS_HELP_STRING([--with-soup], [Use libsoup @<:@default=yes@:>@]),
|
||||
[], [with_soup=check])
|
||||
AS_IF([test x$with_soup != xno ], [
|
||||
[], [with_soup=$with_soup_default])
|
||||
AS_IF([test x$with_soup != xno], [
|
||||
AC_ARG_ENABLE(libsoup_client_certs,
|
||||
AS_HELP_STRING([--enable-libsoup-client-certs],
|
||||
[Require availability of new enough libsoup TLS client cert API (default: auto)]),,
|
||||
@ -120,6 +135,14 @@ if test x$with_soup != xno; then OSTREE_FEATURES="$OSTREE_FEATURES +libsoup"; fi
|
||||
AM_CONDITIONAL(USE_LIBSOUP, test x$with_soup != xno)
|
||||
AM_CONDITIONAL(HAVE_LIBSOUP_CLIENT_CERTS, test x$have_libsoup_client_certs = xyes)
|
||||
|
||||
AS_IF([test x$with_curl = xyes && test x$with_soup = xno], [
|
||||
AC_MSG_ERROR([Curl enabled, but libsoup is not; libsoup is needed for tests])
|
||||
])
|
||||
AM_CONDITIONAL(USE_CURL_OR_SOUP, test x$with_curl != xno || test x$with_soup != xno)
|
||||
AS_IF([test x$with_curl != xno || test x$with_soup != xno],
|
||||
[AC_DEFINE([HAVE_LIBCURL_OR_LIBSOUP], 1, [Define if we have soup or curl])])
|
||||
AS_IF([test x$with_curl = xyes], [fetcher_backend=curl], [test x$with_soup = xyes], [fetcher_backend=libsoup], [fetcher_backend=none])
|
||||
|
||||
m4_ifdef([GOBJECT_INTROSPECTION_CHECK], [
|
||||
GOBJECT_INTROSPECTION_CHECK([1.34.0])
|
||||
])
|
||||
@ -374,8 +397,7 @@ echo "
|
||||
introspection: $found_introspection
|
||||
Rust (internal oxidation): $rust_debug_release
|
||||
rofiles-fuse: $enable_rofiles_fuse
|
||||
libsoup (retrieve remote HTTP repositories): $with_soup
|
||||
libsoup TLS client certs: $have_libsoup_client_certs
|
||||
HTTP backend: $fetcher_backend
|
||||
SELinux: $with_selinux
|
||||
systemd: $have_libsystemd
|
||||
libmount: $with_libmount
|
||||
|
922
src/libostree/ostree-fetcher-curl.c
Normal file
922
src/libostree/ostree-fetcher-curl.c
Normal file
@ -0,0 +1,922 @@
|
||||
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
|
||||
*
|
||||
* Copyright (C) 2016 Colin Walters <walters@verbum.org>
|
||||
*
|
||||
* 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 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
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include <gio/gfiledescriptorbased.h>
|
||||
#include <gio/gunixoutputstream.h>
|
||||
#include <glib-unix.h>
|
||||
#include <curl/curl.h>
|
||||
|
||||
/* These macros came from 7.43.0, but we want to check
|
||||
* for versions a bit earlier than that (to work on CentOS 7),
|
||||
* so define them here if we're using an older version.
|
||||
*/
|
||||
#ifndef CURL_VERSION_BITS
|
||||
#define CURL_VERSION_BITS(x,y,z) ((x)<<16|(y)<<8|z)
|
||||
#endif
|
||||
#ifndef CURL_AT_LEAST_VERSION
|
||||
#define CURL_AT_LEAST_VERSION(x,y,z) (LIBCURL_VERSION_NUM >= CURL_VERSION_BITS(x, y, z))
|
||||
#endif
|
||||
|
||||
#include "ostree-fetcher.h"
|
||||
#include "ostree-enumtypes.h"
|
||||
#include "ostree-repo-private.h"
|
||||
#include "otutil.h"
|
||||
|
||||
#include "ostree-soup-uri.h"
|
||||
|
||||
typedef struct FetcherRequest FetcherRequest;
|
||||
typedef struct SockInfo SockInfo;
|
||||
|
||||
static int sock_cb (CURL *e, curl_socket_t s, int what, void *cbp, void *sockp);
|
||||
static gboolean timer_cb (gpointer data);
|
||||
static void sock_unref (SockInfo *f);
|
||||
static int update_timeout_cb (CURLM *multi, long timeout_ms, void *userp);
|
||||
static void request_unref (FetcherRequest *req);
|
||||
static void initiate_next_curl_request (FetcherRequest *req, GTask *task);
|
||||
static void destroy_and_unref_source (GSource *source);
|
||||
|
||||
struct OstreeFetcher
|
||||
{
|
||||
GObject parent_instance;
|
||||
|
||||
OstreeFetcherConfigFlags config_flags;
|
||||
char *tls_ca_db_path;
|
||||
char *tls_client_cert_path;
|
||||
char *tls_client_key_path;
|
||||
char *cookie_jar_path;
|
||||
char *proxy;
|
||||
struct curl_slist *extra_headers;
|
||||
int tmpdir_dfd;
|
||||
|
||||
GMainContext *mainctx;
|
||||
CURLM *multi;
|
||||
GSource *timer_event;
|
||||
int curl_running;
|
||||
GHashTable *outstanding_requests; /* Set<GTask> */
|
||||
GHashTable *sockets; /* Set<SockInfo> */
|
||||
|
||||
guint64 bytes_transferred;
|
||||
};
|
||||
|
||||
/* Information associated with a request */
|
||||
struct FetcherRequest {
|
||||
guint refcount;
|
||||
GPtrArray *mirrorlist;
|
||||
guint idx;
|
||||
|
||||
char *filename;
|
||||
guint64 current_size;
|
||||
guint64 max_size;
|
||||
OstreeFetcherRequestFlags flags;
|
||||
gboolean is_membuf;
|
||||
GError *caught_write_error;
|
||||
char *out_tmpfile;
|
||||
int out_tmpfile_fd;
|
||||
GString *output_buf;
|
||||
|
||||
CURL *easy;
|
||||
char error[CURL_ERROR_SIZE];
|
||||
|
||||
OstreeFetcher *fetcher;
|
||||
};
|
||||
|
||||
/* Information associated with a specific socket */
|
||||
struct SockInfo {
|
||||
guint refcount;
|
||||
curl_socket_t sockfd;
|
||||
int action;
|
||||
long timeout;
|
||||
GSource *ch;
|
||||
OstreeFetcher *fetcher;
|
||||
};
|
||||
|
||||
enum {
|
||||
PROP_0,
|
||||
PROP_CONFIG_FLAGS
|
||||
};
|
||||
|
||||
G_DEFINE_TYPE (OstreeFetcher, _ostree_fetcher, G_TYPE_OBJECT)
|
||||
|
||||
static void
|
||||
_ostree_fetcher_set_property (GObject *object,
|
||||
guint prop_id,
|
||||
const GValue *value,
|
||||
GParamSpec *pspec)
|
||||
{
|
||||
OstreeFetcher *self = OSTREE_FETCHER (object);
|
||||
|
||||
switch (prop_id)
|
||||
{
|
||||
case PROP_CONFIG_FLAGS:
|
||||
self->config_flags = g_value_get_flags (value);
|
||||
break;
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
_ostree_fetcher_get_property (GObject *object,
|
||||
guint prop_id,
|
||||
GValue *value,
|
||||
GParamSpec *pspec)
|
||||
{
|
||||
OstreeFetcher *self = OSTREE_FETCHER (object);
|
||||
|
||||
switch (prop_id)
|
||||
{
|
||||
case PROP_CONFIG_FLAGS:
|
||||
g_value_set_flags (value, self->config_flags);
|
||||
break;
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
_ostree_fetcher_finalize (GObject *object)
|
||||
{
|
||||
OstreeFetcher *self = OSTREE_FETCHER (object);
|
||||
|
||||
g_free (self->cookie_jar_path);
|
||||
g_free (self->proxy);
|
||||
g_assert_cmpint (g_hash_table_size (self->outstanding_requests), ==, 0);
|
||||
g_clear_pointer (&self->extra_headers, (GDestroyNotify)curl_slist_free_all);
|
||||
g_hash_table_unref (self->outstanding_requests);
|
||||
g_hash_table_unref (self->sockets);
|
||||
g_clear_pointer (&self->timer_event, (GDestroyNotify)destroy_and_unref_source);
|
||||
if (self->mainctx)
|
||||
g_main_context_unref (self->mainctx);
|
||||
curl_multi_cleanup (self->multi);
|
||||
|
||||
G_OBJECT_CLASS (_ostree_fetcher_parent_class)->finalize (object);
|
||||
}
|
||||
|
||||
static void
|
||||
_ostree_fetcher_constructed (GObject *object)
|
||||
{
|
||||
// OstreeFetcher *self = OSTREE_FETCHER (object);
|
||||
|
||||
G_OBJECT_CLASS (_ostree_fetcher_parent_class)->constructed (object);
|
||||
}
|
||||
|
||||
static void
|
||||
_ostree_fetcher_class_init (OstreeFetcherClass *klass)
|
||||
{
|
||||
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
||||
|
||||
gobject_class->set_property = _ostree_fetcher_set_property;
|
||||
gobject_class->get_property = _ostree_fetcher_get_property;
|
||||
gobject_class->finalize = _ostree_fetcher_finalize;
|
||||
gobject_class->constructed = _ostree_fetcher_constructed;
|
||||
|
||||
g_object_class_install_property (gobject_class,
|
||||
PROP_CONFIG_FLAGS,
|
||||
g_param_spec_flags ("config-flags",
|
||||
"",
|
||||
"",
|
||||
OSTREE_TYPE_FETCHER_CONFIG_FLAGS,
|
||||
OSTREE_FETCHER_FLAGS_NONE,
|
||||
G_PARAM_READWRITE |
|
||||
G_PARAM_CONSTRUCT_ONLY |
|
||||
G_PARAM_STATIC_STRINGS));
|
||||
}
|
||||
|
||||
static void
|
||||
_ostree_fetcher_init (OstreeFetcher *self)
|
||||
{
|
||||
self->multi = curl_multi_init();
|
||||
self->outstanding_requests = g_hash_table_new_full (NULL, NULL, (GDestroyNotify)g_object_unref, NULL);
|
||||
self->sockets = g_hash_table_new_full (NULL, NULL, (GDestroyNotify)sock_unref, NULL);
|
||||
curl_multi_setopt (self->multi, CURLMOPT_SOCKETFUNCTION, sock_cb);
|
||||
curl_multi_setopt (self->multi, CURLMOPT_SOCKETDATA, self);
|
||||
curl_multi_setopt (self->multi, CURLMOPT_TIMERFUNCTION, update_timeout_cb);
|
||||
curl_multi_setopt (self->multi, CURLMOPT_TIMERDATA, self);
|
||||
#if CURL_AT_LEAST_VERSION(7, 30, 0)
|
||||
/* Let's do something reasonable here. */
|
||||
curl_multi_setopt (self->multi, CURLMOPT_MAX_TOTAL_CONNECTIONS, 8);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
OstreeFetcher *
|
||||
_ostree_fetcher_new (int tmpdir_dfd,
|
||||
OstreeFetcherConfigFlags flags)
|
||||
{
|
||||
OstreeFetcher *fetcher = g_object_new (OSTREE_TYPE_FETCHER, "config-flags", flags, NULL);
|
||||
fetcher->tmpdir_dfd = tmpdir_dfd;
|
||||
return fetcher;
|
||||
}
|
||||
|
||||
static void
|
||||
destroy_and_unref_source (GSource *source)
|
||||
{
|
||||
g_source_destroy (source);
|
||||
g_source_unref (source);
|
||||
}
|
||||
|
||||
static char *
|
||||
request_get_uri (FetcherRequest *req, guint idx)
|
||||
{
|
||||
SoupURI *baseuri = req->mirrorlist->pdata[idx];
|
||||
if (!req->filename)
|
||||
return soup_uri_to_string (baseuri, FALSE);
|
||||
{ g_autofree char *uristr = soup_uri_to_string (baseuri, FALSE);
|
||||
return g_build_filename (uristr, req->filename, NULL);
|
||||
}
|
||||
}
|
||||
|
||||
static gboolean
|
||||
ensure_tmpfile (FetcherRequest *req, GError **error)
|
||||
{
|
||||
if (req->out_tmpfile_fd == -1)
|
||||
{
|
||||
if (!glnx_open_tmpfile_linkable_at (req->fetcher->tmpdir_dfd, ".",
|
||||
O_WRONLY, &req->out_tmpfile_fd,
|
||||
&req->out_tmpfile,
|
||||
error))
|
||||
return FALSE;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
/* Check for completed transfers, and remove their easy handles */
|
||||
static void
|
||||
check_multi_info (OstreeFetcher *fetcher)
|
||||
{
|
||||
CURLMsg *msg;
|
||||
int msgs_left;
|
||||
|
||||
while ((msg = curl_multi_info_read (fetcher->multi, &msgs_left)) != NULL)
|
||||
{
|
||||
long response;
|
||||
CURL *easy = msg->easy_handle;
|
||||
CURLcode curlres = msg->data.result;
|
||||
GTask *task;
|
||||
FetcherRequest *req;
|
||||
const char *eff_url;
|
||||
gboolean is_file;
|
||||
gboolean continued_request = FALSE;
|
||||
|
||||
if (msg->msg != CURLMSG_DONE)
|
||||
continue;
|
||||
|
||||
curl_easy_getinfo (easy, CURLINFO_PRIVATE, &task);
|
||||
curl_easy_getinfo (easy, CURLINFO_EFFECTIVE_URL, &eff_url);
|
||||
/* We should have limited the protocols; this is what
|
||||
* curl's tool_operate.c does.
|
||||
*/
|
||||
is_file = g_str_has_prefix (eff_url, "file:");
|
||||
g_assert (is_file || g_str_has_prefix (eff_url, "http"));
|
||||
|
||||
req = g_task_get_task_data (task);
|
||||
|
||||
if (req->caught_write_error)
|
||||
g_task_return_error (task, g_steal_pointer (&req->caught_write_error));
|
||||
else if (curlres != CURLE_OK)
|
||||
{
|
||||
if (is_file && curlres == CURLE_FILE_COULDNT_READ_FILE)
|
||||
{
|
||||
/* Handle file not found */
|
||||
g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_NOT_FOUND,
|
||||
"%s",
|
||||
curl_easy_strerror (curlres));
|
||||
}
|
||||
else
|
||||
g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_FAILED, "[%u] %s",
|
||||
curlres,
|
||||
curl_easy_strerror (curlres));
|
||||
}
|
||||
else
|
||||
{
|
||||
curl_easy_getinfo (easy, CURLINFO_RESPONSE_CODE, &response);
|
||||
if (!is_file && !(response >= 200 && response < 300))
|
||||
{
|
||||
GIOErrorEnum giocode;
|
||||
|
||||
/* TODO - share with soup */
|
||||
switch (response)
|
||||
{
|
||||
case 404:
|
||||
case 403:
|
||||
case 410:
|
||||
giocode = G_IO_ERROR_NOT_FOUND;
|
||||
break;
|
||||
default:
|
||||
giocode = G_IO_ERROR_FAILED;
|
||||
}
|
||||
|
||||
if (req->idx + 1 == req->mirrorlist->len)
|
||||
{
|
||||
g_task_return_new_error (task, G_IO_ERROR, giocode,
|
||||
"Server returned HTTP %lu", response);
|
||||
}
|
||||
else
|
||||
{
|
||||
continued_request = TRUE;
|
||||
}
|
||||
}
|
||||
else if (req->is_membuf)
|
||||
{
|
||||
GBytes *ret;
|
||||
if ((req->flags & OSTREE_FETCHER_REQUEST_NUL_TERMINATION) > 0)
|
||||
g_string_append_c (req->output_buf, '\0');
|
||||
ret = g_string_free_to_bytes (req->output_buf);
|
||||
req->output_buf = NULL;
|
||||
g_task_return_pointer (task, ret, (GDestroyNotify)g_bytes_unref);
|
||||
}
|
||||
else
|
||||
{
|
||||
g_autoptr(GError) local_error = NULL;
|
||||
GError **error = &local_error;
|
||||
|
||||
/* TODO - share file naming with soup, and fix it */
|
||||
g_autofree char *tmpfile_path =
|
||||
g_compute_checksum_for_string (G_CHECKSUM_SHA256,
|
||||
eff_url, strlen (eff_url));
|
||||
if (!ensure_tmpfile (req, error))
|
||||
{
|
||||
g_task_return_error (task, g_steal_pointer (&local_error));
|
||||
}
|
||||
else if (fchmod (req->out_tmpfile_fd, 0644) < 0)
|
||||
{
|
||||
glnx_set_error_from_errno (error);
|
||||
g_task_return_error (task, g_steal_pointer (&local_error));
|
||||
}
|
||||
else if (!glnx_link_tmpfile_at (fetcher->tmpdir_dfd,
|
||||
GLNX_LINK_TMPFILE_REPLACE,
|
||||
req->out_tmpfile_fd,
|
||||
req->out_tmpfile,
|
||||
fetcher->tmpdir_dfd,
|
||||
tmpfile_path,
|
||||
error))
|
||||
g_task_return_error (task, g_steal_pointer (&local_error));
|
||||
else
|
||||
{
|
||||
g_task_return_pointer (task, g_steal_pointer (&tmpfile_path), g_free);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
curl_multi_remove_handle (fetcher->multi, easy);
|
||||
if (continued_request)
|
||||
{
|
||||
req->idx++;
|
||||
initiate_next_curl_request (req, task);
|
||||
}
|
||||
else
|
||||
{
|
||||
g_hash_table_remove (fetcher->outstanding_requests, task);
|
||||
if (g_hash_table_size (fetcher->outstanding_requests) == 0)
|
||||
{
|
||||
g_clear_pointer (&fetcher->mainctx, g_main_context_unref);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* Called by glib when our timeout expires */
|
||||
static gboolean
|
||||
timer_cb (gpointer data)
|
||||
{
|
||||
OstreeFetcher *fetcher = data;
|
||||
CURLMcode rc;
|
||||
|
||||
fetcher->timer_event = NULL;
|
||||
rc = curl_multi_socket_action (fetcher->multi, CURL_SOCKET_TIMEOUT, 0, &fetcher->curl_running);
|
||||
g_assert (rc == CURLM_OK);
|
||||
check_multi_info (fetcher);
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/* Update the event timer after curl_multi library calls */
|
||||
static int
|
||||
update_timeout_cb (CURLM *multi, long timeout_ms, void *userp)
|
||||
{
|
||||
OstreeFetcher *fetcher = userp;
|
||||
|
||||
g_clear_pointer (&fetcher->timer_event, (GDestroyNotify)destroy_and_unref_source);
|
||||
|
||||
if (timeout_ms != -1)
|
||||
{
|
||||
fetcher->timer_event = g_timeout_source_new (timeout_ms);
|
||||
g_source_set_callback (fetcher->timer_event, timer_cb, fetcher, NULL);
|
||||
g_source_attach (fetcher->timer_event, fetcher->mainctx);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Called by glib when we get action on a multi socket */
|
||||
static gboolean
|
||||
event_cb (int fd, GIOCondition condition, gpointer data)
|
||||
{
|
||||
OstreeFetcher *fetcher = data;
|
||||
CURLMcode rc;
|
||||
|
||||
int action =
|
||||
(condition & G_IO_IN ? CURL_CSELECT_IN : 0) |
|
||||
(condition & G_IO_OUT ? CURL_CSELECT_OUT : 0);
|
||||
|
||||
rc = curl_multi_socket_action (fetcher->multi, fd, action, &fetcher->curl_running);
|
||||
g_assert (rc == CURLM_OK);
|
||||
|
||||
check_multi_info (fetcher);
|
||||
if (fetcher->curl_running > 0)
|
||||
{
|
||||
return TRUE;
|
||||
}
|
||||
else
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
/* Clean up the SockInfo structure */
|
||||
static void
|
||||
sock_unref (SockInfo *f)
|
||||
{
|
||||
if (!f)
|
||||
return;
|
||||
if (--f->refcount != 0)
|
||||
return;
|
||||
g_clear_pointer (&f->ch, (GDestroyNotify)destroy_and_unref_source);
|
||||
g_free (f);
|
||||
}
|
||||
|
||||
/* Assign information to a SockInfo structure */
|
||||
static void
|
||||
setsock (SockInfo*f, curl_socket_t s, int act, OstreeFetcher *fetcher)
|
||||
{
|
||||
GIOCondition kind =
|
||||
(act&CURL_POLL_IN?G_IO_IN:0)|(act&CURL_POLL_OUT?G_IO_OUT:0);
|
||||
|
||||
f->sockfd = s;
|
||||
f->action = act;
|
||||
g_clear_pointer (&f->ch, (GDestroyNotify)destroy_and_unref_source);
|
||||
/* TODO - investigate new g_source_modify_unix_fd() so changing the poll
|
||||
* flags involves less allocation.
|
||||
*/
|
||||
f->ch = g_unix_fd_source_new (f->sockfd, kind);
|
||||
g_source_set_callback (f->ch, (GSourceFunc) event_cb, fetcher, NULL);
|
||||
g_source_attach (f->ch, fetcher->mainctx);
|
||||
}
|
||||
|
||||
/* Initialize a new SockInfo structure */
|
||||
static void
|
||||
addsock (curl_socket_t s, CURL *easy, int action, OstreeFetcher *fetcher)
|
||||
{
|
||||
SockInfo *fdp = g_new0 (SockInfo, 1);
|
||||
|
||||
fdp->refcount = 1;
|
||||
fdp->fetcher = fetcher;
|
||||
setsock (fdp, s, action, fetcher);
|
||||
curl_multi_assign (fetcher->multi, s, fdp);
|
||||
g_hash_table_add (fetcher->sockets, fdp);
|
||||
}
|
||||
|
||||
/* CURLMOPT_SOCKETFUNCTION */
|
||||
static int
|
||||
sock_cb (CURL *easy, curl_socket_t s, int what, void *cbp, void *sockp)
|
||||
{
|
||||
OstreeFetcher *fetcher = cbp;
|
||||
SockInfo *fdp = (SockInfo*) sockp;
|
||||
|
||||
if (what == CURL_POLL_REMOVE)
|
||||
{
|
||||
if (!g_hash_table_remove (fetcher->sockets, fdp))
|
||||
g_assert_not_reached ();
|
||||
}
|
||||
else
|
||||
{
|
||||
if (!fdp)
|
||||
{
|
||||
addsock (s, easy, what, fetcher);
|
||||
}
|
||||
else
|
||||
{
|
||||
setsock (fdp, s, what, fetcher);
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* CURLOPT_WRITEFUNCTION */
|
||||
static size_t
|
||||
write_cb (void *ptr, size_t size, size_t nmemb, void *data)
|
||||
{
|
||||
const size_t realsize = size * nmemb;
|
||||
GTask *task = data;
|
||||
FetcherRequest *req;
|
||||
|
||||
req = g_task_get_task_data (task);
|
||||
|
||||
if (req->caught_write_error)
|
||||
return -1;
|
||||
|
||||
if (req->max_size > 0)
|
||||
{
|
||||
if (realsize > req->max_size ||
|
||||
(realsize + req->current_size) > req->max_size)
|
||||
{
|
||||
const char *eff_url;
|
||||
curl_easy_getinfo (req->easy, CURLINFO_EFFECTIVE_URL, &eff_url);
|
||||
req->caught_write_error = g_error_new (G_IO_ERROR, G_IO_ERROR_FAILED,
|
||||
"URI %s exceeded maximum size of %" G_GUINT64_FORMAT " bytes",
|
||||
eff_url, req->max_size);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
if (req->is_membuf)
|
||||
g_string_append_len (req->output_buf, ptr, realsize);
|
||||
else
|
||||
{
|
||||
if (!ensure_tmpfile (req, &req->caught_write_error))
|
||||
return -1;
|
||||
g_assert (req->out_tmpfile_fd >= 0);
|
||||
if (glnx_loop_write (req->out_tmpfile_fd, ptr, realsize) < 0)
|
||||
{
|
||||
glnx_set_error_from_errno (&req->caught_write_error);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
req->current_size += realsize;
|
||||
req->fetcher->bytes_transferred += realsize;
|
||||
|
||||
return realsize;
|
||||
}
|
||||
|
||||
/* CURLOPT_PROGRESSFUNCTION */
|
||||
static int
|
||||
prog_cb (void *p, double dltotal, double dlnow, double ult, double uln)
|
||||
{
|
||||
GTask *task = p;
|
||||
FetcherRequest *req;
|
||||
char *eff_url;
|
||||
req = g_task_get_task_data (task);
|
||||
curl_easy_getinfo (req->easy, CURLINFO_EFFECTIVE_URL, &eff_url);
|
||||
g_printerr ("Progress: %s (%g/%g)\n", eff_url, dlnow, dltotal);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
request_unref (FetcherRequest *req)
|
||||
{
|
||||
if (--req->refcount != 0)
|
||||
return;
|
||||
|
||||
g_ptr_array_unref (req->mirrorlist);
|
||||
g_free (req->filename);
|
||||
g_clear_error (&req->caught_write_error);
|
||||
if (req->out_tmpfile_fd != -1)
|
||||
(void) close (req->out_tmpfile_fd);
|
||||
g_free (req->out_tmpfile);
|
||||
if (req->output_buf)
|
||||
g_string_free (req->output_buf, TRUE);
|
||||
curl_easy_cleanup (req->easy);
|
||||
|
||||
g_free (req);
|
||||
}
|
||||
|
||||
int
|
||||
_ostree_fetcher_get_dfd (OstreeFetcher *fetcher)
|
||||
{
|
||||
return fetcher->tmpdir_dfd;
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_set_proxy (OstreeFetcher *self,
|
||||
const char *http_proxy)
|
||||
{
|
||||
g_free (self->proxy);
|
||||
self->proxy = g_strdup (http_proxy);
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_set_cookie_jar (OstreeFetcher *self,
|
||||
const char *jar_path)
|
||||
{
|
||||
g_free (self->cookie_jar_path);
|
||||
self->cookie_jar_path = g_strdup (jar_path);
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_set_client_cert (OstreeFetcher *self,
|
||||
const char *cert_path,
|
||||
const char *key_path)
|
||||
{
|
||||
g_assert ((cert_path == NULL && key_path == NULL)
|
||||
|| (cert_path != NULL && key_path != NULL));
|
||||
g_free (self->tls_client_cert_path);
|
||||
self->tls_client_cert_path = g_strdup (cert_path);
|
||||
g_free (self->tls_client_key_path);
|
||||
self->tls_client_key_path = g_strdup (key_path);
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_set_tls_database (OstreeFetcher *self,
|
||||
const char *dbpath)
|
||||
{
|
||||
g_free (self->tls_ca_db_path);
|
||||
self->tls_ca_db_path = g_strdup (dbpath);
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_set_extra_headers (OstreeFetcher *self,
|
||||
GVariant *extra_headers)
|
||||
{
|
||||
GVariantIter viter;
|
||||
const char *key;
|
||||
const char *value;
|
||||
|
||||
g_clear_pointer (&self->extra_headers, (GDestroyNotify)curl_slist_free_all);
|
||||
|
||||
g_variant_iter_init (&viter, extra_headers);
|
||||
while (g_variant_iter_loop (&viter, "(&s&s)", &key, &value))
|
||||
{
|
||||
g_autofree char *header = g_strdup_printf ("%s: %s", key, value);
|
||||
self->extra_headers = curl_slist_append (self->extra_headers, header);
|
||||
}
|
||||
}
|
||||
|
||||
/* Re-bind all of the outstanding curl items to our new main context */
|
||||
static void
|
||||
adopt_steal_mainctx (OstreeFetcher *self,
|
||||
GMainContext *mainctx)
|
||||
{
|
||||
GHashTableIter hiter;
|
||||
gpointer key, value;
|
||||
|
||||
g_assert (self->mainctx == NULL);
|
||||
self->mainctx = mainctx; /* Transfer */
|
||||
|
||||
if (self->timer_event != NULL)
|
||||
{
|
||||
guint64 readytime = g_source_get_ready_time (self->timer_event);
|
||||
guint64 curtime = g_source_get_time (self->timer_event);
|
||||
guint64 timeout_micros = curtime - readytime;
|
||||
if (timeout_micros < 0)
|
||||
timeout_micros = 0;
|
||||
update_timeout_cb (self->multi, timeout_micros / 1000, self);
|
||||
}
|
||||
|
||||
g_hash_table_iter_init (&hiter, self->sockets);
|
||||
while (g_hash_table_iter_next (&hiter, &key, &value))
|
||||
{
|
||||
SockInfo *fdp = key;
|
||||
setsock (fdp, fdp->sockfd, fdp->action, self);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
initiate_next_curl_request (FetcherRequest *req,
|
||||
GTask *task)
|
||||
{
|
||||
CURLMcode rc;
|
||||
OstreeFetcher *self = req->fetcher;
|
||||
|
||||
req->easy = curl_easy_init ();
|
||||
g_assert (req->easy);
|
||||
|
||||
g_assert_cmpint (req->idx, <, req->mirrorlist->len);
|
||||
|
||||
{ g_autofree char *uri = request_get_uri (req, req->idx);
|
||||
curl_easy_setopt (req->easy, CURLOPT_URL, uri);
|
||||
}
|
||||
|
||||
curl_easy_setopt (req->easy, CURLOPT_USERAGENT, "ostree ");
|
||||
if (self->extra_headers)
|
||||
curl_easy_setopt (req->easy, CURLOPT_HTTPHEADER, self->extra_headers);
|
||||
|
||||
if (self->cookie_jar_path)
|
||||
{
|
||||
rc = curl_easy_setopt (req->easy, CURLOPT_COOKIEFILE, self->cookie_jar_path);
|
||||
g_assert_cmpint (rc, ==, CURLM_OK);
|
||||
rc = curl_easy_setopt (req->easy, CURLOPT_COOKIELIST, "RELOAD");
|
||||
g_assert_cmpint (rc, ==, CURLM_OK);
|
||||
}
|
||||
|
||||
if (self->proxy)
|
||||
{
|
||||
rc = curl_easy_setopt (req->easy, CURLOPT_PROXY, self->proxy);
|
||||
g_assert_cmpint (rc, ==, CURLM_OK);
|
||||
}
|
||||
|
||||
if (self->tls_ca_db_path)
|
||||
curl_easy_setopt (req->easy, CURLOPT_CAINFO, self->tls_ca_db_path);
|
||||
|
||||
if ((self->config_flags & OSTREE_FETCHER_FLAGS_TLS_PERMISSIVE) > 0)
|
||||
curl_easy_setopt (req->easy, CURLOPT_SSL_VERIFYPEER, 0L);
|
||||
|
||||
if (self->tls_client_cert_path)
|
||||
{
|
||||
curl_easy_setopt (req->easy, CURLOPT_SSLCERT, self->tls_client_cert_path);
|
||||
curl_easy_setopt (req->easy, CURLOPT_SSLKEY, self->tls_client_key_path);
|
||||
}
|
||||
|
||||
/* We should only speak HTTP; TODO: only enable file if specified */
|
||||
curl_easy_setopt (req->easy, CURLOPT_PROTOCOLS, (long)(CURLPROTO_HTTP | CURLPROTO_HTTPS | CURLPROTO_FILE));
|
||||
/* Picked the current version in F25 as of 20170127, since
|
||||
* there are numerous HTTP/2 fixes since the original version in
|
||||
* libcurl 7.43.0.
|
||||
*/
|
||||
#if CURL_AT_LEAST_VERSION(7, 51, 0)
|
||||
curl_easy_setopt (req->easy, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_2_0);
|
||||
#endif
|
||||
curl_easy_setopt (req->easy, CURLOPT_WRITEFUNCTION, write_cb);
|
||||
if (g_getenv ("OSTREE_DEBUG_HTTP"))
|
||||
curl_easy_setopt (req->easy, CURLOPT_VERBOSE, 1L);
|
||||
curl_easy_setopt (req->easy, CURLOPT_ERRORBUFFER, req->error);
|
||||
/* Note that the "easy" object's privdata is the task */
|
||||
curl_easy_setopt (req->easy, CURLOPT_NOPROGRESS, 1L);
|
||||
curl_easy_setopt (req->easy, CURLOPT_PROGRESSFUNCTION, prog_cb);
|
||||
curl_easy_setopt (req->easy, CURLOPT_FOLLOWLOCATION, 1L);
|
||||
curl_easy_setopt (req->easy, CURLOPT_CONNECTTIMEOUT, 30L);
|
||||
curl_easy_setopt (req->easy, CURLOPT_LOW_SPEED_LIMIT, 1L);
|
||||
curl_easy_setopt (req->easy, CURLOPT_LOW_SPEED_TIME, 30L);
|
||||
|
||||
/* closure bindings -> task */
|
||||
curl_easy_setopt (req->easy, CURLOPT_PRIVATE, task);
|
||||
curl_easy_setopt (req->easy, CURLOPT_WRITEDATA, task);
|
||||
curl_easy_setopt (req->easy, CURLOPT_PROGRESSDATA, task);
|
||||
|
||||
rc = curl_multi_add_handle (self->multi, req->easy);
|
||||
g_assert (rc == CURLM_OK);
|
||||
}
|
||||
|
||||
static void
|
||||
_ostree_fetcher_request_async (OstreeFetcher *self,
|
||||
GPtrArray *mirrorlist,
|
||||
const char *filename,
|
||||
OstreeFetcherRequestFlags flags,
|
||||
gboolean is_membuf,
|
||||
guint64 max_size,
|
||||
int priority,
|
||||
GCancellable *cancellable,
|
||||
GAsyncReadyCallback callback,
|
||||
gpointer user_data)
|
||||
{
|
||||
g_autoptr(GTask) task = NULL;
|
||||
FetcherRequest *req;
|
||||
g_autoptr(GMainContext) mainctx = g_main_context_ref_thread_default ();
|
||||
|
||||
/* We don't support multiple concurrent main contexts; take
|
||||
* a ref to the first one, and require that later invocations
|
||||
* share it.
|
||||
*/
|
||||
if (g_hash_table_size (self->outstanding_requests) == 0
|
||||
&& mainctx != self->mainctx)
|
||||
{
|
||||
adopt_steal_mainctx (self, g_steal_pointer (&mainctx));
|
||||
}
|
||||
else
|
||||
{
|
||||
g_assert (self->mainctx == mainctx);
|
||||
}
|
||||
|
||||
req = g_new0 (FetcherRequest, 1);
|
||||
req->refcount = 1;
|
||||
req->error[0]='\0';
|
||||
req->fetcher = self;
|
||||
req->mirrorlist = g_ptr_array_ref (mirrorlist);
|
||||
req->filename = g_strdup (filename);
|
||||
req->max_size = max_size;
|
||||
req->flags = flags;
|
||||
req->is_membuf = is_membuf;
|
||||
/* We'll allocate the tmpfile on demand, so we handle
|
||||
* file I/O errors just in the write func.
|
||||
*/
|
||||
req->out_tmpfile_fd = -1;
|
||||
if (req->is_membuf)
|
||||
req->output_buf = g_string_new ("");
|
||||
|
||||
task = g_task_new (self, cancellable, callback, user_data);
|
||||
/* We'll use the GTask priority for our own priority queue. */
|
||||
g_task_set_priority (task, priority);
|
||||
g_task_set_source_tag (task, _ostree_fetcher_request_async);
|
||||
g_task_set_task_data (task, req, (GDestroyNotify) request_unref);
|
||||
|
||||
initiate_next_curl_request (req, task);
|
||||
|
||||
g_hash_table_add (self->outstanding_requests, g_steal_pointer (&task));
|
||||
|
||||
/* Sanity check, I added * 2 just so we don't abort if something odd happens,
|
||||
* but we do want to abort if we're asked to do obviously too many requests.
|
||||
*/
|
||||
g_assert_cmpint (g_hash_table_size (self->outstanding_requests), <,
|
||||
_OSTREE_MAX_OUTSTANDING_FETCHER_REQUESTS * 2);
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_request_to_tmpfile (OstreeFetcher *self,
|
||||
GPtrArray *mirrorlist,
|
||||
const char *filename,
|
||||
guint64 max_size,
|
||||
int priority,
|
||||
GCancellable *cancellable,
|
||||
GAsyncReadyCallback callback,
|
||||
gpointer user_data)
|
||||
{
|
||||
_ostree_fetcher_request_async (self, mirrorlist, filename, 0, FALSE,
|
||||
max_size, priority, cancellable,
|
||||
callback, user_data);
|
||||
}
|
||||
|
||||
gboolean
|
||||
_ostree_fetcher_request_to_tmpfile_finish (OstreeFetcher *self,
|
||||
GAsyncResult *result,
|
||||
char **out_filename,
|
||||
GError **error)
|
||||
{
|
||||
GTask *task;
|
||||
FetcherRequest *req;
|
||||
gpointer ret;
|
||||
|
||||
g_return_val_if_fail (g_task_is_valid (result, self), FALSE);
|
||||
g_return_val_if_fail (g_async_result_is_tagged (result, _ostree_fetcher_request_async), FALSE);
|
||||
|
||||
task = (GTask*)result;
|
||||
req = g_task_get_task_data (task);
|
||||
|
||||
ret = g_task_propagate_pointer (task, error);
|
||||
if (!ret)
|
||||
return FALSE;
|
||||
|
||||
g_assert (!req->is_membuf);
|
||||
g_assert (out_filename);
|
||||
*out_filename = ret;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_request_to_membuf (OstreeFetcher *self,
|
||||
GPtrArray *mirrorlist,
|
||||
const char *filename,
|
||||
OstreeFetcherRequestFlags flags,
|
||||
guint64 max_size,
|
||||
int priority,
|
||||
GCancellable *cancellable,
|
||||
GAsyncReadyCallback callback,
|
||||
gpointer user_data)
|
||||
{
|
||||
_ostree_fetcher_request_async (self, mirrorlist, filename, flags, TRUE,
|
||||
max_size, priority, cancellable,
|
||||
callback, user_data);
|
||||
}
|
||||
|
||||
gboolean
|
||||
_ostree_fetcher_request_to_membuf_finish (OstreeFetcher *self,
|
||||
GAsyncResult *result,
|
||||
GBytes **out_buf,
|
||||
GError **error)
|
||||
{
|
||||
GTask *task;
|
||||
FetcherRequest *req;
|
||||
gpointer ret;
|
||||
|
||||
g_return_val_if_fail (g_task_is_valid (result, self), FALSE);
|
||||
g_return_val_if_fail (g_async_result_is_tagged (result, _ostree_fetcher_request_async), FALSE);
|
||||
|
||||
task = (GTask*)result;
|
||||
req = g_task_get_task_data (task);
|
||||
|
||||
ret = g_task_propagate_pointer (task, error);
|
||||
if (!ret)
|
||||
return FALSE;
|
||||
|
||||
g_assert (req->is_membuf);
|
||||
g_assert (out_buf);
|
||||
*out_buf = ret;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
guint64
|
||||
_ostree_fetcher_bytes_transferred (OstreeFetcher *self)
|
||||
{
|
||||
return self->bytes_transferred;
|
||||
}
|
@ -1332,84 +1332,3 @@ _ostree_fetcher_bytes_transferred (OstreeFetcher *self)
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
void
|
||||
_ostree_fetcher_uri_free (OstreeFetcherURI *uri)
|
||||
{
|
||||
if (uri)
|
||||
soup_uri_free ((SoupURI*)uri);
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_parse (const char *str,
|
||||
GError **error)
|
||||
{
|
||||
SoupURI *soupuri = soup_uri_new (str);
|
||||
if (soupuri == NULL)
|
||||
{
|
||||
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
||||
"Failed to parse uri: %s", str);
|
||||
return NULL;
|
||||
}
|
||||
return (OstreeFetcherURI*)soupuri;
|
||||
}
|
||||
|
||||
static OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_new_path_internal (OstreeFetcherURI *uri,
|
||||
gboolean extend,
|
||||
const char *path)
|
||||
{
|
||||
SoupURI *newuri = soup_uri_copy ((SoupURI*)uri);
|
||||
if (path)
|
||||
{
|
||||
if (extend)
|
||||
{
|
||||
const char *origpath = soup_uri_get_path ((SoupURI*)uri);
|
||||
g_autofree char *newpath = g_build_filename (origpath, path, NULL);
|
||||
soup_uri_set_path (newuri, newpath);
|
||||
}
|
||||
else
|
||||
{
|
||||
soup_uri_set_path (newuri, path);
|
||||
}
|
||||
}
|
||||
return (OstreeFetcherURI*)newuri;
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_new_path (OstreeFetcherURI *uri,
|
||||
const char *path)
|
||||
{
|
||||
return _ostree_fetcher_uri_new_path_internal (uri, FALSE, path);
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_new_subpath (OstreeFetcherURI *uri,
|
||||
const char *subpath)
|
||||
{
|
||||
return _ostree_fetcher_uri_new_path_internal (uri, TRUE, subpath);
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_clone (OstreeFetcherURI *uri)
|
||||
{
|
||||
return _ostree_fetcher_uri_new_subpath (uri, NULL);
|
||||
}
|
||||
|
||||
char *
|
||||
_ostree_fetcher_uri_get_scheme (OstreeFetcherURI *uri)
|
||||
{
|
||||
return g_strdup (soup_uri_get_scheme ((SoupURI*)uri));
|
||||
}
|
||||
|
||||
char *
|
||||
_ostree_fetcher_uri_get_path (OstreeFetcherURI *uri)
|
||||
{
|
||||
return g_strdup (soup_uri_get_path ((SoupURI*)uri));
|
||||
}
|
||||
|
||||
char *
|
||||
_ostree_fetcher_uri_to_string (OstreeFetcherURI *uri)
|
||||
{
|
||||
return soup_uri_to_string ((SoupURI*)uri, FALSE);
|
||||
}
|
118
src/libostree/ostree-fetcher-uri.c
Normal file
118
src/libostree/ostree-fetcher-uri.c
Normal file
@ -0,0 +1,118 @@
|
||||
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
|
||||
*
|
||||
* Copyright (C) 2011,2017 Colin Walters <walters@verbum.org>
|
||||
*
|
||||
* 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 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
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*
|
||||
* Author: Colin Walters <walters@verbum.org>
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
|
||||
|
||||
#ifdef HAVE_LIBCURL
|
||||
#include "ostree-soup-uri.h"
|
||||
#else
|
||||
#define LIBSOUP_USE_UNSTABLE_REQUEST_API
|
||||
#include <libsoup/soup.h>
|
||||
#include <libsoup/soup-requester.h>
|
||||
#include <libsoup/soup-request-http.h>
|
||||
#endif
|
||||
|
||||
#include "ostree-fetcher.h"
|
||||
|
||||
#include "libglnx.h"
|
||||
|
||||
void
|
||||
_ostree_fetcher_uri_free (OstreeFetcherURI *uri)
|
||||
{
|
||||
if (uri)
|
||||
soup_uri_free ((SoupURI*)uri);
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_parse (const char *str,
|
||||
GError **error)
|
||||
{
|
||||
SoupURI *soupuri = soup_uri_new (str);
|
||||
if (soupuri == NULL)
|
||||
{
|
||||
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
||||
"Failed to parse uri: %s", str);
|
||||
return NULL;
|
||||
}
|
||||
return (OstreeFetcherURI*)soupuri;
|
||||
}
|
||||
|
||||
static OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_new_path_internal (OstreeFetcherURI *uri,
|
||||
gboolean extend,
|
||||
const char *path)
|
||||
{
|
||||
SoupURI *newuri = soup_uri_copy ((SoupURI*)uri);
|
||||
if (path)
|
||||
{
|
||||
if (extend)
|
||||
{
|
||||
const char *origpath = soup_uri_get_path ((SoupURI*)uri);
|
||||
g_autofree char *newpath = g_build_filename (origpath, path, NULL);
|
||||
soup_uri_set_path (newuri, newpath);
|
||||
}
|
||||
else
|
||||
{
|
||||
soup_uri_set_path (newuri, path);
|
||||
}
|
||||
}
|
||||
return (OstreeFetcherURI*)newuri;
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_new_path (OstreeFetcherURI *uri,
|
||||
const char *path)
|
||||
{
|
||||
return _ostree_fetcher_uri_new_path_internal (uri, FALSE, path);
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_new_subpath (OstreeFetcherURI *uri,
|
||||
const char *subpath)
|
||||
{
|
||||
return _ostree_fetcher_uri_new_path_internal (uri, TRUE, subpath);
|
||||
}
|
||||
|
||||
OstreeFetcherURI *
|
||||
_ostree_fetcher_uri_clone (OstreeFetcherURI *uri)
|
||||
{
|
||||
return _ostree_fetcher_uri_new_subpath (uri, NULL);
|
||||
}
|
||||
|
||||
char *
|
||||
_ostree_fetcher_uri_get_scheme (OstreeFetcherURI *uri)
|
||||
{
|
||||
return g_strdup (soup_uri_get_scheme ((SoupURI*)uri));
|
||||
}
|
||||
|
||||
char *
|
||||
_ostree_fetcher_uri_get_path (OstreeFetcherURI *uri)
|
||||
{
|
||||
return g_strdup (soup_uri_get_path ((SoupURI*)uri));
|
||||
}
|
||||
|
||||
char *
|
||||
_ostree_fetcher_uri_to_string (OstreeFetcherURI *uri)
|
||||
{
|
||||
return soup_uri_to_string ((SoupURI*)uri, FALSE);
|
||||
}
|
@ -26,7 +26,7 @@
|
||||
#include "ostree.h"
|
||||
#include "otutil.h"
|
||||
|
||||
#ifdef HAVE_LIBSOUP
|
||||
#ifdef HAVE_LIBCURL_OR_LIBSOUP
|
||||
|
||||
#include "ostree-core-private.h"
|
||||
#include "ostree-repo-private.h"
|
||||
@ -3405,7 +3405,7 @@ out:
|
||||
return ret;
|
||||
}
|
||||
|
||||
#else /* HAVE_LIBSOUP */
|
||||
#else /* HAVE_LIBCURL_OR_LIBSOUP */
|
||||
|
||||
gboolean
|
||||
ostree_repo_pull_with_options (OstreeRepo *self,
|
||||
@ -3434,4 +3434,4 @@ ostree_repo_remote_fetch_summary_with_options (OstreeRepo *self,
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
#endif /* HAVE_LIBSOUP */
|
||||
#endif /* HAVE_LIBCURL_OR_LIBSOUP */
|
||||
|
140
src/libostree/ostree-soup-form.c
Normal file
140
src/libostree/ostree-soup-form.c
Normal file
@ -0,0 +1,140 @@
|
||||
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
|
||||
/* soup-form.c : utility functions for HTML forms */
|
||||
|
||||
/*
|
||||
* Copyright 2008 Red Hat, Inc.
|
||||
*/
|
||||
|
||||
/* This one is stripped down to only have soup_form_encode_hash()
|
||||
* and soup_form_encode_valist() which are the only bits that soup-uri.c
|
||||
* calls.
|
||||
*/
|
||||
|
||||
#include <config.h>
|
||||
|
||||
#include <string.h>
|
||||
|
||||
#include "ostree-soup-uri.h"
|
||||
|
||||
/**
|
||||
* SECTION:soup-form
|
||||
* @short_description: HTML form handling
|
||||
* @see_also: #SoupMultipart
|
||||
*
|
||||
* libsoup contains several help methods for processing HTML forms as
|
||||
* defined by <ulink
|
||||
* url="http://www.w3.org/TR/html401/interact/forms.html#h-17.13">the
|
||||
* HTML 4.01 specification</ulink>.
|
||||
**/
|
||||
|
||||
/**
|
||||
* SOUP_FORM_MIME_TYPE_URLENCODED:
|
||||
*
|
||||
* A macro containing the value
|
||||
* <literal>"application/x-www-form-urlencoded"</literal>; the default
|
||||
* MIME type for POSTing HTML form data.
|
||||
*
|
||||
* Since: 2.26
|
||||
**/
|
||||
|
||||
/**
|
||||
* SOUP_FORM_MIME_TYPE_MULTIPART:
|
||||
*
|
||||
* A macro containing the value
|
||||
* <literal>"multipart/form-data"</literal>; the MIME type used for
|
||||
* posting form data that contains files to be uploaded.
|
||||
*
|
||||
* Since: 2.26
|
||||
**/
|
||||
|
||||
#define XDIGIT(c) ((c) <= '9' ? (c) - '0' : ((c) & 0x4F) - 'A' + 10)
|
||||
#define HEXCHAR(s) ((XDIGIT (s[1]) << 4) + XDIGIT (s[2]))
|
||||
|
||||
static void
|
||||
append_form_encoded (GString *str, const char *in)
|
||||
{
|
||||
const unsigned char *s = (const unsigned char *)in;
|
||||
|
||||
while (*s) {
|
||||
if (*s == ' ') {
|
||||
g_string_append_c (str, '+');
|
||||
s++;
|
||||
} else if (!g_ascii_isalnum (*s) && (*s != '-') && (*s != '_')
|
||||
&& (*s != '.'))
|
||||
g_string_append_printf (str, "%%%02X", (int)*s++);
|
||||
else
|
||||
g_string_append_c (str, *s++);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
encode_pair (GString *str, const char *name, const char *value)
|
||||
{
|
||||
g_return_if_fail (name != NULL);
|
||||
g_return_if_fail (value != NULL);
|
||||
|
||||
if (str->len)
|
||||
g_string_append_c (str, '&');
|
||||
append_form_encoded (str, name);
|
||||
g_string_append_c (str, '=');
|
||||
append_form_encoded (str, value);
|
||||
}
|
||||
|
||||
/**
|
||||
* soup_form_encode_hash:
|
||||
* @form_data_set: (element-type utf8 utf8): a hash table containing
|
||||
* name/value pairs (as strings)
|
||||
*
|
||||
* Encodes @form_data_set into a value of type
|
||||
* "application/x-www-form-urlencoded", as defined in the HTML 4.01
|
||||
* spec.
|
||||
*
|
||||
* Note that the HTML spec states that "The control names/values are
|
||||
* listed in the order they appear in the document." Since this method
|
||||
* takes a hash table, it cannot enforce that; if you care about the
|
||||
* ordering of the form fields, use soup_form_encode_datalist().
|
||||
*
|
||||
* Return value: the encoded form
|
||||
**/
|
||||
char *
|
||||
soup_form_encode_hash (GHashTable *form_data_set)
|
||||
{
|
||||
GString *str = g_string_new (NULL);
|
||||
GHashTableIter iter;
|
||||
gpointer name, value;
|
||||
|
||||
g_hash_table_iter_init (&iter, form_data_set);
|
||||
while (g_hash_table_iter_next (&iter, &name, &value))
|
||||
encode_pair (str, name, value);
|
||||
return g_string_free (str, FALSE);
|
||||
}
|
||||
|
||||
/**
|
||||
* soup_form_encode_valist:
|
||||
* @first_field: name of the first form field
|
||||
* @args: pointer to additional values, as in soup_form_encode()
|
||||
*
|
||||
* See soup_form_encode(). This is mostly an internal method, used by
|
||||
* various other methods such as soup_uri_set_query_from_fields() and
|
||||
* soup_form_request_new().
|
||||
*
|
||||
* Return value: the encoded form
|
||||
**/
|
||||
char *
|
||||
soup_form_encode_valist (const char *first_field, va_list args)
|
||||
{
|
||||
GString *str = g_string_new (NULL);
|
||||
const char *name, *value;
|
||||
|
||||
name = first_field;
|
||||
value = va_arg (args, const char *);
|
||||
while (name && value) {
|
||||
encode_pair (str, name, value);
|
||||
|
||||
name = va_arg (args, const char *);
|
||||
if (name)
|
||||
value = va_arg (args, const char *);
|
||||
}
|
||||
|
||||
return g_string_free (str, FALSE);
|
||||
}
|
1483
src/libostree/ostree-soup-uri.c
Normal file
1483
src/libostree/ostree-soup-uri.c
Normal file
File diff suppressed because it is too large
Load Diff
147
src/libostree/ostree-soup-uri.h
Normal file
147
src/libostree/ostree-soup-uri.h
Normal file
@ -0,0 +1,147 @@
|
||||
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
|
||||
|
||||
/*
|
||||
* Copyright 1999-2002 Ximian, Inc.
|
||||
*/
|
||||
|
||||
/* NOTE - taken from the libsoup codebase for use by the ostree curl backend
|
||||
* (yes, ironically enough).
|
||||
*
|
||||
* Please watch for future changes in libsoup.
|
||||
*/
|
||||
|
||||
|
||||
#ifndef SOUP_URI_H
|
||||
#define SOUP_URI_H 1
|
||||
|
||||
/* OSTREECHANGE: make struct private
|
||||
* Only include gio, and skip available definitions.
|
||||
*/
|
||||
#include <gio/gio.h>
|
||||
#define SOUP_AVAILABLE_IN_2_4
|
||||
#define SOUP_AVAILABLE_IN_2_28
|
||||
#define SOUP_AVAILABLE_IN_2_32
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
/* OSTREECHANGE: make struct private */
|
||||
typedef struct _SoupURI SoupURI;
|
||||
|
||||
/* OSTREECHANGE: import soup-misc's interning */
|
||||
#define SOUP_VAR extern
|
||||
#define _SOUP_ATOMIC_INTERN_STRING(variable, value) ((const char *)(g_atomic_pointer_get (&(variable)) ? (variable) : (g_atomic_pointer_set (&(variable), (gpointer)g_intern_static_string (value)), (variable))))
|
||||
#define SOUP_URI_SCHEME_HTTP _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_HTTP, "http")
|
||||
#define SOUP_URI_SCHEME_HTTPS _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_HTTPS, "https")
|
||||
#define SOUP_URI_SCHEME_FTP _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_FTP, "ftp")
|
||||
#define SOUP_URI_SCHEME_FILE _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_FILE, "file")
|
||||
#define SOUP_URI_SCHEME_DATA _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_DATA, "data")
|
||||
#define SOUP_URI_SCHEME_RESOURCE _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_RESOURCE, "resource")
|
||||
#define SOUP_URI_SCHEME_WS _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_WS, "ws")
|
||||
#define SOUP_URI_SCHEME_WSS _SOUP_ATOMIC_INTERN_STRING (_SOUP_URI_SCHEME_WSS, "wss")
|
||||
|
||||
/* OSTREECHANGE: import soup-form bits */
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
char *soup_form_encode_hash (GHashTable *form_data_set);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
char *soup_form_encode_valist (const char *first_field,
|
||||
va_list args);
|
||||
|
||||
SOUP_VAR gpointer _SOUP_URI_SCHEME_HTTP, _SOUP_URI_SCHEME_HTTPS;
|
||||
SOUP_VAR gpointer _SOUP_URI_SCHEME_FTP;
|
||||
SOUP_VAR gpointer _SOUP_URI_SCHEME_FILE, _SOUP_URI_SCHEME_DATA, _SOUP_URI_SCHEME_RESOURCE;
|
||||
SOUP_VAR gpointer _SOUP_URI_SCHEME_WS, _SOUP_URI_SCHEME_WSS;
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
SoupURI *soup_uri_new_with_base (SoupURI *base,
|
||||
const char *uri_string);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
SoupURI *soup_uri_new (const char *uri_string);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
char *soup_uri_to_string (SoupURI *uri,
|
||||
gboolean just_path_and_query);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
SoupURI *soup_uri_copy (SoupURI *uri);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
gboolean soup_uri_equal (SoupURI *uri1,
|
||||
SoupURI *uri2);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_free (SoupURI *uri);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
char *soup_uri_encode (const char *part,
|
||||
const char *escape_extra);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
char *soup_uri_decode (const char *part);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
char *soup_uri_normalize (const char *part,
|
||||
const char *unescape_extra);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
gboolean soup_uri_uses_default_port (SoupURI *uri);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_scheme (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_scheme (SoupURI *uri,
|
||||
const char *scheme);
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_user (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_user (SoupURI *uri,
|
||||
const char *user);
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_password (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_password (SoupURI *uri,
|
||||
const char *password);
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_host (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_host (SoupURI *uri,
|
||||
const char *host);
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
guint soup_uri_get_port (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_port (SoupURI *uri,
|
||||
guint port);
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_path (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_path (SoupURI *uri,
|
||||
const char *path);
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_query (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_query (SoupURI *uri,
|
||||
const char *query);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_query_from_form (SoupURI *uri,
|
||||
GHashTable *form);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_query_from_fields (SoupURI *uri,
|
||||
const char *first_field,
|
||||
...) G_GNUC_NULL_TERMINATED;
|
||||
SOUP_AVAILABLE_IN_2_32
|
||||
const char *soup_uri_get_fragment (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_4
|
||||
void soup_uri_set_fragment (SoupURI *uri,
|
||||
const char *fragment);
|
||||
|
||||
SOUP_AVAILABLE_IN_2_28
|
||||
SoupURI *soup_uri_copy_host (SoupURI *uri);
|
||||
SOUP_AVAILABLE_IN_2_28
|
||||
guint soup_uri_host_hash (gconstpointer key);
|
||||
SOUP_AVAILABLE_IN_2_28
|
||||
gboolean soup_uri_host_equal (gconstpointer v1,
|
||||
gconstpointer v2);
|
||||
|
||||
#define SOUP_URI_IS_VALID(uri) ((uri) && (uri)->scheme && (uri)->path)
|
||||
#define SOUP_URI_VALID_FOR_HTTP(uri) ((uri) && ((uri)->scheme == SOUP_URI_SCHEME_HTTP || (uri)->scheme == SOUP_URI_SCHEME_HTTPS) && (uri)->host && (uri)->path)
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /*SOUP_URI_H*/
|
@ -21,13 +21,12 @@
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include <libsoup/soup.h>
|
||||
|
||||
#include "otutil.h"
|
||||
|
||||
#include "ot-main.h"
|
||||
#include "ot-remote-builtins.h"
|
||||
#include "ostree-repo-private.h"
|
||||
#include "ot-remote-cookie-util.h"
|
||||
|
||||
|
||||
static GOptionEntry option_entries[] = {
|
||||
@ -46,8 +45,6 @@ ot_remote_builtin_add_cookie (int argc, char **argv, GCancellable *cancellable,
|
||||
const char *value;
|
||||
g_autofree char *jar_path = NULL;
|
||||
g_autofree char *cookie_file = NULL;
|
||||
glnx_unref_object SoupCookieJar *jar = NULL;
|
||||
SoupCookie *cookie;
|
||||
|
||||
context = g_option_context_new ("NAME DOMAIN PATH COOKIE_NAME VALUE - Add a cookie to remote");
|
||||
|
||||
@ -70,15 +67,8 @@ ot_remote_builtin_add_cookie (int argc, char **argv, GCancellable *cancellable,
|
||||
cookie_file = g_strdup_printf ("%s.cookies.txt", remote_name);
|
||||
jar_path = g_build_filename (gs_file_get_path_cached (repo->repodir), cookie_file, NULL);
|
||||
|
||||
jar = soup_cookie_jar_text_new (jar_path, FALSE);
|
||||
|
||||
/* Pick a silly long expire time, we're just storing the cookies in the
|
||||
* jar and on pull the jar is read-only so expiry has little actual value */
|
||||
cookie = soup_cookie_new (cookie_name, value, domain, path,
|
||||
SOUP_COOKIE_MAX_AGE_ONE_YEAR * 25);
|
||||
|
||||
/* jar takes ownership of cookie */
|
||||
soup_cookie_jar_add_cookie (jar, cookie);
|
||||
if (!ot_add_cookie_at (AT_FDCWD, jar_path, domain, path, cookie_name, value, error))
|
||||
return FALSE;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
@ -21,14 +21,13 @@
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include <libsoup/soup.h>
|
||||
|
||||
#include "otutil.h"
|
||||
#include <sys/stat.h>
|
||||
|
||||
#include "ot-main.h"
|
||||
#include "ot-remote-builtins.h"
|
||||
#include "ostree-repo-private.h"
|
||||
|
||||
#include "ot-remote-cookie-util.h"
|
||||
|
||||
static GOptionEntry option_entries[] = {
|
||||
{ NULL }
|
||||
@ -45,9 +44,6 @@ ot_remote_builtin_delete_cookie (int argc, char **argv, GCancellable *cancellabl
|
||||
const char *cookie_name;
|
||||
g_autofree char *jar_path = NULL;
|
||||
g_autofree char *cookie_file = NULL;
|
||||
glnx_unref_object SoupCookieJar *jar = NULL;
|
||||
GSList *cookies;
|
||||
gboolean found = FALSE;
|
||||
|
||||
context = g_option_context_new ("NAME DOMAIN PATH COOKIE_NAME- Remote one cookie from remote");
|
||||
|
||||
@ -69,28 +65,8 @@ ot_remote_builtin_delete_cookie (int argc, char **argv, GCancellable *cancellabl
|
||||
cookie_file = g_strdup_printf ("%s.cookies.txt", remote_name);
|
||||
jar_path = g_build_filename (gs_file_get_path_cached (repo->repodir), cookie_file, NULL);
|
||||
|
||||
jar = soup_cookie_jar_text_new (jar_path, FALSE);
|
||||
cookies = soup_cookie_jar_all_cookies (jar);
|
||||
if (!ot_delete_cookie_at (AT_FDCWD, jar_path, domain, path, cookie_name, error))
|
||||
return FALSE;
|
||||
|
||||
while (cookies != NULL)
|
||||
{
|
||||
SoupCookie *cookie = cookies->data;
|
||||
|
||||
if (!strcmp (domain, soup_cookie_get_domain (cookie)) &&
|
||||
!strcmp (path, soup_cookie_get_path (cookie)) &&
|
||||
!strcmp (cookie_name, soup_cookie_get_name (cookie)))
|
||||
{
|
||||
soup_cookie_jar_delete_cookie (jar, cookie);
|
||||
|
||||
found = TRUE;
|
||||
}
|
||||
|
||||
soup_cookie_free (cookie);
|
||||
cookies = g_slist_delete_link (cookies, cookies);
|
||||
}
|
||||
|
||||
if (!found)
|
||||
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED, "Cookie not found in jar");
|
||||
|
||||
return found;
|
||||
return TRUE;
|
||||
}
|
||||
|
@ -21,14 +21,12 @@
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include <libsoup/soup.h>
|
||||
|
||||
#include "otutil.h"
|
||||
|
||||
#include "ot-main.h"
|
||||
#include "ot-remote-builtins.h"
|
||||
#include "ostree-repo-private.h"
|
||||
|
||||
#include "ot-remote-cookie-util.h"
|
||||
|
||||
static GOptionEntry option_entries[] = {
|
||||
{ NULL }
|
||||
@ -42,8 +40,6 @@ ot_remote_builtin_list_cookies (int argc, char **argv, GCancellable *cancellable
|
||||
const char *remote_name;
|
||||
g_autofree char *jar_path = NULL;
|
||||
g_autofree char *cookie_file = NULL;
|
||||
glnx_unref_object SoupCookieJar *jar = NULL;
|
||||
GSList *cookies;
|
||||
|
||||
context = g_option_context_new ("NAME - Show remote repository cookies");
|
||||
|
||||
@ -62,25 +58,8 @@ ot_remote_builtin_list_cookies (int argc, char **argv, GCancellable *cancellable
|
||||
cookie_file = g_strdup_printf ("%s.cookies.txt", remote_name);
|
||||
jar_path = g_build_filename (g_file_get_path (repo->repodir), cookie_file, NULL);
|
||||
|
||||
jar = soup_cookie_jar_text_new (jar_path, TRUE);
|
||||
cookies = soup_cookie_jar_all_cookies (jar);
|
||||
|
||||
while (cookies != NULL)
|
||||
{
|
||||
SoupCookie *cookie = cookies->data;
|
||||
SoupDate *expiry = soup_cookie_get_expires (cookie);
|
||||
|
||||
g_print ("--\n");
|
||||
g_print ("Domain: %s\n", soup_cookie_get_domain (cookie));
|
||||
g_print ("Path: %s\n", soup_cookie_get_path (cookie));
|
||||
g_print ("Name: %s\n", soup_cookie_get_name (cookie));
|
||||
g_print ("Secure: %s\n", soup_cookie_get_secure (cookie) ? "yes" : "no");
|
||||
g_print ("Expires: %s\n", soup_date_to_string (expiry, SOUP_DATE_COOKIE));
|
||||
g_print ("Value: %s\n", soup_cookie_get_value (cookie));
|
||||
|
||||
soup_cookie_free (cookie);
|
||||
cookies = g_slist_delete_link (cookies, cookies);
|
||||
}
|
||||
if (!ot_list_cookies_at (AT_FDCWD, jar_path, error))
|
||||
return FALSE;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
333
src/ostree/ot-remote-cookie-util.c
Normal file
333
src/ostree/ot-remote-cookie-util.c
Normal file
@ -0,0 +1,333 @@
|
||||
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
|
||||
*
|
||||
* Copyright (C) 2015 Red Hat, Inc.
|
||||
* Copyright (C) 2016 Sjoerd Simons <sjoerd@luon.net>
|
||||
*
|
||||
* 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 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
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include "ot-remote-cookie-util.h"
|
||||
|
||||
#ifndef HAVE_LIBCURL
|
||||
#include <libsoup/soup.h>
|
||||
#endif
|
||||
|
||||
#include "otutil.h"
|
||||
#include "ot-main.h"
|
||||
#include "ot-remote-builtins.h"
|
||||
#include "ostree-repo-private.h"
|
||||
|
||||
typedef struct OtCookieParser OtCookieParser;
|
||||
struct OtCookieParser {
|
||||
char *buf;
|
||||
char *iter;
|
||||
|
||||
char *line;
|
||||
char *domain;
|
||||
char *flag;
|
||||
char *path;
|
||||
char *secure;
|
||||
long long unsigned int expiration;
|
||||
char *name;
|
||||
char *value;
|
||||
};
|
||||
void ot_cookie_parser_free (OtCookieParser *parser);
|
||||
G_DEFINE_AUTOPTR_CLEANUP_FUNC(OtCookieParser, ot_cookie_parser_free)
|
||||
|
||||
gboolean
|
||||
ot_parse_cookies_at (int dfd, const char *path,
|
||||
OtCookieParser **out_parser,
|
||||
GCancellable *cancellable,
|
||||
GError **error);
|
||||
gboolean
|
||||
ot_parse_cookies_next (OtCookieParser *parser);
|
||||
|
||||
static void
|
||||
ot_cookie_parser_clear (OtCookieParser *parser)
|
||||
{
|
||||
g_clear_pointer (&parser->domain, (GDestroyNotify)g_free);
|
||||
g_clear_pointer (&parser->flag, (GDestroyNotify)g_free);
|
||||
g_clear_pointer (&parser->path, (GDestroyNotify)g_free);
|
||||
g_clear_pointer (&parser->secure, (GDestroyNotify)g_free);
|
||||
g_clear_pointer (&parser->name, (GDestroyNotify)g_free);
|
||||
g_clear_pointer (&parser->value, (GDestroyNotify)g_free);
|
||||
}
|
||||
|
||||
void
|
||||
ot_cookie_parser_free (OtCookieParser *parser)
|
||||
{
|
||||
ot_cookie_parser_clear (parser);
|
||||
g_free (parser->buf);
|
||||
g_free (parser);
|
||||
}
|
||||
|
||||
gboolean
|
||||
ot_parse_cookies_at (int dfd, const char *path,
|
||||
OtCookieParser **out_parser,
|
||||
GCancellable *cancellable,
|
||||
GError **error)
|
||||
{
|
||||
OtCookieParser *parser;
|
||||
g_autofree char *cookies_content = NULL;
|
||||
glnx_fd_close int infd = -1;
|
||||
|
||||
infd = openat (dfd, path, O_RDONLY | O_CLOEXEC);
|
||||
if (infd < 0)
|
||||
{
|
||||
if (errno != ENOENT)
|
||||
{
|
||||
glnx_set_error_from_errno (error);
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
cookies_content = glnx_fd_readall_utf8 (infd, NULL, cancellable, error);
|
||||
if (!cookies_content)
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
parser = *out_parser = g_new0 (OtCookieParser, 1);
|
||||
parser->buf = g_steal_pointer (&cookies_content);
|
||||
parser->iter = parser->buf;
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
gboolean
|
||||
ot_parse_cookies_next (OtCookieParser *parser)
|
||||
{
|
||||
while (parser->iter)
|
||||
{
|
||||
char *iter = parser->iter;
|
||||
char *next = strchr (iter, '\n');
|
||||
|
||||
if (next)
|
||||
{
|
||||
*next = '\0';
|
||||
parser->iter = next + 1;
|
||||
}
|
||||
else
|
||||
parser->iter = NULL;
|
||||
|
||||
ot_cookie_parser_clear (parser);
|
||||
if (sscanf (iter, "%ms\t%ms\t%ms\t%ms\t%llu\t%ms\t%ms",
|
||||
&parser->domain,
|
||||
&parser->flag,
|
||||
&parser->path,
|
||||
&parser->secure,
|
||||
&parser->expiration,
|
||||
&parser->name,
|
||||
&parser->value) != 7)
|
||||
continue;
|
||||
|
||||
parser->line = iter;
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
gboolean
|
||||
ot_add_cookie_at (int dfd, const char *jar_path,
|
||||
const char *domain, const char *path,
|
||||
const char *name, const char *value,
|
||||
GError **error)
|
||||
{
|
||||
#ifdef HAVE_LIBCURL
|
||||
glnx_fd_close int fd = openat (AT_FDCWD, jar_path, O_WRONLY | O_APPEND | O_CREAT, 0644);
|
||||
g_autofree char *buf = NULL;
|
||||
g_autoptr(GDateTime) now = NULL;
|
||||
g_autoptr(GDateTime) expires = NULL;
|
||||
|
||||
if (fd < 0)
|
||||
{
|
||||
glnx_set_error_from_errno (error);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
now = g_date_time_new_now_utc ();
|
||||
expires = g_date_time_add_years (now, 25);
|
||||
|
||||
/* Adapted from soup-cookie-jar-text.c:write_cookie() */
|
||||
buf = g_strdup_printf ("%s\t%s\t%s\t%s\t%llu\t%s\t%s\n",
|
||||
domain,
|
||||
*domain == '.' ? "TRUE" : "FALSE",
|
||||
path,
|
||||
"FALSE",
|
||||
(long long unsigned)g_date_time_to_unix (expires),
|
||||
name,
|
||||
value);
|
||||
if (glnx_loop_write (fd, buf, strlen (buf)) < 0)
|
||||
{
|
||||
glnx_set_error_from_errno (error);
|
||||
return FALSE;
|
||||
}
|
||||
#else
|
||||
glnx_unref_object SoupCookieJar *jar = NULL;
|
||||
SoupCookie *cookie;
|
||||
|
||||
jar = soup_cookie_jar_text_new (jar_path, FALSE);
|
||||
|
||||
/* Pick a silly long expire time, we're just storing the cookies in the
|
||||
* jar and on pull the jar is read-only so expiry has little actual value */
|
||||
cookie = soup_cookie_new (name, value, domain, path,
|
||||
SOUP_COOKIE_MAX_AGE_ONE_YEAR * 25);
|
||||
|
||||
/* jar takes ownership of cookie */
|
||||
soup_cookie_jar_add_cookie (jar, cookie);
|
||||
#endif
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
gboolean
|
||||
ot_delete_cookie_at (int dfd, const char *jar_path,
|
||||
const char *domain, const char *path,
|
||||
const char *name,
|
||||
GError **error)
|
||||
{
|
||||
gboolean found = FALSE;
|
||||
#ifdef HAVE_LIBCURL
|
||||
glnx_fd_close int tempfile_fd = -1;
|
||||
g_autofree char *tempfile_path = NULL;
|
||||
g_autofree char *dnbuf = NULL;
|
||||
const char *dn = NULL;
|
||||
g_autoptr(OtCookieParser) parser = NULL;
|
||||
|
||||
if (!ot_parse_cookies_at (dfd, jar_path, &parser, NULL, error))
|
||||
return FALSE;
|
||||
|
||||
dnbuf = g_strdup (jar_path);
|
||||
dn = dirname (dnbuf);
|
||||
if (!glnx_open_tmpfile_linkable_at (AT_FDCWD, dn, O_WRONLY | O_CLOEXEC,
|
||||
&tempfile_fd, &tempfile_path,
|
||||
error))
|
||||
return FALSE;
|
||||
|
||||
while (ot_parse_cookies_next (parser))
|
||||
{
|
||||
if (strcmp (domain, parser->domain) == 0 &&
|
||||
strcmp (path, parser->path) == 0 &&
|
||||
strcmp (name, parser->name) == 0)
|
||||
{
|
||||
found = TRUE;
|
||||
/* Match, skip writing this one */
|
||||
continue;
|
||||
}
|
||||
|
||||
if (glnx_loop_write (tempfile_fd, parser->line, strlen (parser->line)) < 0 ||
|
||||
glnx_loop_write (tempfile_fd, "\n", 1) < 0)
|
||||
{
|
||||
glnx_set_error_from_errno (error);
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
if (!glnx_link_tmpfile_at (AT_FDCWD, GLNX_LINK_TMPFILE_REPLACE,
|
||||
tempfile_fd,
|
||||
tempfile_path,
|
||||
AT_FDCWD, jar_path,
|
||||
error))
|
||||
return FALSE;
|
||||
#else
|
||||
GSList *cookies;
|
||||
glnx_unref_object SoupCookieJar *jar = NULL;
|
||||
|
||||
jar = soup_cookie_jar_text_new (jar_path, FALSE);
|
||||
cookies = soup_cookie_jar_all_cookies (jar);
|
||||
|
||||
while (cookies != NULL)
|
||||
{
|
||||
SoupCookie *cookie = cookies->data;
|
||||
|
||||
if (!strcmp (domain, soup_cookie_get_domain (cookie)) &&
|
||||
!strcmp (path, soup_cookie_get_path (cookie)) &&
|
||||
!strcmp (name, soup_cookie_get_name (cookie)))
|
||||
{
|
||||
soup_cookie_jar_delete_cookie (jar, cookie);
|
||||
|
||||
found = TRUE;
|
||||
}
|
||||
|
||||
soup_cookie_free (cookie);
|
||||
cookies = g_slist_delete_link (cookies, cookies);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!found)
|
||||
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED, "Cookie not found in jar");
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
||||
gboolean
|
||||
ot_list_cookies_at (int dfd, const char *jar_path, GError **error)
|
||||
{
|
||||
#ifdef HAVE_LIBCURL
|
||||
glnx_fd_close int tempfile_fd = -1;
|
||||
g_autofree char *tempfile_path = NULL;
|
||||
g_autofree char *dnbuf = NULL;
|
||||
const char *dn = NULL;
|
||||
g_autoptr(OtCookieParser) parser = NULL;
|
||||
|
||||
if (!ot_parse_cookies_at (AT_FDCWD, jar_path, &parser, NULL, error))
|
||||
return FALSE;
|
||||
|
||||
dnbuf = dirname (g_strdup (jar_path));
|
||||
dn = dnbuf;
|
||||
if (!glnx_open_tmpfile_linkable_at (AT_FDCWD, dn, O_WRONLY | O_CLOEXEC,
|
||||
&tempfile_fd, &tempfile_path,
|
||||
error))
|
||||
return FALSE;
|
||||
|
||||
while (ot_parse_cookies_next (parser))
|
||||
{
|
||||
g_autoptr(GDateTime) expires = g_date_time_new_from_unix_utc (parser->expiration);
|
||||
g_autofree char *expires_str = g_date_time_format (expires, "%Y-%m-%d %H:%M:%S +0000");
|
||||
|
||||
g_print ("--\n");
|
||||
g_print ("Domain: %s\n", parser->domain);
|
||||
g_print ("Path: %s\n", parser->path);
|
||||
g_print ("Name: %s\n", parser->name);
|
||||
g_print ("Secure: %s\n", parser->secure);
|
||||
g_print ("Expires: %s\n", expires_str);
|
||||
g_print ("Value: %s\n", parser->value);
|
||||
}
|
||||
#else
|
||||
glnx_unref_object SoupCookieJar *jar = soup_cookie_jar_text_new (jar_path, TRUE);
|
||||
GSList *cookies = soup_cookie_jar_all_cookies (jar);
|
||||
|
||||
while (cookies != NULL)
|
||||
{
|
||||
SoupCookie *cookie = cookies->data;
|
||||
SoupDate *expiry = soup_cookie_get_expires (cookie);
|
||||
|
||||
g_print ("--\n");
|
||||
g_print ("Domain: %s\n", soup_cookie_get_domain (cookie));
|
||||
g_print ("Path: %s\n", soup_cookie_get_path (cookie));
|
||||
g_print ("Name: %s\n", soup_cookie_get_name (cookie));
|
||||
g_print ("Secure: %s\n", soup_cookie_get_secure (cookie) ? "yes" : "no");
|
||||
g_print ("Expires: %s\n", soup_date_to_string (expiry, SOUP_DATE_COOKIE));
|
||||
g_print ("Value: %s\n", soup_cookie_get_value (cookie));
|
||||
|
||||
soup_cookie_free (cookie);
|
||||
cookies = g_slist_delete_link (cookies, cookies);
|
||||
}
|
||||
#endif
|
||||
return TRUE;
|
||||
}
|
42
src/ostree/ot-remote-cookie-util.h
Normal file
42
src/ostree/ot-remote-cookie-util.h
Normal file
@ -0,0 +1,42 @@
|
||||
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*-
|
||||
*
|
||||
* Copyright (C) 2017 Colin Walters <walters@verbum.org>
|
||||
*
|
||||
* 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 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
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include "libglnx.h"
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
gboolean
|
||||
ot_add_cookie_at (int dfd, const char *jar_path,
|
||||
const char *domain, const char *path,
|
||||
const char *name, const char *value,
|
||||
GError **error);
|
||||
|
||||
gboolean
|
||||
ot_delete_cookie_at (int dfd, const char *jar_path,
|
||||
const char *domain, const char *path,
|
||||
const char *name,
|
||||
GError **error);
|
||||
|
||||
gboolean
|
||||
ot_list_cookies_at (int dfd, const char *jar_path, GError **error);
|
||||
|
||||
G_END_DECLS
|
@ -52,6 +52,10 @@ NULL=
|
||||
# If yes, test failures break the build; if no, they are reported but ignored
|
||||
: "${ci_test_fatal:=yes}"
|
||||
|
||||
# ci_configopts:
|
||||
# Additional args for configure
|
||||
: "${ci_configopts:=}"
|
||||
|
||||
if [ -n "$ci_docker" ]; then
|
||||
exec docker run \
|
||||
--env=ci_docker="" \
|
||||
@ -59,6 +63,7 @@ if [ -n "$ci_docker" ]; then
|
||||
--env=ci_sudo=yes \
|
||||
--env=ci_test="${ci_test}" \
|
||||
--env=ci_test_fatal="${ci_test_fatal}" \
|
||||
--env=ci_configopts="${ci_configopts}" \
|
||||
--privileged \
|
||||
ci-image \
|
||||
tests/ci-build.sh
|
||||
@ -81,6 +86,7 @@ make="make -j${ci_parallel} V=1 VERBOSE=1"
|
||||
../configure \
|
||||
--enable-always-build-tests \
|
||||
--enable-installed-tests \
|
||||
${ci_configopts}
|
||||
"$@"
|
||||
|
||||
${make}
|
||||
|
@ -50,6 +50,9 @@ NULL=
|
||||
# Typical values for ci_distro=fedora might be 25, rawhide
|
||||
: "${ci_suite:=jessie}"
|
||||
|
||||
# ci_configopts: Additional arguments for configure
|
||||
: "${ci_configopts:=}"
|
||||
|
||||
if [ $(id -u) = 0 ]; then
|
||||
sudo=
|
||||
else
|
||||
@ -104,6 +107,7 @@ case "$ci_distro" in
|
||||
libmount-dev \
|
||||
libselinux1-dev \
|
||||
libsoup2.4-dev \
|
||||
libcurl4-openssl-dev \
|
||||
procps \
|
||||
zlib1g-dev \
|
||||
${NULL}
|
||||
|
@ -28,12 +28,9 @@ setup_fake_remote_repo1 "archive-z2" "" \
|
||||
"--expected-cookies foo=bar --expected-cookies baz=badger"
|
||||
|
||||
assert_fail (){
|
||||
set +e
|
||||
$@
|
||||
if [ $? = 0 ] ; then
|
||||
echo 1>&2 "$@ did not fail"; exit 1
|
||||
if $@; then
|
||||
(echo 1>&2 "$@ did not fail"; exit 1)
|
||||
fi
|
||||
set -euo pipefail
|
||||
}
|
||||
|
||||
cd ${test_tmpdir}
|
||||
@ -50,12 +47,16 @@ echo "ok, setup done"
|
||||
# Add 2 cookies, pull should succeed now
|
||||
${CMD_PREFIX} ostree --repo=repo remote add-cookie origin 127.0.0.1 / foo bar
|
||||
${CMD_PREFIX} ostree --repo=repo remote add-cookie origin 127.0.0.1 / baz badger
|
||||
assert_file_has_content repo/origin.cookies.txt foo.*bar
|
||||
assert_file_has_content repo/origin.cookies.txt baz.*badger
|
||||
${CMD_PREFIX} ostree --repo=repo pull origin main
|
||||
|
||||
echo "ok, initial cookie pull succeeded"
|
||||
|
||||
# Delete one cookie, if successful pulls will fail again
|
||||
${CMD_PREFIX} ostree --repo=repo remote delete-cookie origin 127.0.0.1 / baz badger
|
||||
assert_file_has_content repo/origin.cookies.txt foo.*bar
|
||||
assert_not_file_has_content repo/origin.cookies.txt baz.*badger
|
||||
assert_fail ${CMD_PREFIX} ostree --repo=repo pull origin main
|
||||
|
||||
echo "ok, delete succeeded"
|
||||
@ -63,6 +64,8 @@ echo "ok, delete succeeded"
|
||||
# Re-add the removed cooking and things succeed again, verified the removal
|
||||
# removed exactly one cookie
|
||||
${CMD_PREFIX} ostree --repo=repo remote add-cookie origin 127.0.0.1 / baz badger
|
||||
assert_file_has_content repo/origin.cookies.txt foo.*bar
|
||||
assert_file_has_content repo/origin.cookies.txt baz.*badger
|
||||
${CMD_PREFIX} ostree --repo=repo pull origin main
|
||||
|
||||
echo "ok, second cookie pull succeeded"
|
||||
|
Loading…
x
Reference in New Issue
Block a user