5. Programme client TCP : talker

5.1. Utilisation des sockets avec le client TCP

Le fait que le protocole TCP soit un service orienté connexion entraîne un changement important dans le code source du programme client étudié précédemment. Le contrôle d'erreur est directement intégré dans la couche transport et n'est plus à la charge de la couche application. Il n'est donc plus nécessaire de mettre en œuvre un mécanisme de gestion de temporisation.

Autre changement, il est maintenant nécessaire d'établir la connexion avant d'échanger la moindre information. Cette opération ce fait à l'aide de la fonction connect.

if (connect(socketDescriptor, servinfo->ai_addr, servinfo->ai_addrlen) == -1) {
  close(socketDescriptor);
  perror("connect");
  exit(EXIT_FAILURE);
}

En cas d'échec de cette demande d'établissement de connexion, on abandonne le traitement.

5.2. Patch code source

Patch du programme tcp-talker.c :

--- udp-talker.c    2016-10-25 16:22:31.473862464 +0200
+++ tcp-talker.c        2016-10-25 16:22:31.473862464 +0200
@@ -19,8 +19,6 @@
   int socketDescriptor, status;
   unsigned int msgLength;
   struct addrinfo hints, *servinfo;
-  struct timeval timeVal;
-  fd_set readSet;
   char msg[MSG_ARRAY_SIZE], serverPort[PORT_ARRAY_SIZE];
 
   puts("Entrez le nom du serveur ou son adresse IP : ");
@@ -33,7 +31,7 @@
 
   memset(&hints, 0, sizeof hints);
   hints.ai_family = AF_INET;
-  hints.ai_socktype = SOCK_DGRAM;
+  hints.ai_socktype = SOCK_STREAM;
 
   if ((status = getaddrinfo(msg, serverPort, &hints, &servinfo)) != 0) {
     fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(status));
@@ -46,6 +44,15 @@
     exit(EXIT_FAILURE);
   }
 
+  if (connect(socketDescriptor, servinfo->ai_addr, servinfo->ai_addrlen) == -1) {
+    close(socketDescriptor);
+    perror("connect");
+    exit(EXIT_FAILURE);
+  }
+
+  // Libération de la mémoire occupée par les enregistrements
+  freeaddrinfo(servinfo);
+
   puts("\nEntrez quelques caractères au clavier.");
   puts("Le serveur les modifiera et les renverra.");
   puts("Pour sortir, entrez une ligne avec le caractère '.' uniquement.");
@@ -69,13 +76,6 @@
         exit(EXIT_FAILURE);
       }
 
-      // Attente de la réponse pendant une seconde.
-      FD_ZERO(&readSet);
-      FD_SET(socketDescriptor, &readSet);
-      timeVal.tv_sec = 1;
-      timeVal.tv_usec = 0;
-
-      if (select(socketDescriptor+1, &readSet, NULL, NULL, &timeVal)) {
         // Lecture de la ligne modifiée par le serveur.
         memset(msg, 0, sizeof msg);  // Mise à zéro du tampon
         if (recv(socketDescriptor, msg, sizeof msg, 0) == -1) {
@@ -83,13 +83,10 @@
           close(socketDescriptor);
           exit(EXIT_FAILURE);
         }
+    }
 
         printf("Message traité : %s\n", msg);
-      }
-      else {
-        puts("Pas de réponse dans la seconde.");
-      }
-    }
+
     // Invite de commande pour l'utilisateur et lecture des caractères jusqu'à la
     // limite MAX_MSG. Puis suppression du saut de ligne en mémoire tampon.
     // Comme ci-dessus.