25#include <gnutls/x509.h>
30#include <sys/socket.h>
38#define G_LOG_DOMAIN "libgvm util"
44 const gchar *, gnutls_session_t *,
45 gnutls_certificate_credentials_t *);
60 if (fcntl (client_connection->
socket, F_SETFL, O_NONBLOCK) == -1)
62 g_warning (
"%s: failed to set server socket flag: %s\n", __func__,
67 if (shutdown (client_connection->
socket, SHUT_RDWR) == -1)
69 if (errno == ENOTCONN)
71 g_warning (
"%s: failed to shutdown server socket: %s\n", __func__,
76 if (close (client_connection->
socket) == -1)
78 g_warning (
"%s: failed to close server socket: %s\n", __func__,
94 if (client_connection->
tls)
116 ret = gnutls_certificate_verify_peers2 (session, &status);
119 g_warning (
"%s: failed to verify peers: %s", __func__,
120 gnutls_strerror (ret));
124 if (status & GNUTLS_CERT_INVALID)
125 g_warning (
"%s: the certificate is not trusted", __func__);
127 if (status & GNUTLS_CERT_SIGNER_NOT_CA)
128 g_warning (
"%s: the certificate's issuer is not a CA", __func__);
130 if (status & GNUTLS_CERT_INSECURE_ALGORITHM)
131 g_warning (
"%s: the certificate was signed using an insecure algorithm",
134 if (status & GNUTLS_CERT_SIGNER_NOT_FOUND)
135 g_warning (
"%s: the certificate hasn't got a known issuer", __func__);
137 if (status & GNUTLS_CERT_REVOKED)
138 g_warning (
"%s: the certificate has been revoked", __func__);
140 if (status & GNUTLS_CERT_EXPIRED)
141 g_warning (
"%s: the certificate has expired", __func__);
143 if (status & GNUTLS_CERT_NOT_ACTIVATED)
144 g_warning (
"%s: the certificate is not yet activated", __func__);
167 if (!(f = fopen (file,
"r")) || fseek (f, 0, SEEK_END) != 0
168 || (filelen = ftell (f)) < 0 || fseek (f, 0, SEEK_SET) != 0
169 || !(ptr = g_malloc0 ((
size_t) filelen))
170 || fread (ptr, 1, (
size_t) filelen, f) < (
size_t) filelen)
177 loaded_file->data = ptr;
178 loaded_file->size = filelen;
262 const gnutls_datum_t *req_ca_rdn,
int nreqs,
263 const gnutls_pk_algorithm_t *sign_algos,
264 int sign_algos_length, gnutls_retr2_st *st)
268 static gnutls_x509_crt_t crt;
269 static gnutls_x509_privkey_t key;
275 (void) sign_algos_length;
278 gnutls_x509_crt_init (&crt);
279 ret = gnutls_x509_crt_import (crt, &data, GNUTLS_X509_FMT_PEM);
283 st->cert.x509 = &crt;
284 st->cert_type = GNUTLS_CRT_X509;
289 gnutls_x509_privkey_init (&key);
290 ret = gnutls_x509_privkey_import (key, &data, GNUTLS_X509_FMT_PEM);
295 st->key_type = GNUTLS_PRIVKEY_X509;
314 const char *ca_mem,
const char *pub_mem,
315 const char *priv_mem,
int verify)
319 struct addrinfo address_hints;
320 struct addrinfo *addresses, *address;
324 gnutls_certificate_credentials_t credentials;
327 if (port < 1 || port > 65535)
329 g_warning (
"Failed to create client TLS session. "
338 g_warning (
"Failed to create client TLS session. Invalid host %s", host);
348 g_warning (
"Failed to create client TLS session.");
352 if (ca_mem && pub_mem && priv_mem)
357 gnutls_certificate_set_retrieve_function (credentials,
363 port_string = g_strdup_printf (
"%i", port);
367 memset (&address_hints, 0,
sizeof (address_hints));
368 address_hints.ai_family = AF_UNSPEC;
369 address_hints.ai_socktype = SOCK_STREAM;
370 address_hints.ai_protocol = 0;
372 if (getaddrinfo (host, port_string, &address_hints, &addresses))
374 g_free (port_string);
375 g_warning (
"Failed to get server addresses for %s: %s", host,
376 gai_strerror (errno));
377 gnutls_deinit (*session);
378 gnutls_certificate_free_credentials (credentials);
381 g_free (port_string);
385 for (address = addresses; address; address = address->ai_next)
389 if (address->ai_family == AF_INET6)
390 server_socket = socket (PF_INET6, SOCK_STREAM, 0);
392 server_socket = socket (PF_INET, SOCK_STREAM, 0);
393 if (server_socket == -1)
395 g_warning (
"Failed to create server socket");
396 freeaddrinfo (addresses);
397 gnutls_deinit (*session);
398 gnutls_certificate_free_credentials (credentials);
404 if (connect (server_socket, address->ai_addr, address->ai_addrlen) == -1)
406 close (server_socket);
412 freeaddrinfo (addresses);
416 g_warning (
"Failed to connect to server");
417 gnutls_deinit (*session);
418 gnutls_certificate_free_credentials (credentials);
422 g_debug (
" Connected to server '%s' port %d.", host, port);
430 close (server_socket);
431 gnutls_deinit (*session);
432 gnutls_certificate_free_credentials (credentials);
434 close (server_socket);
439 close (server_socket);
443 return server_socket;
462 int port,
const char *ca_mem,
const char *pub_mem,
463 const char *priv_mem)
466 ca_mem && pub_mem && priv_mem);
524 unsigned int retries;
526 gnutls_transport_set_ptr (*session,
527 (gnutls_transport_ptr_t) GSIZE_TO_POINTER (socket));
532 int ret = gnutls_handshake (*session);
535 if (ret == GNUTLS_E_AGAIN || ret == GNUTLS_E_INTERRUPTED)
538 usleep (MIN ((retries - 10) * 10000, 5000000));
543 g_debug (
"Failed to shake hands with server '%s' port %d: %s", host,
544 port, gnutls_strerror (ret));
546 g_debug (
"Failed to shake hands with peer: %s", gnutls_strerror (ret));
547 if (shutdown (socket, SHUT_RDWR) == -1)
548 g_debug (
"Failed to shutdown server socket");
552 g_debug (
" Shook hands with server '%s' port %d.", host, port);
554 g_debug (
" Shook hands with peer.");
590 va_list ap,
int quiet)
595 left = vasprintf (&
string, fmt, ap);
605 g_debug (
" send %d from %.*s[...]", left, left < 30 ? left : 30,
607 count = gnutls_record_send (*session,
string, left);
610 if (count == GNUTLS_E_INTERRUPTED)
613 if (count == GNUTLS_E_REHANDSHAKE)
617 g_message (
" %s rehandshake", __func__);
620 g_warning (
"Failed to write to server: %s", gnutls_strerror (count));
628 g_debug (
"= server closed");
633 g_debug (
"=> %.*s", (
int) count,
string);
659 char *string_start, *string;
662 left = vasprintf (&
string, fmt, ap);
666 string_start = string;
672 g_debug (
" send %d from %.*s[...]", left, left < 30 ? left : 30,
674 count = write (socket,
string, left);
677 if (errno == EINTR || errno == EAGAIN)
679 g_warning (
"Failed to write to server: %s", strerror (errno));
684 g_debug (
"=> %.*s", (
int) count,
string);
693 g_free (string_start);
710 va_list ap,
int quiet)
808 va_start (ap, format);
828 va_start (ap, format);
848 va_start (ap, format);
869 va_start (ap, format);
892 va_start (ap, format);
893 msg = g_markup_vprintf_escaped (format, ap);
917 va_start (ap, format);
918 msg = g_markup_vprintf_escaped (format, ap);
944 va_start (ap, format);
945 msg = g_markup_vprintf_escaped (format, ap);
966 const char *format, ...)
972 va_start (ap, format);
973 msg = g_markup_vprintf_escaped (format, ap);
991 gcry_control (GCRYCTL_ENABLE_QUICK_RANDOM, 0);
994 if (gnutls_global_init ())
996 g_warning (
"Failed to initialize GNUTLS.");
1000 if (gnutls_certificate_allocate_credentials (server_credentials))
1002 g_warning (
"%s: failed to allocate server credentials\n", __func__);
1021 gnutls_session_t *server_session,
1022 gnutls_certificate_credentials_t *server_credentials)
1026 if (gnutls_init (server_session, end_type))
1028 g_warning (
"%s: failed to initialise server session\n", __func__);
1040 err = gnutls_priority_set_direct (*server_session,
1041 priority ? priority :
"NORMAL", NULL);
1044 g_warning (
"%s: failed to set tls priorities: %s\n", __func__,
1045 gnutls_strerror (err));
1046 gnutls_deinit (*server_session);
1050 if (gnutls_credentials_set (*server_session, GNUTLS_CRD_CERTIFICATE,
1051 *server_credentials))
1053 g_warning (
"%s: failed to set server credentials\n", __func__);
1054 gnutls_deinit (*server_session);
1058 if (end_type == GNUTLS_SERVER)
1059 gnutls_certificate_server_set_request (*server_session,
1060 GNUTLS_CERT_REQUEST);
1080 const gchar *ca_cert_file,
const gchar *cert_file,
1081 const gchar *key_file, gnutls_session_t *server_session,
1082 gnutls_certificate_credentials_t *server_credentials)
1087 if (cert_file && key_file)
1091 ret = gnutls_certificate_set_x509_key_file (
1092 *server_credentials, cert_file, key_file, GNUTLS_X509_FMT_PEM);
1095 g_warning (
"%s: failed to set credentials key file: %s\n", __func__,
1096 gnutls_strerror (ret));
1097 g_warning (
"%s: cert file: %s\n", __func__, cert_file);
1098 g_warning (
"%s: key file : %s\n", __func__, key_file);
1099 gnutls_certificate_free_credentials (*server_credentials);
1108 ret = gnutls_certificate_set_x509_trust_file (
1109 *server_credentials, ca_cert_file, GNUTLS_X509_FMT_PEM);
1112 g_warning (
"%s: failed to set credentials trust file: %s\n", __func__,
1113 gnutls_strerror (ret));
1114 g_warning (
"%s: trust file: %s\n", __func__, ca_cert_file);
1115 gnutls_certificate_free_credentials (*server_credentials);
1121 server_credentials))
1123 gnutls_certificate_free_credentials (*server_credentials);
1145 gchar *key_file, gnutls_session_t *server_session,
1146 gnutls_certificate_credentials_t *server_credentials)
1149 server_session, server_credentials);
1167 const char *pub_key,
const char *priv_key,
1168 gnutls_session_t *session,
1169 gnutls_certificate_credentials_t *credentials)
1174 if (pub_key && priv_key)
1177 gnutls_datum_t pub, priv;
1179 pub.data = (
void *) pub_key;
1180 pub.size = strlen (pub_key);
1181 priv.data = (
void *) priv_key;
1182 priv.size = strlen (priv_key);
1184 ret = gnutls_certificate_set_x509_key_mem (*credentials, &pub, &priv,
1185 GNUTLS_X509_FMT_PEM);
1188 g_warning (
"%s: %s\n", __func__, gnutls_strerror (ret));
1196 gnutls_datum_t data;
1198 data.data = (
void *) ca_cert;
1199 data.size = strlen (ca_cert);
1200 ret = gnutls_certificate_set_x509_trust_mem (*credentials, &data,
1201 GNUTLS_X509_FMT_PEM);
1204 g_warning (
"%s: %s\n", __func__, gnutls_strerror (ret));
1205 gnutls_certificate_free_credentials (*credentials);
1212 gnutls_certificate_free_credentials (*credentials);
1229 const char *dhparams_file)
1232 gnutls_datum_t data;
1234 if (!creds || !dhparams_file)
1241#ifndef __clang_analyzer__
1243 gnutls_dh_params_t params = g_malloc0 (
sizeof (gnutls_dh_params_t));
1244 ret = gnutls_dh_params_import_pkcs3 (params, &data, GNUTLS_X509_FMT_PEM);
1252 gnutls_certificate_set_dh_params (creds, params);
1272 gnutls_certificate_credentials_t server_credentials)
1276 if (fcntl (server_socket, F_SETFL, O_NONBLOCK) == -1)
1278 g_warning (
"%s: failed to set server socket flag: %s\n", __func__,
1285 int ret = gnutls_bye (server_session, GNUTLS_SHUT_WR);
1286 if (ret == GNUTLS_E_AGAIN || ret == GNUTLS_E_INTERRUPTED)
1292 g_debug (
" Failed to gnutls_bye: %s\n",
1293 gnutls_strerror ((
int) ret));
1307 if (server_credentials)
1309 if (close (server_socket) == -1)
1311 g_warning (
"%s: failed to close server socket: %s\n", __func__,
1315 gnutls_deinit (server_session);
1316 gnutls_certificate_free_credentials (server_credentials);
1320 gnutls_deinit (server_session);
1321 close (server_socket);
1324 gnutls_global_deinit ();
int gvm_get_host_type(const gchar *str_stripped)
Determines the host type in a buffer.
Definition hosts.c:814
Protos and data structures for Hosts collections and single hosts objects.
host_type
Definition hosts.h:34
@ HOST_TYPE_NAME
Definition hosts.h:35
@ HOST_TYPE_IPV6
Definition hosts.h:40
@ HOST_TYPE_IPV4
Definition hosts.h:36
static int close_unix(gvm_connection_t *client_connection)
Close UNIX socket connection.
Definition serverutils.c:57
static int server_new_internal(unsigned int, const char *, const gchar *, const gchar *, const gchar *, gnutls_session_t *, gnutls_certificate_credentials_t *)
Make a session for connecting to a server.
Definition serverutils.c:1079
int set_gnutls_dhparams(gnutls_certificate_credentials_t creds, const char *dhparams_file)
Set a gnutls session's Diffie-Hellman parameters.
Definition serverutils.c:1228
static const char * get_cert_pub_mem()
Get public certificate from cert_pub_mem.
Definition serverutils.c:240
int gvm_server_new(unsigned int end_type, gchar *ca_cert_file, gchar *cert_file, gchar *key_file, gnutls_session_t *server_session, gnutls_certificate_credentials_t *server_credentials)
Make a session for connecting to a server.
Definition serverutils.c:1144
int gvm_connection_sendf(gvm_connection_t *connection, const char *format,...)
Format and send a string to the server.
Definition serverutils.c:823
int gvm_server_open_with_cert(gnutls_session_t *session, const char *host, int port, const char *ca_mem, const char *pub_mem, const char *priv_mem)
Connect to the server using a given host, port and cert.
Definition serverutils.c:461
int gvm_server_sendf_xml_quiet(gnutls_session_t *session, const char *format,...)
Format and send an XML string to the server.
Definition serverutils.c:938
static void set_cert_priv_mem(const char *data)
Save cert_priv_mem with private certificate.
Definition serverutils.c:216
int gvm_server_vsendf(gnutls_session_t *session, const char *fmt, va_list ap)
Send a string to the server.
Definition serverutils.c:727
static const char * get_cert_priv_mem()
Get private certificate from cert_priv_mem.
Definition serverutils.c:229
int gvm_server_close(int socket, gnutls_session_t session)
Close a server connection and its socket.
Definition serverutils.c:493
int gvm_server_open(gnutls_session_t *session, const char *host, int port)
Connect to the server using a given host and port.
Definition serverutils.c:479
static int gvm_connection_vsendf_internal(gvm_connection_t *connection, const char *fmt, va_list ap, int quiet)
Send a string to the connection.
Definition serverutils.c:709
static int client_cert_callback(gnutls_session_t session, const gnutls_datum_t *req_ca_rdn, int nreqs, const gnutls_pk_algorithm_t *sign_algos, int sign_algos_length, gnutls_retr2_st *st)
Callback function to be called in order to retrieve the certificate to be used in the handshake.
Definition serverutils.c:261
int gvm_connection_sendf_xml(gvm_connection_t *connection, const char *format,...)
Format and send an XML string to the server.
Definition serverutils.c:911
static char * cert_pub_mem
Definition serverutils.c:195
static int gvm_server_sendf_quiet(gnutls_session_t *session, const char *format,...)
Format and send a string to the server.
Definition serverutils.c:843
void gvm_connection_close(gvm_connection_t *connection)
Close a server connection and its socket.
Definition serverutils.c:504
static int server_attach_internal(int, gnutls_session_t *, const char *, int)
Attach a socket to a session, and shake hands with the peer.
Definition serverutils.c:521
static int server_new_gnutls_init(gnutls_certificate_credentials_t *server_credentials)
Initialize a server session.
Definition serverutils.c:988
int gvm_socket_vsendf(int socket, const char *fmt, va_list ap)
Send a string to the server.
Definition serverutils.c:742
int load_gnutls_file(const char *file, gnutls_datum_t *loaded_file)
Loads a file's data into gnutls_datum_t struct.
Definition serverutils.c:161
int gvm_server_new_mem(unsigned int end_type, const char *ca_cert, const char *pub_key, const char *priv_key, gnutls_session_t *session, gnutls_certificate_credentials_t *credentials)
Make a session for connecting to a server, with certificates stored in memory.
Definition serverutils.c:1166
static char * cert_priv_mem
Definition serverutils.c:196
void gvm_connection_free(gvm_connection_t *client_connection)
Free connection.
Definition serverutils.c:92
int gvm_server_sendf_xml(gnutls_session_t *session, const char *format,...)
Format and send an XML string to the server.
Definition serverutils.c:886
static int gvm_connection_sendf_quiet(gvm_connection_t *connection, const char *format,...)
Format and send a string to the server.
Definition serverutils.c:863
static int unix_vsendf_internal(int socket, const char *fmt, va_list ap, int quiet)
Send a string to the server.
Definition serverutils.c:657
static int server_new_gnutls_set(unsigned int end_type, const char *priority, gnutls_session_t *server_session, gnutls_certificate_credentials_t *server_credentials)
Set the server credentials.
Definition serverutils.c:1020
static int gvm_connection_vsendf(gvm_connection_t *connection, const char *fmt, va_list ap)
Send a string to the server.
Definition serverutils.c:757
void unload_gnutls_file(gnutls_datum_t *data)
Unloads a gnutls_datum_t struct's data.
Definition serverutils.c:189
static int gvm_connection_vsendf_quiet(gvm_connection_t *connection, const char *fmt, va_list ap)
Send a string to the server, refraining from logging besides warnings.
Definition serverutils.c:788
int gvm_server_verify(gnutls_session_t session)
Verify certificate.
Definition serverutils.c:111
static int gvm_server_vsendf_internal(gnutls_session_t *session, const char *fmt, va_list ap, int quiet)
Send a string to the server.
Definition serverutils.c:589
static int gvm_server_vsendf_quiet(gnutls_session_t *session, const char *fmt, va_list ap)
Send a string to the server, refraining from logging besides warnings.
Definition serverutils.c:773
int gvm_server_sendf(gnutls_session_t *session, const char *format,...)
Format and send a string to the server.
Definition serverutils.c:803
int gvm_server_free(int server_socket, gnutls_session_t server_session, gnutls_certificate_credentials_t server_credentials)
Cleanup a server session.
Definition serverutils.c:1271
int gvm_server_attach(int socket, gnutls_session_t *session)
Attach a socket to a session, and shake hands with the peer.
Definition serverutils.c:569
int gvm_server_open_verify(gnutls_session_t *session, const char *host, int port, const char *ca_mem, const char *pub_mem, const char *priv_mem, int verify)
Connect to the server using a given host, port and cert.
Definition serverutils.c:313
static void set_cert_pub_mem(const char *data)
Save cert_pub_mem with public certificate.
Definition serverutils.c:204
int gvm_connection_sendf_xml_quiet(gvm_connection_t *connection, const char *format,...)
Format and send an XML string to the server.
Definition serverutils.c:965
GnuTLS based functions for server communication - header file.
Connection.
Definition serverutils.h:30
int tls
Whether uses TCP-TLS (vs UNIX socket).
Definition serverutils.h:31
gnutls_certificate_credentials_t credentials
Credentials.
Definition serverutils.h:34
int socket
Socket.
Definition serverutils.h:32
gnutls_session_t session
Session.
Definition serverutils.h:33