/*
 * 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.cts.deviceowner;

import static junit.framework.Assert.assertEquals;
import static junit.framework.Assert.assertFalse;
import static junit.framework.Assert.assertTrue;
import static junit.framework.Assert.fail;

import static org.junit.Assert.assertArrayEquals;

import android.app.admin.DevicePolicyManager;
import android.content.ComponentName;
import android.content.Context;
import android.support.test.InstrumentationRegistry;
import android.support.test.runner.AndroidJUnit4;

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

import java.util.Collections;
import java.util.List;

@RunWith(AndroidJUnit4.class)
public class AffiliationTest {

    private DevicePolicyManager mDevicePolicyManager;
    private ComponentName mAdminComponent;

    @Before
    public void setUp() {
        Context context = InstrumentationRegistry.getContext();
        mDevicePolicyManager = (DevicePolicyManager)
                context.getSystemService(Context.DEVICE_POLICY_SERVICE);
        mAdminComponent = BaseDeviceOwnerTest.BasicAdminReceiver.getComponentName(context);
    }

    @Test
    public void testSetAffiliationId1() {
        setAffiliationIds(Collections.singletonList("id.number.1"));
    }

    @Test
    public void testSetAffiliationId2() {
        setAffiliationIds(Collections.singletonList("id.number.2"));
    }

    @Test
    public void testLockTaskMethodsThrowExceptionIfUnaffiliated() {
        checkLockTaskMethodsThrow();
    }

    /** Assumes that the calling user is already affiliated before calling this method */
    @Test
    public void testSetLockTaskPackagesClearedIfUserBecomesUnaffiliated() {
        final String[] packages = {"package1", "package2"};
        mDevicePolicyManager.setLockTaskPackages(mAdminComponent, packages);
        assertArrayEquals(packages, mDevicePolicyManager.getLockTaskPackages(mAdminComponent));
        assertTrue(mDevicePolicyManager.isLockTaskPermitted("package1"));
        assertFalse(mDevicePolicyManager.isLockTaskPermitted("package3"));

        final List<String> previousAffiliationIds =
                mDevicePolicyManager.getAffiliationIds(mAdminComponent);
        try {
            // Clearing affiliation ids for this user. Lock task methods unavailable.
            setAffiliationIds(Collections.<String>emptyList());
            checkLockTaskMethodsThrow();
            assertFalse(mDevicePolicyManager.isLockTaskPermitted("package1"));

            // Affiliating the user again. Previously set packages have been cleared.
            setAffiliationIds(previousAffiliationIds);
            assertEquals(0, mDevicePolicyManager.getLockTaskPackages(mAdminComponent).length);
            assertFalse(mDevicePolicyManager.isLockTaskPermitted("package1"));
        } finally {
            mDevicePolicyManager.setAffiliationIds(mAdminComponent, previousAffiliationIds);
        }
    }

    private void setAffiliationIds(List<String> ids) {
        mDevicePolicyManager.setAffiliationIds(mAdminComponent, ids);
        assertEquals(ids, mDevicePolicyManager.getAffiliationIds(mAdminComponent));
    }

    private void checkLockTaskMethodsThrow() {
        try {
            mDevicePolicyManager.setLockTaskPackages(mAdminComponent, new String[0]);
            fail("setLockTaskPackages did not throw expected SecurityException");
        } catch (SecurityException expected) {
        }
        try {
            mDevicePolicyManager.getLockTaskPackages(mAdminComponent);
            fail("getLockTaskPackages did not throw expected SecurityException");
        } catch (SecurityException expected) {
        }
    }
}
