1999-12-29 15:20:26 +01:00
|
|
|
/*****************************************************************************
|
|
|
|
* _ _ ____ _
|
|
|
|
* Project ___| | | | _ \| |
|
|
|
|
* / __| | | | |_) | |
|
|
|
|
* | (__| |_| | _ <| |___
|
|
|
|
* \___|\___/|_| \_\_____|
|
|
|
|
*
|
2001-01-03 10:29:33 +01:00
|
|
|
* Copyright (C) 2000, Daniel Stenberg, <daniel@haxx.se>, et al.
|
1999-12-29 15:20:26 +01:00
|
|
|
*
|
2001-01-03 10:29:33 +01:00
|
|
|
* In order to be useful for every potential user, curl and libcurl are
|
|
|
|
* dual-licensed under the MPL and the MIT/X-derivate licenses.
|
1999-12-29 15:20:26 +01:00
|
|
|
*
|
2001-01-03 10:29:33 +01:00
|
|
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
|
|
|
* copies of the Software, and permit persons to whom the Software is
|
|
|
|
* furnished to do so, under the terms of the MPL or the MIT/X-derivate
|
|
|
|
* licenses. You may pick one of these licenses.
|
1999-12-29 15:20:26 +01:00
|
|
|
*
|
2001-01-03 10:29:33 +01:00
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
1999-12-29 15:20:26 +01:00
|
|
|
*
|
2001-01-03 10:29:33 +01:00
|
|
|
* $Id$
|
|
|
|
*****************************************************************************/
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-05-22 16:12:12 +02:00
|
|
|
/*
|
|
|
|
* The original SSL code was written by
|
|
|
|
* Linas Vepstas <linas@linas.org> and Sampo Kellomaki <sampo@iki.fi>
|
|
|
|
*/
|
|
|
|
|
2000-08-24 16:26:33 +02:00
|
|
|
#include "setup.h"
|
1999-12-29 15:20:26 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "urldata.h"
|
|
|
|
#include "sendf.h"
|
2000-03-03 00:01:35 +01:00
|
|
|
#include "formdata.h" /* for the boundary function */
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
#ifdef USE_SSLEAY
|
2001-03-07 18:08:20 +01:00
|
|
|
#include <openssl/rand.h>
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-04-24 23:41:29 +02:00
|
|
|
/* The last #include file should be: */
|
|
|
|
#ifdef MALLOCDEBUG
|
|
|
|
#include "memdebug.h"
|
|
|
|
#endif
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
static char global_passwd[64];
|
|
|
|
|
|
|
|
static int passwd_callback(char *buf, int num, int verify
|
|
|
|
#if OPENSSL_VERSION_NUMBER >= 0x00904100L
|
|
|
|
/* This was introduced in 0.9.4, we can set this
|
|
|
|
using SSL_CTX_set_default_passwd_cb_userdata()
|
|
|
|
*/
|
|
|
|
, void *userdata
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if(verify)
|
|
|
|
fprintf(stderr, "%s\n", buf);
|
|
|
|
else {
|
|
|
|
if(num > strlen(global_passwd)) {
|
|
|
|
strcpy(buf, global_passwd);
|
|
|
|
return strlen(buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2001-03-06 01:04:58 +01:00
|
|
|
static
|
|
|
|
bool seed_enough(struct connectdata *conn, /* unused for now */
|
|
|
|
int nread)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_RAND_STATUS
|
|
|
|
/* only available in OpenSSL 0.9.5a and later */
|
|
|
|
if(RAND_status())
|
|
|
|
return TRUE;
|
|
|
|
#else
|
|
|
|
if(nread > 500)
|
|
|
|
/* this is a very silly decision to make */
|
|
|
|
return TRUE;
|
|
|
|
#endif
|
|
|
|
return FALSE; /* not enough */
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-01-05 11:11:41 +01:00
|
|
|
static
|
2001-03-06 01:04:58 +01:00
|
|
|
int random_the_seed(struct connectdata *conn)
|
|
|
|
{
|
|
|
|
char *buf = conn->data->buffer; /* point to the big buffer */
|
|
|
|
int nread=0;
|
2001-03-12 16:47:17 +01:00
|
|
|
struct UrlData *data=conn->data;
|
2001-03-06 01:04:58 +01:00
|
|
|
|
|
|
|
/* Q: should we add support for a random file name as a libcurl option?
|
2001-03-12 16:47:17 +01:00
|
|
|
A: Yes, it is here */
|
|
|
|
|
|
|
|
#ifndef RANDOM_FILE
|
|
|
|
/* if RANDOM_FILE isn't defined, we only perform this if an option tells
|
|
|
|
us to! */
|
|
|
|
if(data->ssl.random_file)
|
|
|
|
#define RANDOM_FILE "" /* doesn't matter won't be used */
|
2001-03-06 01:04:58 +01:00
|
|
|
#endif
|
2001-03-12 16:47:17 +01:00
|
|
|
{
|
|
|
|
/* let the option override the define */
|
|
|
|
nread += RAND_load_file((data->ssl.random_file?
|
|
|
|
data->ssl.random_file:RANDOM_FILE),
|
|
|
|
16384);
|
2001-03-06 01:04:58 +01:00
|
|
|
if(seed_enough(conn, nread))
|
|
|
|
return nread;
|
|
|
|
}
|
|
|
|
|
2001-03-12 16:47:17 +01:00
|
|
|
#if defined(HAVE_RAND_EGD)
|
2001-03-06 01:04:58 +01:00
|
|
|
/* only available in OpenSSL 0.9.5 and later */
|
2001-03-12 16:47:17 +01:00
|
|
|
/* EGD_SOCKET is set at configure time or not at all */
|
|
|
|
#ifndef EGD_SOCKET
|
|
|
|
/* If we don't have the define set, we only do this if the egd-option
|
|
|
|
is set */
|
|
|
|
if(data->ssl.egdsocket)
|
|
|
|
#define EGD_SOCKET "" /* doesn't matter won't be used */
|
|
|
|
#endif
|
2001-03-07 18:08:20 +01:00
|
|
|
{
|
2001-03-12 16:47:17 +01:00
|
|
|
/* If there's an option and a define, the option overrides the
|
|
|
|
define */
|
|
|
|
int ret = RAND_egd(data->ssl.egdsocket?data->ssl.egdsocket:EGD_SOCKET);
|
2001-03-07 18:08:20 +01:00
|
|
|
if(-1 != ret) {
|
|
|
|
nread += ret;
|
|
|
|
if(seed_enough(conn, nread))
|
|
|
|
return nread;
|
|
|
|
}
|
2001-03-06 01:04:58 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* If we get here, it means we need to seed the PRNG using a "silly"
|
|
|
|
approach! */
|
|
|
|
#ifdef HAVE_RAND_SCREEN
|
|
|
|
/* This one gets a random value by reading the currently shown screen */
|
|
|
|
RAND_screen();
|
|
|
|
nread = 100; /* just a value */
|
|
|
|
#else
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
char *area = Curl_FormBoundary();
|
|
|
|
if(!area)
|
|
|
|
return 3; /* out of memory */
|
|
|
|
|
|
|
|
len = strlen(area);
|
|
|
|
RAND_seed(area, len);
|
|
|
|
|
|
|
|
free(area); /* now remove the random junk */
|
|
|
|
}
|
2001-03-14 09:20:41 +01:00
|
|
|
#endif
|
2001-03-06 01:04:58 +01:00
|
|
|
|
2001-03-12 16:47:17 +01:00
|
|
|
/* generates a default path for the random seed file */
|
|
|
|
buf[0]=0; /* blank it first */
|
|
|
|
RAND_file_name(buf, BUFSIZE);
|
|
|
|
if ( buf[0] ) {
|
|
|
|
/* we got a file name to try */
|
|
|
|
nread += RAND_load_file(buf, 16384);
|
|
|
|
if(seed_enough(conn, nread))
|
|
|
|
return nread;
|
|
|
|
}
|
|
|
|
|
2001-03-06 01:04:58 +01:00
|
|
|
infof(conn->data, "Your connection is using a weak random seed!\n");
|
|
|
|
return nread;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
int cert_stuff(struct connectdata *conn,
|
2001-01-05 11:11:41 +01:00
|
|
|
char *cert_file,
|
|
|
|
char *key_file)
|
1999-12-29 15:20:26 +01:00
|
|
|
{
|
2001-03-06 01:04:58 +01:00
|
|
|
struct UrlData *data = conn->data;
|
1999-12-29 15:20:26 +01:00
|
|
|
if (cert_file != NULL) {
|
|
|
|
SSL *ssl;
|
|
|
|
X509 *x509;
|
|
|
|
|
|
|
|
if(data->cert_passwd) {
|
|
|
|
/*
|
|
|
|
* If password has been given, we store that in the global
|
|
|
|
* area (*shudder*) for a while:
|
|
|
|
*/
|
|
|
|
strcpy(global_passwd, data->cert_passwd);
|
|
|
|
/* Set passwd callback: */
|
2001-02-20 18:35:51 +01:00
|
|
|
SSL_CTX_set_default_passwd_cb(conn->ssl.ctx, passwd_callback);
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if (SSL_CTX_use_certificate_file(conn->ssl.ctx,
|
1999-12-29 15:20:26 +01:00
|
|
|
cert_file,
|
|
|
|
SSL_FILETYPE_PEM) <= 0) {
|
|
|
|
failf(data, "unable to set certificate file (wrong password?)\n");
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
if (key_file == NULL)
|
|
|
|
key_file=cert_file;
|
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if (SSL_CTX_use_PrivateKey_file(conn->ssl.ctx,
|
1999-12-29 15:20:26 +01:00
|
|
|
key_file,
|
|
|
|
SSL_FILETYPE_PEM) <= 0) {
|
|
|
|
failf(data, "unable to set public key file\n");
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
ssl=SSL_new(conn->ssl.ctx);
|
1999-12-29 15:20:26 +01:00
|
|
|
x509=SSL_get_certificate(ssl);
|
|
|
|
|
|
|
|
if (x509 != NULL)
|
|
|
|
EVP_PKEY_copy_parameters(X509_get_pubkey(x509),
|
|
|
|
SSL_get_privatekey(ssl));
|
|
|
|
SSL_free(ssl);
|
|
|
|
|
|
|
|
/* If we are using DSA, we can copy the parameters from
|
|
|
|
* the private key */
|
|
|
|
|
|
|
|
|
|
|
|
/* Now we know that a key and cert have been set against
|
|
|
|
* the SSL context */
|
2001-02-20 18:35:51 +01:00
|
|
|
if (!SSL_CTX_check_private_key(conn->ssl.ctx)) {
|
1999-12-29 15:20:26 +01:00
|
|
|
failf(data, "Private key does not match the certificate public key\n");
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* erase it now */
|
|
|
|
memset(global_passwd, 0, sizeof(global_passwd));
|
|
|
|
}
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
2001-01-05 11:11:41 +01:00
|
|
|
static
|
1999-12-29 15:20:26 +01:00
|
|
|
int cert_verify_callback(int ok, X509_STORE_CTX *ctx)
|
|
|
|
{
|
|
|
|
X509 *err_cert;
|
|
|
|
char buf[256];
|
|
|
|
|
|
|
|
err_cert=X509_STORE_CTX_get_current_cert(ctx);
|
|
|
|
X509_NAME_oneline(X509_get_subject_name(err_cert),buf,256);
|
|
|
|
|
2001-06-29 09:38:11 +02:00
|
|
|
return ok;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2001-05-30 10:00:29 +02:00
|
|
|
#ifdef USE_SSLEAY
|
|
|
|
/* "global" init done? */
|
|
|
|
static int init_ssl=0;
|
|
|
|
#endif
|
|
|
|
|
2001-05-28 16:12:43 +02:00
|
|
|
/* Global init */
|
|
|
|
void Curl_SSL_init(void)
|
|
|
|
{
|
|
|
|
#ifdef USE_SSLEAY
|
|
|
|
|
|
|
|
/* make sure this is only done once */
|
2001-05-30 10:00:29 +02:00
|
|
|
if(0 != init_ssl)
|
2001-05-28 16:12:43 +02:00
|
|
|
return;
|
|
|
|
|
2001-05-30 10:00:29 +02:00
|
|
|
init_ssl++; /* never again */
|
2001-05-28 16:12:43 +02:00
|
|
|
|
|
|
|
/* Lets get nice error messages */
|
|
|
|
SSL_load_error_strings();
|
|
|
|
|
|
|
|
/* Setup all the global SSL stuff */
|
|
|
|
SSLeay_add_ssl_algorithms();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Global cleanup */
|
|
|
|
void Curl_SSL_cleanup(void)
|
|
|
|
{
|
|
|
|
#ifdef USE_SSLEAY
|
2001-05-30 10:00:29 +02:00
|
|
|
if(init_ssl) {
|
|
|
|
/* only cleanup if we did a previous init */
|
|
|
|
|
|
|
|
/* Free the SSL error strings */
|
|
|
|
ERR_free_strings();
|
2001-05-28 16:12:43 +02:00
|
|
|
|
2001-05-30 10:00:29 +02:00
|
|
|
/* EVP_cleanup() removes all ciphers and digests from the
|
|
|
|
table. */
|
|
|
|
EVP_cleanup();
|
|
|
|
}
|
2001-05-28 16:12:43 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
/* ====================================================== */
|
2001-05-12 11:29:56 +02:00
|
|
|
CURLcode
|
2001-02-20 18:35:51 +01:00
|
|
|
Curl_SSLConnect(struct connectdata *conn)
|
1999-12-29 15:20:26 +01:00
|
|
|
{
|
2001-05-12 11:29:56 +02:00
|
|
|
CURLcode retcode = CURLE_OK;
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
#ifdef USE_SSLEAY
|
2001-02-20 18:35:51 +01:00
|
|
|
struct UrlData *data = conn->data;
|
|
|
|
int err;
|
|
|
|
char * str;
|
|
|
|
SSL_METHOD *req_method;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
/* mark this is being ssl enabled from here on out. */
|
|
|
|
conn->ssl.use = TRUE;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-03-06 01:04:58 +01:00
|
|
|
/* Make funny stuff to get random input */
|
|
|
|
random_the_seed(conn);
|
2000-03-03 00:01:35 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
switch(data->ssl.version) {
|
|
|
|
default:
|
|
|
|
req_method = SSLv23_client_method();
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
req_method = SSLv2_client_method();
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
req_method = SSLv3_client_method();
|
|
|
|
break;
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
conn->ssl.ctx = SSL_CTX_new(req_method);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if(!conn->ssl.ctx) {
|
|
|
|
failf(data, "SSL: couldn't create a context!");
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_OUT_OF_MEMORY;
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if(data->cert) {
|
2001-03-06 01:04:58 +01:00
|
|
|
if (!cert_stuff(conn, data->cert, data->cert)) {
|
2001-06-12 20:22:52 +02:00
|
|
|
/* failf() is already done in cert_stuff() */
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_SSL_CONNECT_ERROR;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if(data->ssl.verifypeer){
|
|
|
|
SSL_CTX_set_verify(conn->ssl.ctx,
|
|
|
|
SSL_VERIFY_PEER|SSL_VERIFY_FAIL_IF_NO_PEER_CERT|
|
|
|
|
SSL_VERIFY_CLIENT_ONCE,
|
|
|
|
cert_verify_callback);
|
|
|
|
if (!SSL_CTX_load_verify_locations(conn->ssl.ctx,
|
|
|
|
data->ssl.CAfile,
|
|
|
|
data->ssl.CApath)) {
|
|
|
|
failf(data,"error setting cerficate verify locations\n");
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_SSL_CONNECT_ERROR;
|
2000-10-30 12:53:40 +01:00
|
|
|
}
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
SSL_CTX_set_verify(conn->ssl.ctx, SSL_VERIFY_NONE, cert_verify_callback);
|
2000-10-30 12:53:40 +01:00
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
/* Lets make an SSL structure */
|
|
|
|
conn->ssl.handle = SSL_new (conn->ssl.ctx);
|
|
|
|
SSL_set_connect_state (conn->ssl.handle);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
conn->ssl.server_cert = 0x0;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
/* pass the raw socket into the SSL layers */
|
|
|
|
SSL_set_fd (conn->ssl.handle, conn->firstsocket);
|
|
|
|
err = SSL_connect (conn->ssl.handle);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if (-1 == err) {
|
|
|
|
err = ERR_get_error();
|
|
|
|
failf(data, "SSL: %s", ERR_error_string(err, NULL));
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_SSL_CONNECT_ERROR;
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
/* Informational message */
|
|
|
|
infof (data, "SSL connection using %s\n",
|
|
|
|
SSL_get_cipher(conn->ssl.handle));
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
/* Get server's certificate (note: beware of dynamic allocation) - opt */
|
|
|
|
/* major serious hack alert -- we should check certificates
|
|
|
|
* to authenticate the server; otherwise we risk man-in-the-middle
|
|
|
|
* attack
|
|
|
|
*/
|
|
|
|
|
|
|
|
conn->ssl.server_cert = SSL_get_peer_certificate (conn->ssl.handle);
|
|
|
|
if(!conn->ssl.server_cert) {
|
|
|
|
failf(data, "SSL: couldn't get peer certificate!");
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_SSL_PEER_CERTIFICATE;
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
|
|
|
infof (data, "Server certificate:\n");
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
str = X509_NAME_oneline (X509_get_subject_name (conn->ssl.server_cert),
|
|
|
|
NULL, 0);
|
|
|
|
if(!str) {
|
|
|
|
failf(data, "SSL: couldn't get X509-subject!");
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_SSL_CONNECT_ERROR;
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
|
|
|
infof(data, "\t subject: %s\n", str);
|
|
|
|
CRYPTO_free(str);
|
|
|
|
|
|
|
|
str = X509_NAME_oneline (X509_get_issuer_name (conn->ssl.server_cert),
|
|
|
|
NULL, 0);
|
|
|
|
if(!str) {
|
|
|
|
failf(data, "SSL: couldn't get X509-issuer name!");
|
2001-05-12 11:29:56 +02:00
|
|
|
return CURLE_SSL_CONNECT_ERROR;
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
|
|
|
infof(data, "\t issuer: %s\n", str);
|
|
|
|
CRYPTO_free(str);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
/* We could do all sorts of certificate verification stuff here before
|
|
|
|
deallocating the certificate. */
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
if(data->ssl.verifypeer) {
|
|
|
|
data->ssl.certverifyresult=SSL_get_verify_result(conn->ssl.handle);
|
2001-05-12 11:29:56 +02:00
|
|
|
failf(data, "SSL certificate verify result: %d\n",
|
|
|
|
data->ssl.certverifyresult);
|
|
|
|
retcode = CURLE_SSL_PEER_CERTIFICATE;
|
2001-02-20 18:35:51 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
data->ssl.certverifyresult=0;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2001-02-20 18:35:51 +01:00
|
|
|
X509_free(conn->ssl.server_cert);
|
1999-12-29 15:20:26 +01:00
|
|
|
#else /* USE_SSLEAY */
|
2001-02-20 18:35:51 +01:00
|
|
|
/* this is for "-ansi -Wall -pedantic" to stop complaining! (rabe) */
|
2001-03-14 11:15:42 +01:00
|
|
|
(void) conn;
|
1999-12-29 15:20:26 +01:00
|
|
|
#endif
|
2001-05-12 11:29:56 +02:00
|
|
|
return retcode;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|