chromium/mojo/public/java/system/javatests/src/org/chromium/mojo/system/impl/WatcherImplTest.java

// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

package org.chromium.mojo.system.impl;

import androidx.test.filters.SmallTest;

import org.junit.After;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;

import org.chromium.base.test.BaseJUnit4ClassRunner;
import org.chromium.mojo.MojoTestRule;
import org.chromium.mojo.system.Core;
import org.chromium.mojo.system.Handle;
import org.chromium.mojo.system.InvalidHandle;
import org.chromium.mojo.system.MessagePipeHandle;
import org.chromium.mojo.system.MojoException;
import org.chromium.mojo.system.MojoResult;
import org.chromium.mojo.system.Pair;
import org.chromium.mojo.system.Watcher;
import org.chromium.mojo.system.Watcher.Callback;

import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;

/** Testing the Watcher. */
@RunWith(BaseJUnit4ClassRunner.class)
public class WatcherImplTest {
    @Rule public MojoTestRule mTestRule = new MojoTestRule();

    private List<Handle> mHandlesToClose = new ArrayList<Handle>();
    private Watcher mWatcher;
    private Core mCore;

    /**
     * @see MojoTestCase#setUp()
     */
    @Before
    public void setUp() {
        mWatcher = new WatcherImpl();
        mCore = CoreImpl.getInstance();
    }

    /**
     * @see MojoTestCase#tearDown()
     */
    @After
    public void tearDown() {
        mWatcher.destroy();
        MojoException toThrow = null;
        for (Handle handle : mHandlesToClose) {
            try {
                handle.close();
            } catch (MojoException e) {
                if (toThrow == null) {
                    toThrow = e;
                }
            }
        }
        if (toThrow != null) {
            throw toThrow;
        }
    }

    private void addHandlePairToClose(Pair<? extends Handle, ? extends Handle> handles) {
        mHandlesToClose.add(handles.first);
        mHandlesToClose.add(handles.second);
    }

    private static class WatcherResult implements Callback {
        private int mResult = Integer.MIN_VALUE;
        private MessagePipeHandle mReadPipe;

        /**
         * @param readPipe A MessagePipeHandle to read from when onResult triggers success.
         */
        public WatcherResult(MessagePipeHandle readPipe) {
            mReadPipe = readPipe;
        }

        public WatcherResult() {
            this(null);
        }

        /**
         * @see Callback#onResult(int)
         */
        @Override
        public void onResult(int result) {
            this.mResult = result;

            if (result == MojoResult.OK && mReadPipe != null) {
                mReadPipe.readMessage(MessagePipeHandle.ReadFlags.NONE);
            }
        }

        /**
         * @return the result
         */
        public int getResult() {
            return mResult;
        }
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testCorrectResult() {
        // Checking a correct result.
        Pair<MessagePipeHandle, MessagePipeHandle> handles = mCore.createMessagePipe(null);
        addHandlePairToClose(handles);
        final WatcherResult watcherResult = new WatcherResult(handles.first);
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mWatcher.start(handles.first, Core.HandleSignals.READABLE, watcherResult);
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        handles.second.writeMessage(
                ByteBuffer.allocateDirect(1), null, MessagePipeHandle.WriteFlags.NONE);
        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(MojoResult.OK, watcherResult.getResult());
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testClosingPeerHandle() {
        // Closing the peer handle.
        Pair<MessagePipeHandle, MessagePipeHandle> handles = mCore.createMessagePipe(null);
        addHandlePairToClose(handles);

        final WatcherResult watcherResult = new WatcherResult();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mWatcher.start(handles.first, Core.HandleSignals.READABLE, watcherResult);
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        handles.second.close();
        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(MojoResult.FAILED_PRECONDITION, watcherResult.getResult());
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testClosingWatchedHandle() {
        // Closing the peer handle.
        Pair<MessagePipeHandle, MessagePipeHandle> handles = mCore.createMessagePipe(null);
        addHandlePairToClose(handles);

        final WatcherResult watcherResult = new WatcherResult();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mWatcher.start(handles.first, Core.HandleSignals.READABLE, watcherResult);
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        handles.first.close();
        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(MojoResult.CANCELLED, watcherResult.getResult());
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testInvalidHandle() {
        // Closing the peer handle.
        Pair<MessagePipeHandle, MessagePipeHandle> handles = mCore.createMessagePipe(null);
        addHandlePairToClose(handles);

        final WatcherResult watcherResult = new WatcherResult();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        handles.first.close();
        Assert.assertEquals(
                MojoResult.INVALID_ARGUMENT,
                mWatcher.start(handles.first, Core.HandleSignals.READABLE, watcherResult));
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testDefaultInvalidHandle() {
        final WatcherResult watcherResult = new WatcherResult();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        Assert.assertEquals(
                MojoResult.INVALID_ARGUMENT,
                mWatcher.start(InvalidHandle.INSTANCE, Core.HandleSignals.READABLE, watcherResult));
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testCancel() {
        // Closing the peer handle.
        Pair<MessagePipeHandle, MessagePipeHandle> handles = mCore.createMessagePipe(null);
        addHandlePairToClose(handles);

        final WatcherResult watcherResult = new WatcherResult();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mWatcher.start(handles.first, Core.HandleSignals.READABLE, watcherResult);
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mWatcher.cancel();
        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        handles.second.writeMessage(
                ByteBuffer.allocateDirect(1), null, MessagePipeHandle.WriteFlags.NONE);
        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());
    }

    /** Testing {@link Watcher} implementation. */
    @Test
    @SmallTest
    public void testImmediateCancelOnInvalidHandle() {
        // Closing the peer handle.
        Pair<MessagePipeHandle, MessagePipeHandle> handles = mCore.createMessagePipe(null);
        addHandlePairToClose(handles);

        final WatcherResult watcherResult = new WatcherResult();
        handles.first.close();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());

        mWatcher.start(handles.first, Core.HandleSignals.READABLE, watcherResult);
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());
        mWatcher.cancel();

        mTestRule.runLoopUntilIdle();
        Assert.assertEquals(Integer.MIN_VALUE, watcherResult.getResult());
    }
}