TCPを使う(acceptした相手の確認)

TCPサーバはlistenした後にクライアントからのコネクションをacceptしますが、どのクライアントからのコネクションをacceptしたかを知りたい場合があります。 ここでは、接続相手を確認する方法を説明します。

acceptした相手を表示するサンプル

acceptの第二引数は、接続した相手に関する情報を含んでいます。 以下のサンプルは、acceptした相手のIPアドレスとポート番号を表示します。 IPアドレスの表示にはinet_ntoaを利用します。 inet_ntoaは、引数として渡したstruct in_addrを表現する文字列を返します。


#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

int
main()
{
 int sock0;
 struct sockaddr_in addr;
 struct sockaddr_in client;
 socklen_t len;
 int sock;

 sock0 = socket(AF_INET, SOCK_STREAM, 0);

 addr.sin_family = AF_INET;
 addr.sin_port = htons(12345);
 addr.sin_addr.s_addr = INADDR_ANY;
 addr.sin_len = sizeof(addr);

 bind(sock0, (struct sockaddr *)&addr, sizeof(addr));

 listen(sock0, 5);

 while (1) {
   len = sizeof(client);
   sock = accept(sock0, (struct sockaddr *)&client, &len);

   printf("accepted connection from %s, port=%d\n",
          inet_ntoa(client.sin_addr), ntohs(client.sin_port));

   send(sock, "HELLO", 5, 0);

   close(sock);
 }

 close(sock0);

 return 0;
}

このサンプルコードのポイントは、acceptの引数の内容をprintfしているところです。 今回利用している通信形態はAF_INET+SOCK_STREAMなので、IPv4+TCPということになります。 IPv4+TCPのソケットでacceptを行う場合、acceptの第二引数の実体としてはstruct sockaddr_inを利用します。 struct sockaddr_inに格納される情報は、通信相手のIPアドレスとソースポート(source port、送信元ポート)です。 上記サンプルコードでは、それらの情報をprintfしています。

上記サンプルコードは、TCPで接続してくる相手に関する情報を表示するため、適切に改造すればアクセスログを生成する簡易なサーバが作成できます。 上記サンプルコードを改造して色々試して下さい。

IPv6基礎検定

YouTubeチャンネルやってます!