Не удается обнаружить службу с помощью 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, и есть две ошибки, которые блокируют нас:

  1. в public void onServiceFound(NsdServiceInfo service) метод, код для проверки имени службы неверен. Насколько я понимаю, другое устройство в той же сети может также использовать то же имя службы, поэтому удалите if (service.getServiceName().equals(mServiceName)) кодовый блок.
  2. То же самое происходит в public void onServiceResolved(NsdServiceInfo serviceInfo) кодовый блок. пожалуйста, удалите if (serviceInfo.getServiceName().equals(mServiceName)) кодовый блок.
Другие вопросы по тегам