/*
 * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

/*
 * @test
 * @bug 8132734 8144062
 * @summary Test the extended API and the aliasing additions in JarFile that
 *          support multi-release jar files
 * @library /lib/testlibrary/java/util/jar
 * @build Compiler JarBuilder CreateMultiReleaseTestJars
 * @run testng MultiReleaseJarIterators
 */

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.stream.Collectors;
import java.util.zip.ZipFile;

import static java.util.jar.JarFile.Release;

import org.testng.Assert;
import org.testng.annotations.AfterClass;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Test;


public class MultiReleaseJarIterators {

    static final int MAJOR_VERSION = Runtime.version().major();

    String userdir = System.getProperty("user.dir", ".");
    File unversioned = new File(userdir, "unversioned.jar");
    File multirelease = new File(userdir, "multi-release.jar");
    Map<String,JarEntry> uvEntries = new HashMap<>();
    Map<String,JarEntry> mrEntries = new HashMap<>();
    Map<String,JarEntry> baseEntries = new HashMap<>();
    Map<String,JarEntry> v9Entries = new HashMap<>();
    Map<String, JarEntry> v10Entries = new HashMap<>();

    @BeforeClass
    public void initialize() throws Exception {
        CreateMultiReleaseTestJars creator = new CreateMultiReleaseTestJars();
        creator.compileEntries();
        creator.buildUnversionedJar();
        creator.buildMultiReleaseJar();

        try (JarFile jf = new JarFile(multirelease)) {
            for (Enumeration<JarEntry> e = jf.entries(); e.hasMoreElements(); ) {
                JarEntry je = e.nextElement();
                String name = je.getName();
                mrEntries.put(name, je);
                if (name.startsWith("META-INF/versions/")) {
                    if (name.startsWith("META-INF/versions/9/")) {
                        v9Entries.put(name.substring(20), je);
                    } else if (name.startsWith("META-INF/versions/10/")) {
                        v10Entries.put(name.substring(21), je);
                    }
                } else {
                    baseEntries.put(name, je);
                }
            }
        }
        Assert.assertEquals(mrEntries.size(), 14);
        Assert.assertEquals(baseEntries.size(), 6);
        Assert.assertEquals(v9Entries.size(), 5);
        Assert.assertEquals(v10Entries.size(), 3);

        try (JarFile jf = new JarFile(unversioned)) {
            jf.entries().asIterator().forEachRemaining(je -> uvEntries.put(je.getName(), je));
        }
        Assert.assertEquals(uvEntries.size(), 6);
    }

    @AfterClass
    public void close() throws IOException {
        Files.delete(unversioned.toPath());
        Files.delete(multirelease.toPath());
    }

    @Test
    public void testMultiReleaseJar() throws IOException {
        try (JarFile jf = new JarFile(multirelease, true, ZipFile.OPEN_READ)) {
            testEnumeration(jf, mrEntries);
            testStream(jf, mrEntries);
        }

        try (JarFile jf = new JarFile(multirelease, true, ZipFile.OPEN_READ, Release.BASE)) {
            testEnumeration(jf, baseEntries);
            testStream(jf, baseEntries);
        }

        try (JarFile jf = new JarFile(multirelease, true, ZipFile.OPEN_READ, Release.VERSION_9)) {
            testEnumeration(jf, v9Entries);
            testStream(jf, v9Entries);
        }

        try (JarFile jf = new JarFile(multirelease, true, ZipFile.OPEN_READ, Release.RUNTIME)) {
            Map<String,JarEntry> expectedEntries;
            switch (MAJOR_VERSION) {
                case 9:
                    expectedEntries = v9Entries;
                    break;
                case 10:  // won't get here until JDK 10
                    expectedEntries = v10Entries;
                    break;
                default:
                    expectedEntries = baseEntries;
                    break;
            }

            testEnumeration(jf, expectedEntries);
            testStream(jf, expectedEntries);
        }
    }

    @Test
    public void testUnversionedJar() throws IOException {
        try (JarFile jf = new JarFile(unversioned, true, ZipFile.OPEN_READ)) {
            testEnumeration(jf, uvEntries);
            testStream(jf, uvEntries);
        }

        try (JarFile jf = new JarFile(unversioned, true, ZipFile.OPEN_READ, Release.BASE)) {
            testEnumeration(jf, uvEntries);
            testStream(jf, uvEntries);
        }

        try (JarFile jf = new JarFile(unversioned, true, ZipFile.OPEN_READ, Release.VERSION_9)) {
            testEnumeration(jf, uvEntries);
            testStream(jf, uvEntries);
        }

        try (JarFile jf = new JarFile(unversioned, true, ZipFile.OPEN_READ, Release.RUNTIME)) {
            testEnumeration(jf, uvEntries);
            testStream(jf, uvEntries);
        }
    }

    private void testEnumeration(JarFile jf, Map<String,JarEntry> expectedEntries) {
        Map<String, JarEntry> actualEntries = new HashMap<>();
        for (Enumeration<JarEntry> e = jf.entries(); e.hasMoreElements(); ) {
            JarEntry je = e.nextElement();
            actualEntries.put(je.getName(), je);
        }

        testEntries(jf, actualEntries, expectedEntries);
    }


    private void testStream(JarFile jf, Map<String,JarEntry> expectedEntries) {
        Map<String,JarEntry> actualEntries = jf.stream().collect(Collectors.toMap(je -> je.getName(), je -> je));

        testEntries(jf, actualEntries, expectedEntries);
    }

    private void testEntries(JarFile jf, Map<String,JarEntry> actualEntries, Map<String,JarEntry> expectedEntries) {
        /* For multi-release jar files constructed with a Release object,
         * actualEntries contain versionedEntries that are considered part of the
         * public API.  They have a 1-1 correspondence with baseEntries,
         * so entries that are not part of the public API won't be present,
         * i.e. those entries with a name that starts with version/PackagePrivate
         * in this particular jar file (multi-release.jar)
         */

        Map<String,JarEntry> entries;
        if (expectedEntries == mrEntries) {
            Assert.assertEquals(actualEntries.size(), mrEntries.size());
            entries = mrEntries;
        } else if (expectedEntries == uvEntries) {
            Assert.assertEquals(actualEntries.size(), uvEntries.size());
            entries = uvEntries;
        } else {
            Assert.assertEquals(actualEntries.size(), baseEntries.size());  // this is correct
            entries = baseEntries;
        }

        entries.keySet().forEach(name -> {
            JarEntry ee = expectedEntries.get(name);
            if (ee == null) ee = entries.get(name);
            JarEntry ae = actualEntries.get(name);
            try {
                compare(jf, ae, ee);
            } catch (IOException x) {
                throw new RuntimeException(x);
            }
        });
    }

    private void compare(JarFile jf, JarEntry actual, JarEntry expected) throws IOException {
        byte[] abytes;
        byte[] ebytes;

        try (InputStream is = jf.getInputStream(actual)) {
            abytes = is.readAllBytes();
        }

        try (InputStream is = jf.getInputStream(expected)) {
            ebytes = is.readAllBytes();
        }

        Assert.assertEquals(abytes, ebytes);
    }
}