/*
 * Copyright (c) 2003, 2025, 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 4899022 8003887 8355342
 * @summary Look for erroneous representation of drive letter
 * @run junit GetCanonicalPath
 */

import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import java.util.stream.Stream;

import org.junit.jupiter.api.Assumptions;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.condition.EnabledOnOs;
import org.junit.jupiter.api.condition.OS;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;
import org.junit.jupiter.params.provider.ValueSource;

import static org.junit.jupiter.api.Assertions.*;

public class GetCanonicalPath {
    private static Stream<Arguments> pathProviderWindows() {
        List<Arguments> list = new ArrayList<Arguments>();

        File dir = new File(System.getProperty("user.dir", "."));
        char drive = dir.getPath().charAt(0);

        String pathname = drive + ":\\";
        list.add(Arguments.of(pathname, pathname));

        list.add(Arguments.of(drive + ":", dir.toString()));

        String name = "foo";
        pathname = "\\\\?\\" + name;
        list.add(Arguments.of(pathname, new File(dir, name).toString()));
        pathname = "\\\\?\\" + drive + ":" + name;
        list.add(Arguments.of(pathname, new File(dir, name).toString()));

        pathname = "foo\\bar\\gus";
        list.add(Arguments.of(pathname, new File(dir, pathname).toString()));

        pathname = drive + ":\\foo\\bar\\gus";
        list.add(Arguments.of(pathname, pathname));

        pathname = "\\\\server\\share\\foo\\bar\\gus";
        list.add(Arguments.of(pathname, pathname));

        pathname = "\\\\localhost\\" + drive + "$\\Users\\file.dat";
        list.add(Arguments.of(pathname, pathname));

        list.add(Arguments.of("\\\\?\\" + drive + ":\\Users\\file.dat",
                              drive + ":\\Users\\file.dat"));
        list.add(Arguments.of("\\\\?\\UNC\\localhost\\" + drive + "$\\Users\\file.dat",
                              "\\\\localhost\\" + drive + "$\\Users\\file.dat"));

        return list.stream();
    }

    private static Stream<Arguments> pathProviderUnix() {
        return Stream.of(
            Arguments.of("/../../../../../a/b/c", "/a/b/c"),
            Arguments.of("/../../../../../a/../b/c", "/b/c"),
            Arguments.of("/../../../../../a/../../b/c", "/b/c"),
            Arguments.of("/../../../../../a/../../../b/c", "/b/c"),
            Arguments.of("/../../../../../a/../../../../b/c", "/b/c")
        );
    }

    @ParameterizedTest
    @EnabledOnOs({OS.AIX, OS.LINUX, OS.MAC})
    @MethodSource("pathProviderUnix")
    void goodPathsUnix(String pathname, String expected) throws IOException {
        File file = new File(pathname);
        String canonicalPath = file.getCanonicalPath();
        assertEquals(expected, canonicalPath);
    }

    @ParameterizedTest
    @EnabledOnOs(OS.WINDOWS)
    @ValueSource(strings = {"\\\\?", "\\\\?\\UNC", "\\\\?\\UNC\\"})
    void badPathsWindows(String pathname) {
        assertThrows(IOException.class, () -> new File(pathname).getCanonicalPath());
    }

    @ParameterizedTest
    @EnabledOnOs(OS.WINDOWS)
    @MethodSource("pathProviderWindows")
    void goodPathsWindows(String pathname, String expected) throws IOException {
        File file = new File(pathname);
        String canonicalPath = file.getCanonicalPath();
        assertEquals(expected, canonicalPath);
    }

    @Test
    @EnabledOnOs(OS.WINDOWS)
    void driveLetter() throws IOException {
        String path = new File("c:/").getCanonicalPath();
        assertFalse(path.length() > 3, "Drive letter incorrectly represented");
    }

