Не удается обнаружить службу с помощью Network Service Discovery (NSD) на Android
Я новичок, пытающийся использовать NSD в приложении, которое я создаю, чтобы найти пиров, использующих один и тот же сервис в той же сети Wi-Fi, и подключиться к нему.
[См. Код] У меня есть одно устройство под управлением HostActivity.java и одно - под управлением GuestActivity.java. Нажатие плавающей кнопки в Host регистрирует службу на порту 9000 (предварительно настроенную на данный момент), а нажатие на нее в гостевой системе запускает обнаружение службы для нее. Оба подключены к одной сети Wi-Fi.
В моем logcat я вижу, что регистрация в HostActivity.java работает нормально - я получаю необходимые сообщения журнала из NsdHelper.java. Тем не менее, при запуске GuestActivity.java просто регистрируется "Обнаружение службы запущено" и никогда не заканчивается преобразование в службу, существующую в сети.
Что я делаю неправильно? Пожалуйста помоги.
Соответствующий код -
HostActivity.java
NsdHelper mNsdHelper;
public static final String TAG = "ABC";
private final int ABC_PORT = 9000;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_shared_list);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
mNsdHelper = new NsdHelper(this);
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Snackbar.make(view, "Started Registration", Snackbar.LENGTH_LONG)
.setAction("CLOSE", null).show();
mNsdHelper.initializeNsd();
mNsdHelper.registerService(ABC_PORT);
Snackbar.make(view, "Completed Registration", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
}
});
}
@Override
protected void onPause() {
super.onPause();
}
@Override
protected void onResume() {
super.onResume();
if (mNsdHelper != null) {
mNsdHelper.initializeNsd();
mNsdHelper.registerService(ABC_PORT);
}
}
@Override
protected void onDestroy() {
mNsdHelper.tearDown();
super.onDestroy();
}
GuestActivity.java
NsdHelper mNsdHelper;
public static final String TAG = "ABC";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_shared_list);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
mNsdHelper = new NsdHelper(this);
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Snackbar.make(view, "Searching for available services", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
mNsdHelper.initializeNsd();
mNsdHelper.discoverServices();
}
});
}
@Override
protected void onPause() {
super.onPause();
}
@Override
protected void onResume() {
super.onResume();
if(mNsdHelper != null) {
mNsdHelper.initializeNsd();
mNsdHelper.discoverServices();
}
}
@Override
protected void onDestroy() {
mNsdHelper.tearDown();
super.onDestroy();
}
NsdHelper.java - содержит все детали реализации NSD
public class NsdHelper {
Context mContext;
NsdManager mNsdManager;
NsdManager.ResolveListener mResolveListener;
NsdManager.DiscoveryListener mDiscoveryListener;
NsdManager.RegistrationListener mRegistrationListener;
private static final String SERVICE_TYPE = "_http._tcp.";
private static final String TAG = "NsdHelper";
private String mServiceName = "ABC";
// First step : register a NsdServiceInfo object to advertise your service
NsdServiceInfo mService;
public NsdHelper(Context context) {
mContext = context;
mNsdManager = (NsdManager) context.getSystemService(Context.NSD_SERVICE);
}
public void initializeNsd() {
initializeRegistrationListener();
initializeDiscoveryListener();
initializeResolveListener();
}
// Check success of service registration
public void initializeRegistrationListener() {
mRegistrationListener = new NsdManager.RegistrationListener() {
@Override
public void onServiceRegistered(NsdServiceInfo nsdServiceInfo) {
mServiceName = nsdServiceInfo.getServiceName();
Log.d(TAG, "Registered name : " + mServiceName);
}
@Override
public void onRegistrationFailed(NsdServiceInfo nsdServiceInfo, int arg1) {
Log.d(TAG, "Registration Failed");
}
@Override
public void onServiceUnregistered(NsdServiceInfo nsdServiceInfo) {
Log.d(TAG, "Unregistered");
}
@Override
public void onUnregistrationFailed(NsdServiceInfo serviceInfo, int errorCode) {
}
};
}
public void registerService(int port) {
NsdServiceInfo serviceInfo = new NsdServiceInfo();
serviceInfo.setPort(port);
serviceInfo.setServiceName(mServiceName);
serviceInfo.setServiceType(SERVICE_TYPE);
mNsdManager.registerService(
serviceInfo, NsdManager.PROTOCOL_DNS_SD, mRegistrationListener);
}
// To discover services of the type you're looking for on the network
public void initializeDiscoveryListener() {
mDiscoveryListener = new NsdManager.DiscoveryListener() {
@Override
public void onDiscoveryStarted(String regType) {
Log.d(TAG, "Service discovery started");
}
@Override
public void onServiceFound(NsdServiceInfo service) {
Log.d(TAG, "Service discovery success : " + service);
Log.d(TAG, "Host = "+ service.getServiceName());
Log.d(TAG, "port = " + String.valueOf(service.getPort()));
if (!service.getServiceType().equals(SERVICE_TYPE)) {
Log.d(TAG, "Unknown Service Type: " + service.getServiceType());
} else if (service.getServiceName().equals(mServiceName)) {
Log.d(TAG, "Same machine: " + mServiceName);
} else if (service.getServiceName().contains(mServiceName)){
mNsdManager.resolveService(service, mResolveListener);
}
}
@Override
public void onServiceLost(NsdServiceInfo service) {
Log.e(TAG, "service lost" + service);
if (mService == service) {
mService = null;
Log.e(TAG, "service lost" + service);
}
}
@Override
public void onDiscoveryStopped(String serviceType) {
Log.i(TAG, "Discovery stopped: " + serviceType);
}
@Override
public void onStartDiscoveryFailed(String serviceType, int errorCode) {
Log.e(TAG, "Discovery failed: Error code:" + errorCode);
mNsdManager.stopServiceDiscovery(this);
}
@Override
public void onStopDiscoveryFailed(String serviceType, int errorCode) {
Log.e(TAG, "Discovery failed: Error code:" + errorCode);
mNsdManager.stopServiceDiscovery(this);
}
};
}
public void discoverServices() {
mNsdManager.discoverServices(
SERVICE_TYPE, NsdManager.PROTOCOL_DNS_SD, mDiscoveryListener);
}
// Connection handshake
public void initializeResolveListener() {
mResolveListener = new NsdManager.ResolveListener() {
@Override
public void onResolveFailed(NsdServiceInfo serviceInfo, int errorCode) {
Log.e(TAG, "Resolve failed" + errorCode);
}
@Override
public void onServiceResolved(NsdServiceInfo serviceInfo) {
Log.e(TAG, "Resolve Succeeded. " + serviceInfo);
if (serviceInfo.getServiceName().equals(mServiceName)) {
Log.d(TAG, "Same IP.");
return;
}
mService = serviceInfo;
int port = mService.getPort();
InetAddress host = mService.getHost();
}
};
}
public void stopDiscovery() {
if (mNsdManager != null) {
mNsdManager.stopServiceDiscovery(mDiscoveryListener);
}
}
public NsdServiceInfo getChosenServiceInfo() {
return mService;
}
public void tearDown() {
if (mNsdManager != null) {
if (mRegistrationListener != null) {
mNsdManager.unregisterService(mRegistrationListener);
}
if (mDiscoveryListener != null) {
mNsdManager.stopServiceDiscovery(mDiscoveryListener);
}
}
}
}
1 ответ
Кажется, вы следовали примеру кода на сайте разработчиков Android, и есть две ошибки, которые блокируют нас:
- в
public void onServiceFound(NsdServiceInfo service)
метод, код для проверки имени службы неверен. Насколько я понимаю, другое устройство в той же сети может также использовать то же имя службы, поэтому удалитеif (service.getServiceName().equals(mServiceName))
кодовый блок. - То же самое происходит в
public void onServiceResolved(NsdServiceInfo serviceInfo)
кодовый блок. пожалуйста, удалитеif (serviceInfo.getServiceName().equals(mServiceName))
кодовый блок.