#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:
Вы можете сделать это с помощью сервера-посредника, который помогает установить соединение между телефонами.
- Создайте серверную программу, которая может принимать сокеты UDP и может хранить общедоступный адрес телефона, который подключается к нему
- Подключитесь к этому серверу с помощью телефона A, чтобы зарегистрировать его публичный адрес
- Используйте телефон B для подключения к серверу для запроса публичного адреса телефона A
- Получив этот адрес, вы можете создать прямой сокет с телефона A на телефон B, используя его общедоступный адрес. NAT сделает все остальное, а маршрутизатор отправит пакеты на личный адрес телефона
Комментарии:
1. Что, если вы хотите использовать TCP вместо UDP? Если это невозможно… Я думаю, мне пришлось бы реализовать все, что TCP дает вам бесплатно, но в UDP…
2. Я думаю, что оператор будет блокировать любые прямые входящие соединения с A, с которыми A ранее напрямую не связывался