Отправка сообщений P2P с использованием сквозного NAT в Android

#android #p2p #nat #kryonet

#Android #p2p #nat #kryonet

Вопрос:

Я пытаюсь отправить сообщение с одного телефона на другой, используя Kryonet (который использует связь через сокеты), и я намерен использовать сквозную передачу NAT, сохраняя публичный адрес клиентов.

Используется следующий код:

 public class MainActivity extends Activity implements View.OnClickListener
{
  /**
   * Called when the activity is first created.
   */
  @Override
  public void onCreate(Bundle savedInstanceState)
  {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    Button button = ((Button) findViewById(R.id.btnTest));
    button.setOnClickListener(this);
    TextView textViewOwn = ((TextView) findViewById(R.id.lblOwnIP));
    textViewOwn.setText(getLocalIpAddress() "n" GetGateway());
    try
    {
      Server server = new Server();
      server.start();
      server.bind(54555, 54777);
      Log.i("DEBUG", "Server is listening");
      final TextView textView = ((TextView) findViewById(R.id.txtMessage));
      server.addListener(new Listener()
      {
        public void received(Connection connection, Object object)
        {
          if (object instanceof String)
          {
            String request = (String) object;
            Log.i("DEBUG", request);
            ShowMessage(request, textView);
            String response = "Thanks";
            connection.sendTCP(response);
          }
        }
      });

    }
    catch (Exception e)
    {
      e.printStackTrace();
    }
  }

  public static String getLocalIpAddress() {
    try {
      for (Enumeration<NetworkInterface> en = NetworkInterface.getNetworkInterfaces(); en.hasMoreElements();) {
        NetworkInterface intf = en.nextElement();
        for (Enumeration<InetAddress> enumIpAddr = intf.getInetAddresses(); enumIpAddr.hasMoreElements();) {
          InetAddress inetAddress = enumIpAddr.nextElement();
          if (!inetAddress.isLoopbackAddress() amp;amp; inetAddress instanceof Inet4Address) {
            return inetAddress.getHostAddress();
          }
        }
      }
    } catch (SocketException ex) {
      ex.printStackTrace();
    }
    return null;
  }

  private String GetGateway()
  {
    DhcpInfo d;
    WifiManager wifii;
    wifii= (WifiManager) getSystemService(Context.WIFI_SERVICE);
    d=wifii.getDhcpInfo();
    int gatewayip = d.gateway;
    String mask = Formatter.formatIpAddress(gatewayip);
    return mask;
  }

  private void ShowMessage(final String request, final TextView textView)
  {
    runOnUiThread(new Runnable()
    {
      @Override
      public void run()
      {
        textView.setText(request);
      }
    });
  }

  @Override
  public void onClick(View view)
  {
    final String ip = ((EditText) findViewById(R.id.txtIP)).getText().toString();
    if (TextUtils.isEmpty(ip))
    {
      Toast.makeText(this, "No IP address", Toast.LENGTH_SHORT).show();
      return;
    }
    Runnable runnable = new Runnable()
    {
      public void run()
      {
        try
        {
          Client client = new Client();
          client.start();

          client.connect(5000, ip, 54555, 54777);
          Log.i("DEBUG", "Client is sending: " ip);
          runOnUiThread(new Runnable()
          {
            @Override
            public void run()
            {
              Toast.makeText(MainActivity.this, "Client is sending: " ip, Toast.LENGTH_SHORT).show();
            }
          });

          String request = "Here is the request";
          client.sendTCP(request);
        }
        catch (final Exception e)
        {
          e.printStackTrace();
          runOnUiThread(new Runnable()
          {
            @Override
            public void run()
            {
              Toast.makeText(MainActivity.this, "Client has errors: " e.toString(), Toast.LENGTH_SHORT).show();

            }
          });
        }
      }
    };
       Thread thread=new Thread(runnable);
    thread.start();
  }
}
 

Если телефоны подключены к одному и тому же Wi-Fi, все работает нормально. Если один из них подключен к 3G, я получаю внешний IP-адрес (например: 100.65.96 ..) и локальный адрес в качестве шлюза. Поскольку знание внешнего IP-адреса является ключом для NAT PT, я подумал, что этого достаточно, чтобы получить сообщение, но я получаю тайм-аут.

Что еще я должен реализовать, чтобы телефон 3G получил сообщение?

Ответ №1:

Вы можете сделать это с помощью сервера-посредника, который помогает установить соединение между телефонами.

  1. Создайте серверную программу, которая может принимать сокеты UDP и может хранить общедоступный адрес телефона, который подключается к нему
  2. Подключитесь к этому серверу с помощью телефона A, чтобы зарегистрировать его публичный адрес
  3. Используйте телефон B для подключения к серверу для запроса публичного адреса телефона A
  4. Получив этот адрес, вы можете создать прямой сокет с телефона A на телефон B, используя его общедоступный адрес. NAT сделает все остальное, а маршрутизатор отправит пакеты на личный адрес телефона

Комментарии:

1. Что, если вы хотите использовать TCP вместо UDP? Если это невозможно… Я думаю, мне пришлось бы реализовать все, что TCP дает вам бесплатно, но в UDP…

2. Я думаю, что оператор будет блокировать любые прямые входящие соединения с A, с которыми A ранее напрямую не связывался