Improve GTK-DOC coverage.
[gnutls.git] / tests / openpgpself.c
blob67b84b7afb79e9f25efa69b7950d2a18a069f633
1 /*
2 * Copyright (C) 2004, 2005, 2006, 2007, 2008, 2009, 2010 Free Software
3 * Foundation, Inc.
5 * Author: Simon Josefsson
7 * This file is part of GNUTLS.
9 * GNUTLS is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 3 of the License, or
12 * (at your option) any later version.
14 * GNUTLS is distributed in the hope that it will be useful, but
15 * WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with GNUTLS; if not, write to the Free Software Foundation,
21 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 /* Parts copied from GnuTLS example programs. */
26 #ifdef HAVE_CONFIG_H
27 # include <config.h>
28 #endif
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include <string.h>
33 #include <sys/types.h>
34 #include <netinet/in.h>
35 #include <sys/socket.h>
36 #include <sys/wait.h>
37 #include <arpa/inet.h>
38 #include <unistd.h>
39 #include <gnutls/gnutls.h>
40 #include <gnutls/openpgp.h>
42 #include "utils.h"
44 #include "ex-session-info.c"
45 #include "ex-x509-info.c"
46 #include "tcp.c"
48 pid_t child;
50 static void
51 tls_log_func (int level, const char *str)
53 fprintf (stderr, "%s |<%d>| %s", child ? "server" : "client", level, str);
56 /* A very basic TLS client, with anonymous authentication.
59 #define MAX_BUF 1024
60 #define MSG "Hello TLS"
62 static unsigned char cert_txt[] =
63 "-----BEGIN PGP PUBLIC KEY BLOCK-----\n"
64 "Version: GnuPG v1.0.6 (GNU/Linux)\n"
65 "Comment: For info see http://www.gnupg.org\n"
66 "\n"
67 "mQGiBDxnlY0RBACAsWUhi/goBvpvTBgL8fFPwBAuD04VYFEtC7+4pBp6kFsHjUR7\n"
68 "TTUkBsOk2PvMHrDdv0+C4x2CH8YGP1e+O0f2yLWk8Uu+kkF12yiqbbvDEiCdeJT6\n"
69 "c3vIstY8vJ9Jso5g/LB8Xggq88R7jXFS3hH+WC5v/6P6SARfzXl457cVewCgvxSf\n"
70 "Gsm9mFospJ0B3RGyg5MB0d8D/RQQryJCGdR2nLe4VfctPL2QBD/1XhtubqEbetaV\n"
71 "PxssqrJdA+eplBRT7UHokSBahM8gmSmNuSrLDujPfEtaMg6YIkB+Kq0VeJLE0cXT\n"
72 "ZIH29KJlI/qk1xG4K7D6B0cKaHC/L4BIoKcQLJzfTIPw3frS4jVeNaQZNHSVqZ8/\n"
73 "VmOMA/9rkNtccQ4RVd9WTFoHKvT4vfiISEOIzKGmcBY9Hymq7MCci3mNe4CDImkv\n"
74 "ZgnjDlJAM91CX1ODthPLBqvyhnMhhxDnaDl4Nh42uPMSr9JEW2IwoIbFne10ihGT\n"
75 "O4lBS1C28UfSGEMm/8JBMtxAjbYy3BYzUtCMA+bGBG6Voe5i5LQlRHIuIFdobyAo\n"
76 "Tm8gY29tbWVudHMpIDx3aG9Ad2hvaXMub3JnPohdBBMRAgAdBQI8Z5WNBQkDwmcA\n"
77 "BQsHCgMEAxUDAgMWAgECF4AACgkQNRRc6qfZPD+WWACfeJnLyfbpTDB7mDh3aATb\n"
78 "+0PXz28AoKRdApBVM6Bty+vWyXH6HfF6ZTj+\n"
79 "=m8dH\n" "-----END PGP PUBLIC KEY BLOCK-----\n";
80 const gnutls_datum_t cert = { cert_txt, sizeof (cert_txt) };
82 static unsigned char key_txt[] =
83 "-----BEGIN PGP PRIVATE KEY BLOCK-----\n"
84 "Version: GnuPG v1.0.6 (GNU/Linux)\n"
85 "Comment: For info see http://www.gnupg.org\n"
86 "\n"
87 "lQG7BDxnlY0RBACAsWUhi/goBvpvTBgL8fFPwBAuD04VYFEtC7+4pBp6kFsHjUR7\n"
88 "TTUkBsOk2PvMHrDdv0+C4x2CH8YGP1e+O0f2yLWk8Uu+kkF12yiqbbvDEiCdeJT6\n"
89 "c3vIstY8vJ9Jso5g/LB8Xggq88R7jXFS3hH+WC5v/6P6SARfzXl457cVewCgvxSf\n"
90 "Gsm9mFospJ0B3RGyg5MB0d8D/RQQryJCGdR2nLe4VfctPL2QBD/1XhtubqEbetaV\n"
91 "PxssqrJdA+eplBRT7UHokSBahM8gmSmNuSrLDujPfEtaMg6YIkB+Kq0VeJLE0cXT\n"
92 "ZIH29KJlI/qk1xG4K7D6B0cKaHC/L4BIoKcQLJzfTIPw3frS4jVeNaQZNHSVqZ8/\n"
93 "VmOMA/9rkNtccQ4RVd9WTFoHKvT4vfiISEOIzKGmcBY9Hymq7MCci3mNe4CDImkv\n"
94 "ZgnjDlJAM91CX1ODthPLBqvyhnMhhxDnaDl4Nh42uPMSr9JEW2IwoIbFne10ihGT\n"
95 "O4lBS1C28UfSGEMm/8JBMtxAjbYy3BYzUtCMA+bGBG6Voe5i5AAAnjMCLPrxGdgE\n"
96 "I0xXdwCQ4Sh2diNECAj9JiM6RFNBX2ZhY3RvcjoAAK9cun7/j4AUMmdvIy5UMJph\n"
97 "A6eq6atP/SYjOkRTQV9mYWN0b3I6AACvVjUuomodmmyCggPHWdeVSzpX3ODEHf0m\n"
98 "IzpEU0FfZmFjdG9yOgAAr2Iv9H2aSH+vJKGYW/BO4ehQwwFck7u0JURyLiBXaG8g\n"
99 "KE5vIGNvbW1lbnRzKSA8d2hvQHdob2lzLm9yZz6IXQQTEQIAHQUCPGeVjQUJA8Jn\n"
100 "AAULBwoDBAMVAwIDFgIBAheAAAoJEDUUXOqn2Tw/llgAnjBPQdWxIqBCQGlcI2K/\n"
101 "gLkZR1ARAJ9kaAeJYERc0bV/vlm0ot7UDdr+bQ==\n"
102 "=4M0W\n" "-----END PGP PRIVATE KEY BLOCK-----\n";
103 const gnutls_datum_t key = { key_txt, sizeof (key_txt) };
105 static void
106 client (void)
108 int ret, sd, ii;
109 gnutls_session_t session;
110 char buffer[MAX_BUF + 1];
111 gnutls_certificate_credentials_t xcred;
113 gnutls_global_init ();
115 gnutls_global_set_log_function (tls_log_func);
116 gnutls_global_set_log_level (2);
118 gnutls_certificate_allocate_credentials (&xcred);
120 /* sets the trusted cas file
122 success ("Setting key files...\n");
124 ret = gnutls_certificate_set_openpgp_key_mem (xcred, &cert, &key,
125 GNUTLS_OPENPGP_FMT_BASE64);
126 if (ret < 0)
128 fail ("Could not set key files...\n");
131 /* Initialize TLS session
133 gnutls_init (&session, GNUTLS_CLIENT);
135 /* Use default priorities */
136 gnutls_set_default_priority (session);
138 /* put the x509 credentials to the current session
140 gnutls_credentials_set (session, GNUTLS_CRD_CERTIFICATE, xcred);
142 /* connect to the peer
144 success ("Connecting...\n");
145 sd = tcp_connect ();
147 gnutls_transport_set_ptr (session, (gnutls_transport_ptr_t) sd);
149 /* Perform the TLS handshake
151 ret = gnutls_handshake (session);
153 if (ret < 0)
155 fail ("client: Handshake failed\n");
156 gnutls_perror (ret);
157 goto end;
159 else
161 success ("client: Handshake was completed\n");
164 success ("client: TLS version is: %s\n",
165 gnutls_protocol_get_name (gnutls_protocol_get_version (session)));
167 /* see the Getting peer's information example */
168 print_info (session);
170 gnutls_record_send (session, MSG, strlen (MSG));
172 ret = gnutls_record_recv (session, buffer, MAX_BUF);
173 if (ret == 0)
175 success ("client: Peer has closed the TLS connection\n");
176 goto end;
178 else if (ret < 0)
180 fail ("client: Error: %s\n", gnutls_strerror (ret));
181 goto end;
184 printf ("- Received %d bytes: ", ret);
185 for (ii = 0; ii < ret; ii++)
187 fputc (buffer[ii], stdout);
189 fputs ("\n", stdout);
191 gnutls_bye (session, GNUTLS_SHUT_RDWR);
193 end:
195 tcp_close (sd);
197 gnutls_deinit (session);
199 gnutls_certificate_free_credentials (xcred);
201 gnutls_global_deinit ();
204 /* This is a sample TLS 1.0 echo server, using X.509 authentication.
207 #define SA struct sockaddr
208 #define MAX_BUF 1024
209 #define PORT 5556 /* listen to 5556 port */
210 #define DH_BITS 1024
212 /* These are global */
213 gnutls_certificate_credentials_t pgp_cred;
215 static gnutls_session_t
216 initialize_tls_session (void)
218 gnutls_session_t session;
220 gnutls_init (&session, GNUTLS_SERVER);
222 /* avoid calling all the priority functions, since the defaults
223 * are adequate.
225 gnutls_set_default_priority (session);
227 gnutls_credentials_set (session, GNUTLS_CRD_CERTIFICATE, pgp_cred);
229 /* request client certificate if any.
231 gnutls_certificate_server_set_request (session, GNUTLS_CERT_REQUEST);
233 gnutls_dh_set_prime_bits (session, DH_BITS);
235 return session;
238 static gnutls_dh_params_t dh_params;
240 static int
241 generate_dh_params (void)
243 const gnutls_datum_t p3 = { (char *) pkcs3, strlen (pkcs3) };
244 /* Generate Diffie-Hellman parameters - for use with DHE
245 * kx algorithms. These should be discarded and regenerated
246 * once a day, once a week or once a month. Depending on the
247 * security requirements.
249 gnutls_dh_params_init (&dh_params);
250 return gnutls_dh_params_import_pkcs3 (dh_params, &p3, GNUTLS_X509_FMT_PEM);
253 int err, listen_sd, i;
254 int sd, ret;
255 struct sockaddr_in sa_serv;
256 struct sockaddr_in sa_cli;
257 int client_len;
258 char topbuf[512];
259 gnutls_session_t session;
260 char buffer[MAX_BUF + 1];
261 int optval = 1;
263 static unsigned char server_crt_txt[] =
264 "-----BEGIN PGP PUBLIC KEY BLOCK-----\n"
265 "Version: GnuPG v1.4.6 (GNU/Linux)\n"
266 "\n"
267 "mNEER2PogwEGINdIR4u5PR4SwADWwj/ztgtoi7XVbmlfbQTHpBYFxTSC88pISSNy\n"
268 "V/rgnlqunYP77F7aHL4KUReN3v9sKw01xSGEfox/JmlqUUg6CVvTjdeLfkuVIBnH\n"
269 "j+2KMlaxezp7IxtPaTXpXcSf8iOuVq7UX7p6tKbppKXO5GgmfA88VUVvGBs1/PQp\n"
270 "WKQdGrj+6I3RRmDN/hna1jGU/N23230Hbx+bu7g9cviiSh10ri7rdDhVJ67tRkRG\n"
271 "Usy3XO6dWC7EmzZlEO8AEQEAAbQQdGVzdDMuZ251dGxzLm9yZ4kBAAQTAQIAJgUC\n"
272 "R2PogwIbAwUJCWYBgAYLCQgHAwIEFQIIAwQWAgMBAh4BAheAAAoJEKAh4/gImZBR\n"
273 "96QGH3E3zynETuQS3++hGMvMXq2mDJeT2e8964y/ifIOBpr2K2isuLYnrtGKyxi+\n"
274 "ZptyHv6ymR3bDvio50cjnoT/WK1onosOJvtijGBS+U/ooq3im7ExpeQYXc/zpYsX\n"
275 "OmB5m6BvdomUp2PMqdxsmOPoaRkSYx5R2Rlo/z3csodl6sp3k465Y/jg7L4gkxDz\n"
276 "XJM+CS1xMhcOF0gBhppqLnG67x0ow847Pydstzkw0sOqedkLPuScaHNnlAWQ7QH6\n"
277 "mbbpqHJwekS4jQRHiKV8AQQA0iZ81WXypLI4ZE2+hYfBCnfMVfQF/vPgvASxhwri\n"
278 "GDa9Zc2f/VfakfNiwZgHH6iCeppHBiP2jljnbuOsL6f1R+0FsnyTVwHbuEU7IU2y\n"
279 "+J0/s0z3wcx9sx8T7brP5z5F2hdagBsD9YFGCifHDAEew4mmAisY0i2QHVIuXJFj\n"
280 "4RMAEQEAAYkBhwQYAQIADwUCR4ilfAIbAgUJEOrPgACoCRCgIeP4CJmQUZ0gBBkB\n"
281 "AgAGBQJHiKV8AAoJEIN7b7QuD+F2AEcEAKAjhO9kSOE8UuwEOKlwsWL9LUUSkHJj\n"
282 "c/ca0asLAerzrHsldRAcwCbWkVxBBHySw2CLFjzpgdXhwRtsytMgHaapfAPbinAW\n"
283 "jCPIEJx2gDZeZnTgi4DVbZn5E3UzHGyL69MEoXr5t+vpiemQFd/nGD+h/Q2A76od\n"
284 "gvAryRvS1Soj8bcGHjUflayXGOSvaD8P2V5Vz0hS82QZcqWxD8qUBqbcB8atokmO\n"
285 "IYxhKyRmO58T5Ma+iaxBTUIwee+pBYDgdH6E2dh9xLlwwzZKaCcIRCQcObkLsMVo\n"
286 "fZJo+m0Xf8zI57NeQF+hXJhW7lIrWgQVr8IVp/lgo76acLHfL/t1n0Nhg4r2srz2\n"
287 "fpP2w5laQ0qImYLnZhGFHU+rJUyFaHfhD8/svN2LuZkO570pjV/K68EaHnEfk5b8\n"
288 "jWu/euohwcCwf20M1kTo3Bg=\n"
289 "=Xjon\n" "-----END PGP PUBLIC KEY BLOCK-----\n";
290 const gnutls_datum_t server_crt = { server_crt_txt, sizeof (server_crt_txt) };
292 static unsigned char server_key_txt[] =
293 "-----BEGIN PGP PRIVATE KEY BLOCK-----\n"
294 "Version: GnuPG v1.4.6 (GNU/Linux)\n"
295 "\n"
296 "lQLGBEdj6IMBBiDXSEeLuT0eEsAA1sI/87YLaIu11W5pX20Ex6QWBcU0gvPKSEkj\n"
297 "clf64J5arp2D++xe2hy+ClEXjd7/bCsNNcUhhH6MfyZpalFIOglb043Xi35LlSAZ\n"
298 "x4/tijJWsXs6eyMbT2k16V3En/Ijrlau1F+6erSm6aSlzuRoJnwPPFVFbxgbNfz0\n"
299 "KVikHRq4/uiN0UZgzf4Z2tYxlPzdt9t9B28fm7u4PXL4okoddK4u63Q4VSeu7UZE\n"
300 "RlLMt1zunVguxJs2ZRDvABEBAAEABhwMx6crpb75ko5gXl9gsYSMj9O/YyCvU7Fi\n"
301 "l8FnZ0dKMz3qs7jXyFlttLjh1DzYkXN6PAN5yp3+wnbK/e5eVeNSdo2WpJOwrVWO\n"
302 "7pcQovHoKklAjmU98olaRhpv6BBTK+0tGUFaRrmrrYuz2xnwf3+kIpt4ahYW2dr9\n"
303 "B+/pvBSVC/sv2+3PEQSsXlWCYVgkQ7WBN4GQdyjjxhQpcWdf8Z6unx4zuS3s7GGM\n"
304 "4WaDxmDNCFlTGdrKPQeogtS3LVF9OiRCOvIlAxDmDvnC3zAwO/IvDUHFED9x9hmK\n"
305 "MeVwCg8rwDMptVYN2hm+bjNzjV4pimUVd+w7edjEky0Jd/6tTH01CBUWxs9Pfup2\n"
306 "cQ9zkYcVz1bwcoqeyRzFCJgi6PiVT38QFEvyusoVkwMQ747D6p7y+R52MEcIvcLb\n"
307 "lBXhRviz3rW+Sch4+ohUPvBU41saM5B6UcOmhdPfdvPriI4qXwFxusGWt98NN3aW\n"
308 "Ns2/L9kMX/SWnN6Elfj5hrrExDZ2CE60uuvfj+O/uXfO8LUDENE4vQrC399KLbJw\n"
309 "uCaqjqLysYA9EY/Nv8RFGkk1UM4ViW8v1/95D95F9WqochSYH8Phr3br0chDxofb\n"
310 "rnm6dUPE8uiriNaKWdoiUNSuvumh9lVixmRI923+4imu3scq+rlJAZ20EHRlc3Qz\n"
311 "LmdudXRscy5vcmeJAQAEEwECACYFAkdj6IMCGwMFCQlmAYAGCwkIBwMCBBUCCAME\n"
312 "FgIDAQIeAQIXgAAKCRCgIeP4CJmQUfekBh9xN88pxE7kEt/voRjLzF6tpgyXk9nv\n"
313 "PeuMv4nyDgaa9itorLi2J67RissYvmabch7+spkd2w74qOdHI56E/1itaJ6LDib7\n"
314 "YoxgUvlP6KKt4puxMaXkGF3P86WLFzpgeZugb3aJlKdjzKncbJjj6GkZEmMeUdkZ\n"
315 "aP893LKHZerKd5OOuWP44Oy+IJMQ81yTPgktcTIXDhdIAYaaai5xuu8dKMPOOz8n\n"
316 "bLc5MNLDqnnZCz7knGhzZ5QFkO0B+pm26ahycHpEnQHXBEeIpXwBBADSJnzVZfKk\n"
317 "sjhkTb6Fh8EKd8xV9AX+8+C8BLGHCuIYNr1lzZ/9V9qR82LBmAcfqIJ6mkcGI/aO\n"
318 "WOdu46wvp/VH7QWyfJNXAdu4RTshTbL4nT+zTPfBzH2zHxPtus/nPkXaF1qAGwP1\n"
319 "gUYKJ8cMAR7DiaYCKxjSLZAdUi5ckWPhEwARAQABAAP3QKGVoNi52HXEN3ttUCyB\n"
320 "Q1CDurh0MLDQoHomY3MGfI4VByk2YKMb2el4IJqyHrUbBYjTpHY31W2CSIdWfoTU\n"
321 "DIik49CQaUpR13dJXEiG4d+nyETFutEalTQI4hMjABD9l1XvZP7Ll3YWmqN8Cam5\n"
322 "JY23YAy2Noqbc3AcEut4+QIA1zcv8EU1QVqOwjSybRdm6HKK/A2bMqnITeUR/ikm\n"
323 "IuU4lhijm/d1qS6ZBehRvvYa9MY4V7BGEQLWSlyc5aYJ/wIA+fmRv0lHSs78QSUg\n"
324 "uRbNv6Aa6CXEOXmG+TpIaf/RWrPmBpdG8AROBVo1wmwG8oQaIjeX3RjKXfL3HTDD\n"
325 "CxNg7QIA06tApdo2j1gr3IrroUwQ7yvi56ELB1Lv+W3WLN8lzCfQ6Fs+7IJRrC2R\n"
326 "0uzLMGOsSORGAFIbAuLIMpc6rHCeS50hiQGHBBgBAgAPBQJHiKV8AhsCBQkQ6s+A\n"
327 "AKgJEKAh4/gImZBRnSAEGQECAAYFAkeIpXwACgkQg3tvtC4P4XYARwQAoCOE72RI\n"
328 "4TxS7AQ4qXCxYv0tRRKQcmNz9xrRqwsB6vOseyV1EBzAJtaRXEEEfJLDYIsWPOmB\n"
329 "1eHBG2zK0yAdpql8A9uKcBaMI8gQnHaANl5mdOCLgNVtmfkTdTMcbIvr0wShevm3\n"
330 "6+mJ6ZAV3+cYP6H9DYDvqh2C8CvJG9LVKiPxtwYeNR+VrJcY5K9oPw/ZXlXPSFLz\n"
331 "ZBlypbEPypQGptwHxq2iSY4hjGErJGY7nxPkxr6JrEFNQjB576kFgOB0foTZ2H3E\n"
332 "uXDDNkpoJwhEJBw5uQuwxWh9kmj6bRd/zMjns15AX6FcmFbuUitaBBWvwhWn+WCj\n"
333 "vppwsd8v+3WfQ2GDivayvPZ+k/bDmVpDSoiZgudmEYUdT6slTIVod+EPz+y83Yu5\n"
334 "mQ7nvSmNX8rrwRoecR+TlvyNa7966iHBwLB/bQzWROjcGA==\n"
335 "=mZnW\n" "-----END PGP PRIVATE KEY BLOCK-----\n";
336 const gnutls_datum_t server_key = { server_key_txt, sizeof (server_key_txt) };
338 static void
339 server_start (void)
341 /* Socket operations
343 listen_sd = socket (AF_INET, SOCK_STREAM, 0);
344 if (err == -1)
346 perror ("socket");
347 fail ("server: socket failed\n");
348 return;
351 memset (&sa_serv, '\0', sizeof (sa_serv));
352 sa_serv.sin_family = AF_INET;
353 sa_serv.sin_addr.s_addr = INADDR_ANY;
354 sa_serv.sin_port = htons (PORT); /* Server Port number */
356 setsockopt (listen_sd, SOL_SOCKET, SO_REUSEADDR, (void *) &optval,
357 sizeof (int));
359 err = bind (listen_sd, (SA *) & sa_serv, sizeof (sa_serv));
360 if (err == -1)
362 perror ("bind");
363 fail ("server: bind failed\n");
364 return;
367 err = listen (listen_sd, 1024);
368 if (err == -1)
370 perror ("listen");
371 fail ("server: listen failed\n");
372 return;
375 success ("server: ready. Listening to port '%d'.\n", PORT);
378 static void
379 server (void)
381 /* this must be called once in the program
383 gnutls_global_init ();
385 gnutls_global_set_log_function (tls_log_func);
386 if (debug)
387 gnutls_global_set_log_level (4711);
389 gnutls_certificate_allocate_credentials (&pgp_cred);
391 ret = gnutls_certificate_set_openpgp_key_mem2 (pgp_cred, &server_crt,
392 &server_key, "auto",
393 GNUTLS_OPENPGP_FMT_BASE64);
394 if (err < 0)
396 fail ("Could not set server key files...\n");
399 success ("Launched, setting DH parameters...\n");
401 generate_dh_params ();
403 gnutls_certificate_set_dh_params (pgp_cred, dh_params);
405 client_len = sizeof (sa_cli);
407 session = initialize_tls_session ();
409 sd = accept (listen_sd, (SA *) & sa_cli, &client_len);
411 success ("server: connection from %s, port %d\n",
412 inet_ntop (AF_INET, &sa_cli.sin_addr, topbuf,
413 sizeof (topbuf)), ntohs (sa_cli.sin_port));
415 gnutls_transport_set_ptr (session, (gnutls_transport_ptr_t) sd);
416 ret = gnutls_handshake (session);
417 if (ret < 0)
419 close (sd);
420 gnutls_deinit (session);
421 fail ("server: Handshake has failed (%s)\n\n", gnutls_strerror (ret));
422 return;
424 success ("server: Handshake was completed\n");
426 success ("server: TLS version is: %s\n",
427 gnutls_protocol_get_name (gnutls_protocol_get_version (session)));
429 /* see the Getting peer's information example */
430 print_info (session);
432 i = 0;
433 for (;;)
435 memset (buffer, 0, MAX_BUF + 1);
436 ret = gnutls_record_recv (session, buffer, MAX_BUF);
438 if (ret == 0)
440 success ("server: Peer has closed the GNUTLS connection\n");
441 break;
443 else if (ret < 0)
445 fail ("server: Received corrupted data(%d). Closing...\n", ret);
446 break;
448 else if (ret > 0)
450 /* echo data back to the client
452 gnutls_record_send (session, buffer, strlen (buffer));
455 /* do not wait for the peer to close the connection.
457 gnutls_bye (session, GNUTLS_SHUT_WR);
459 close (sd);
460 gnutls_deinit (session);
462 close (listen_sd);
464 gnutls_certificate_free_credentials (pgp_cred);
466 gnutls_dh_params_deinit (dh_params);
468 gnutls_global_deinit ();
470 success ("server: finished\n");
474 void
475 doit (void)
477 server_start ();
478 if (error_count)
479 return;
481 child = fork ();
482 if (child < 0)
484 perror ("fork");
485 fail ("fork");
486 return;
489 if (child)
491 int status;
492 /* parent */
493 server ();
494 wait (&status);
496 else
497 client ();