/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.server; import static libcore.junit.util.compat.CoreCompatChangeRule.DisableCompatChanges; import static libcore.junit.util.compat.CoreCompatChangeRule.EnableCompatChanges; import static org.junit.Assert.assertEquals; import static org.mockito.Mockito.any; import static org.mockito.Mockito.doReturn; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.never; import static org.mockito.Mockito.reset; import static org.mockito.Mockito.timeout; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.when; import android.compat.testing.PlatformCompatChangeRule; import android.content.ContentResolver; import android.content.Context; import android.net.nsd.NsdManager; import android.net.nsd.NsdServiceInfo; import android.os.Build; import android.os.Handler; import android.os.HandlerThread; import android.os.Looper; import android.os.Message; import androidx.test.filters.SmallTest; import com.android.server.NsdService.DaemonConnection; import com.android.server.NsdService.DaemonConnectionSupplier; import com.android.server.NsdService.NativeCallbackReceiver; import com.android.testutils.DevSdkIgnoreRule; import com.android.testutils.DevSdkIgnoreRunner; import com.android.testutils.HandlerUtils; import org.junit.After; import org.junit.Before; import org.junit.Rule; import org.junit.Test; import org.junit.rules.TestRule; import org.junit.runner.RunWith; import org.mockito.ArgumentCaptor; import org.mockito.Mock; import org.mockito.MockitoAnnotations; import org.mockito.Spy; // TODOs: // - test client can send requests and receive replies // - test NSD_ON ENABLE/DISABLED listening @RunWith(DevSdkIgnoreRunner.class) @SmallTest @DevSdkIgnoreRule.IgnoreUpTo(Build.VERSION_CODES.R) public class NsdServiceTest { static final int PROTOCOL = NsdManager.PROTOCOL_DNS_SD; private static final long CLEANUP_DELAY_MS = 500; private static final long TIMEOUT_MS = 500; @Rule public TestRule compatChangeRule = new PlatformCompatChangeRule(); @Mock Context mContext; @Mock ContentResolver mResolver; @Mock NsdService.NsdSettings mSettings; NativeCallbackReceiver mDaemonCallback; @Spy DaemonConnection mDaemon = new DaemonConnection(mDaemonCallback); HandlerThread mThread; TestHandler mHandler; @Before public void setUp() throws Exception { MockitoAnnotations.initMocks(this); mThread = new HandlerThread("mock-service-handler"); mThread.start(); doReturn(true).when(mDaemon).execute(any()); mHandler = new TestHandler(mThread.getLooper()); when(mContext.getContentResolver()).thenReturn(mResolver); } @After public void tearDown() throws Exception { if (mThread != null) { mThread.quit(); mThread = null; } } @Test @DisableCompatChanges(NsdManager.RUN_NATIVE_NSD_ONLY_IF_LEGACY_APPS) public void testPreSClients() { when(mSettings.isEnabled()).thenReturn(true); NsdService service = makeService(); // Pre S client connected, the daemon should be started. NsdManager client1 = connectClient(service); waitForIdle(); verify(mDaemon, times(1)).maybeStart(); verifyDaemonCommands("start-service"); NsdManager client2 = connectClient(service); waitForIdle(); verify(mDaemon, times(1)).maybeStart(); client1.disconnect(); // Still 1 client remains, daemon shouldn't be stopped. waitForIdle(); verify(mDaemon, never()).maybeStop(); client2.disconnect(); // All clients are disconnected, the daemon should be stopped. verifyDelayMaybeStopDaemon(CLEANUP_DELAY_MS); verifyDaemonCommands("stop-service"); } @Test @EnableCompatChanges(NsdManager.RUN_NATIVE_NSD_ONLY_IF_LEGACY_APPS) public void testNoDaemonStartedWhenClientsConnect() { when(mSettings.isEnabled()).thenReturn(true); NsdService service = makeService(); // Creating an NsdManager will not cause any cmds executed, which means // no daemon is started. NsdManager client1 = connectClient(service); waitForIdle(); verify(mDaemon, never()).execute(any()); // Creating another NsdManager will not cause any cmds executed. NsdManager client2 = connectClient(service); waitForIdle(); verify(mDaemon, never()).execute(any()); // If there is no active request, try to clean up the daemon // every time the client disconnects. client1.disconnect(); verifyDelayMaybeStopDaemon(CLEANUP_DELAY_MS); reset(mDaemon); client2.disconnect(); verifyDelayMaybeStopDaemon(CLEANUP_DELAY_MS); client1.disconnect(); client2.disconnect(); } @Test @EnableCompatChanges(NsdManager.RUN_NATIVE_NSD_ONLY_IF_LEGACY_APPS) public void testClientRequestsAreGCedAtDisconnection() { when(mSettings.isEnabled()).thenReturn(true); NsdService service = makeService(); NsdManager client = connectClient(service); waitForIdle(); verify(mDaemon, never()).maybeStart(); verify(mDaemon, never()).execute(any()); NsdServiceInfo request = new NsdServiceInfo("a_name", "a_type"); request.setPort(2201); // Client registration request NsdManager.RegistrationListener listener1 = mock(NsdManager.RegistrationListener.class); client.registerService(request, PROTOCOL, listener1); waitForIdle(); verify(mDaemon, times(1)).maybeStart(); verifyDaemonCommands("start-service", "register 2 a_name a_type 2201"); // Client discovery request NsdManager.DiscoveryListener listener2 = mock(NsdManager.DiscoveryListener.class); client.discoverServices("a_type", PROTOCOL, listener2); waitForIdle(); verify(mDaemon, times(1)).maybeStart(); verifyDaemonCommand("discover 3 a_type"); // Client resolve request NsdManager.ResolveListener listener3 = mock(NsdManager.ResolveListener.class); client.resolveService(request, listener3); waitForIdle(); verify(mDaemon, times(1)).maybeStart(); verifyDaemonCommand("resolve 4 a_name a_type local."); // Client disconnects, stop the daemon after CLEANUP_DELAY_MS. client.disconnect(); verifyDelayMaybeStopDaemon(CLEANUP_DELAY_MS); // checks that request are cleaned verifyDaemonCommands("stop-register 2", "stop-discover 3", "stop-resolve 4", "stop-service"); client.disconnect(); } @Test @EnableCompatChanges(NsdManager.RUN_NATIVE_NSD_ONLY_IF_LEGACY_APPS) public void testCleanupDelayNoRequestActive() { when(mSettings.isEnabled()).thenReturn(true); NsdService service = makeService(); NsdManager client = connectClient(service); NsdServiceInfo request = new NsdServiceInfo("a_name", "a_type"); request.setPort(2201); NsdManager.RegistrationListener listener1 = mock(NsdManager.RegistrationListener.class); client.registerService(request, PROTOCOL, listener1); waitForIdle(); verify(mDaemon, times(1)).maybeStart(); verifyDaemonCommands("start-service", "register 2 a_name a_type 2201"); client.unregisterService(listener1); verifyDaemonCommand("stop-register 2"); verifyDelayMaybeStopDaemon(CLEANUP_DELAY_MS); verifyDaemonCommand("stop-service"); reset(mDaemon); client.disconnect(); // Client disconnects, after CLEANUP_DELAY_MS, maybeStop the daemon. verifyDelayMaybeStopDaemon(CLEANUP_DELAY_MS); } private void waitForIdle() { HandlerUtils.waitForIdle(mHandler, TIMEOUT_MS); } NsdService makeService() { DaemonConnectionSupplier supplier = (callback) -> { mDaemonCallback = callback; return mDaemon; }; NsdService service = new NsdService(mContext, mSettings, mHandler, supplier, CLEANUP_DELAY_MS); verify(mDaemon, never()).execute(any(String.class)); return service; } NsdManager connectClient(NsdService service) { return new NsdManager(mContext, service); } void verifyDelayMaybeStopDaemon(long cleanupDelayMs) { waitForIdle(); // Stop daemon shouldn't be called immediately. verify(mDaemon, never()).maybeStop(); // Clean up the daemon after CLEANUP_DELAY_MS. verify(mDaemon, timeout(cleanupDelayMs + TIMEOUT_MS)).maybeStop(); } void verifyDaemonCommands(String... wants) { verifyDaemonCommand(String.join(" ", wants), wants.length); } void verifyDaemonCommand(String want) { verifyDaemonCommand(want, 1); } void verifyDaemonCommand(String want, int n) { waitForIdle(); final ArgumentCaptor argumentsCaptor = ArgumentCaptor.forClass(Object.class); verify(mDaemon, times(n)).execute(argumentsCaptor.capture()); String got = ""; for (Object o : argumentsCaptor.getAllValues()) { got += o + " "; } assertEquals(want, got.trim()); // rearm deamon for next command verification reset(mDaemon); doReturn(true).when(mDaemon).execute(any()); } public static class TestHandler extends Handler { public Message lastMessage; TestHandler(Looper looper) { super(looper); } @Override public void handleMessage(Message msg) { lastMessage = obtainMessage(); lastMessage.copyFrom(msg); } } }