    @Test
    @EnabledOnOs(OS.WINDOWS)
    void mappedDrive() throws IOException {
        // find the first unused drive letter
        char drive = '[';
        var roots = Set.of(new File(".").listRoots());
        for (int i = 4; i < 26; i++) {
            char c = (char)('A' + i);
            if (!roots.contains(new File(c + ":\\"))) {
                drive = c;
                break;
            }
        }
        assertFalse(drive == '['); // '[' is next after 'Z'

        // map the first unused drive letter to the cwd
        String cwd = System.getProperty("user.dir");
        Runtime rt = Runtime.getRuntime();
        String share =
            "\\\\localhost\\" + cwd.charAt(0) + "$" + cwd.substring(2);
        try {
            Process p = rt.exec(new String[] {"net", "use", drive + ":", share});
            assertEquals(0, p.waitFor());
        } catch (InterruptedException x) {
            fail(x);
        }

        // check that the canonical path name and its content are as expected
        try {
            final String filename = "file.txt";
            final String text = "This is some text";
            Files.writeString(Path.of(share, filename), text);
            File file = new File(drive + ":\\" + filename);
            String canonicalPath = file.getCanonicalPath();
            assertEquals(drive + ":\\" + filename, canonicalPath);
            assertEquals(text, Files.readString(Path.of(canonicalPath)));
        } finally {
            try {
                Process p = rt.exec(new String[] {"net", "use", drive + ":", "/Delete"});
                assertEquals(0, p.waitFor());
            } catch (InterruptedException x) {
                fail(x);
            }
        }
    }

    // Create a File with the given pathname and return the File as a Path
    private static Path createFile(String pathname) throws IOException {
        File file = new File(pathname);
        file.deleteOnExit();
        return file.toPath();
    }

    private static boolean supportsLinks = true;
    private static String linkMessage;

    private static Path link;
    private static Path sublink;
    private static Path subsub;

    @BeforeAll
    static void createSymlinks() throws IOException {
        final String DIR     = "dir";
        final String SUBDIR  = "subdir";
        final String TARGET  = "target.txt";
        final String LINK    = "link";
        final String SUBLINK = "sublink";
        final String FILE    = "file.txt";

        // Create directories dir/subdir
        Path dir = createFile(DIR);
        Path subdir = createFile(dir.resolve(SUBDIR).toString());
        Files.createDirectories(subdir);

        // Create file dir/subdir/target.txt
        Path target = createFile(subdir.resolve(TARGET).toString());
        Files.createFile(target);

        // Create symbolic link link -> dir
        link = createFile(Path.of(LINK).toString());
        try {
            Files.createSymbolicLink(link, dir);
        } catch (UnsupportedOperationException | IOException x) {
            if (OS.WINDOWS.isCurrentOs()) {
                supportsLinks = false;
                linkMessage = "\"" + x.getMessage() + "\"";
                return;
            } else {
                throw x;
            }
        }

        sublink = createFile(Path.of(DIR, SUBDIR, SUBLINK).toString());
        Path file = createFile(Path.of(DIR, SUBDIR, FILE).toString());
        Files.createFile(file);

        // Create symbolic link dir/subdir/sublink -> file.txt
        Files.createSymbolicLink(sublink, Path.of(FILE));
        sublink.toFile().deleteOnExit();

        subsub = createFile(Path.of(LINK, SUBDIR, SUBLINK).toString());
    }

    @Test
    void linkToDir() throws IOException {
        Assumptions.assumeTrue(supportsLinks, linkMessage);

        // Check link evaluates to dir
        assertEquals(link.toRealPath().toString(),
                     link.toFile().getCanonicalPath());
    }

    @Test
    void linkToFile() throws IOException {
        Assumptions.assumeTrue(supportsLinks, linkMessage);

        // Check sublink evaluates to file.txt
        assertEquals(sublink.toRealPath().toString(),
                     sublink.toFile().getCanonicalPath());
    }

    @Test
    void linkToFileInSubdir() throws IOException {
        Assumptions.assumeTrue(supportsLinks, linkMessage);

        // Check link/subdir/sublink evaluates to dir/subdir/file.txt
        assertEquals(subsub.toRealPath().toString(),
                     subsub.toFile().getCanonicalPath());
    }
}
