2016-10-26 7 views
-1

私は、着信接続をポート6555でリッスンし、「サーバーへようこそ!」というC++ソケットプログラムを実行しているubuntu Linuxサーバーを持っています。彼らへ。 localhostからtelnetすると動作しますが、ネットワーク外からtelnetすると動作しません。ローカルのネットワークの外部からC++のlinuxソケットプログラムに接続できません

ローカルネットワーク管理者は、彼が80

ために、以前に行ったように、彼は私も80ポート上で動作しているWebサーバーを持っている、と私はtelnetのできる、サーバのローカルIPにポート6555を転送した私に言っています私はどこからでもその問題に気づくことができません。

私のサーバー上のiptables(ufw)ファイアウォールを一時的に無効にして接続しようとしましたが、違いはありませんでした。

オープンポートチェッカーは、ポートが閉じていることを示します。これは転送が機能していないことを意味しますか? netutils.ccでgetSrvSocket()と

#include <iostream> 
#include <sys/types.h> 
#include <sys/socket.h> 
#include <netdb.h> 
#include <cstring> 
#include <netinet/in.h> 
#include <unistd.h> 

#include "../include/netutils.hh" 

int main() { 
    sockaddr_storage clt_addr; 
    socklen_t sin_size; 
    int cltSock = -1; 
    const std::string port = "6555"; 
    int servSock = -1; 
    try { 
     servSock = getSrvSocket(port, 10); 
     while (true) { 
      sin_size = sizeof clt_addr; 
      cltSock = accept(servSock, (sockaddr*)&clt_addr, &sin_size); 
      if (cltSock == -1) continue; 
      if (!fork()) { 
       close(servSock); 
       send(cltSock, "Welcome to the server!\n", 23, 0); 
       while(true); 
      } 
      close(cltSock); 
     } 

    } 
    catch(std::string excpt) 
    { 
     std::cerr << excpt << '\n'; 
     return -1; 
    } 
    return 0; 

} 

:ここ コード、Gでコンパイル++あるgetaddrinfo()を呼び出すとき

int getSrvSocket(std::string port, int BACKLOG) { 
    int yes = 1; 
    addrinfo *p; 
    addrinfo hints; 
    addrinfo *servinfo; 
    memset(&hints, 0, sizeof hints); 
    hints.ai_family = AF_UNSPEC;  //Can use either IPv4 or IPv6 
    hints.ai_socktype = SOCK_STREAM; 
    hints.ai_flags = AI_PASSIVE;  //Automatically set IP address 
    int sock = -1; 
    if (getaddrinfo(NULL, port.c_str(), &hints, &servinfo) !=0) { 
     throw std::string("Error: Could not resolve address info"); 
    } 

    for (p = servinfo; p != NULL; p = p->ai_next) { 
     sock = socket(p->ai_family, p->ai_socktype, p->ai_protocol); 
     if (sock == -1) continue; 
     if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &yes, 
        sizeof(int)) == -1) { 
      throw std::string("Error: setsockopt() failure"); 
     } 
     if (bind(sock, p->ai_addr, p->ai_addrlen) == -1) { 
      close(sock); 
      continue; 
     } 
     break; 
    } 
    freeaddrinfo(servinfo); 
    if (NULL == p) throw std::string("Error: Server failed to bind"); 
    if (listen(sock, BACKLOG) == -1) { 
     throw std::string("Error: Failed to to start listen()"); 
    } 
    struct sigaction sa; 
    sa.sa_handler = sigchld_handler; 
    sigemptyset(&sa.sa_mask); 
    sa.sa_flags = SA_RESTART; 

    if (sigaction(SIGCHLD, &sa, NULL) == -1) { 
     throw std::string("Error: sigaction() failed"); 
    } 
    return sock; 
} 

答えて

1

getSrvSocket()AF_UNSPECを指定され、それが出力できることを意味し、IPv4とIPv6の両方アドレス。しかし、1つのアドレスにバインドして停止するだけです。だから実際にはIPv6アドレスにバインドしていて、クライアントが代わりにIPv4を使用して接続しようとしている(またはその逆)ことは明らかにうまくいかないことがです。

IPv4とIPv6の両方をサポートする場合は、getaddrinfo()が返す各アドレスに別のリスニングソケットをバインドする必要があります。正常にバインドしたアドレスもログに記録して、どのクライアントに接続できるか(また、ネットワーク管理者に転送するように指示する必要がある)を把握する必要があります。

AF_UNSPECgetaddrinfo()を使用しないでください。代わりにAF_INET(IPv4)またはAF_INET6(IPv6)を使用してください。

+0

AF_INETを使用するように変更してテストしました。それはまだlocalhostで働いていましたが、依然として外部ネットワークからは機能しませんでした。 – RobNo

+0

これを展開するには、システムで 'IPV6_V6ONLY'がデフォルトでtrueになっているとエラーになることもあります。 –

+0

IPV6_V6ONLYがtrueまたはfalseであるかどうかを確認するにはどうすればよいですか?私はプロセスがnetstat -tulpnを実行してからIPv4で実行していると思うし、TCPを取得する0 0 0.0.0.0:6555 0.0.0.0:* LISTEN 10292/netsrv – RobNo

関連する問題