| // Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| package com.android.tools.r8.desugar.desugaredlibrary.jdk11; |
| |
| import static com.android.tools.r8.desugar.desugaredlibrary.test.CompilationSpecification.DEFAULT_SPECIFICATIONS; |
| import static com.android.tools.r8.desugar.desugaredlibrary.test.LibraryDesugaringSpecification.JDK11_PATH; |
| import static java.nio.file.attribute.PosixFilePermission.OWNER_READ; |
| import static java.nio.file.attribute.PosixFilePermission.OWNER_WRITE; |
| |
| import com.android.tools.r8.TestParameters; |
| import com.android.tools.r8.ToolHelper.DexVm.Version; |
| import com.android.tools.r8.desugar.desugaredlibrary.DesugaredLibraryTestBase; |
| import com.android.tools.r8.desugar.desugaredlibrary.test.CompilationSpecification; |
| import com.android.tools.r8.desugar.desugaredlibrary.test.LibraryDesugaringSpecification; |
| import com.android.tools.r8.utils.StringUtils; |
| import com.google.common.collect.ImmutableList; |
| import java.io.IOException; |
| import java.nio.ByteBuffer; |
| import java.nio.channels.FileChannel; |
| import java.nio.channels.SeekableByteChannel; |
| import java.nio.charset.StandardCharsets; |
| import java.nio.file.DirectoryStream; |
| import java.nio.file.Files; |
| import java.nio.file.LinkOption; |
| import java.nio.file.OpenOption; |
| import java.nio.file.Path; |
| import java.nio.file.StandardOpenOption; |
| import java.nio.file.attribute.AclFileAttributeView; |
| import java.nio.file.attribute.BasicFileAttributeView; |
| import java.nio.file.attribute.BasicFileAttributes; |
| import java.nio.file.attribute.DosFileAttributeView; |
| import java.nio.file.attribute.DosFileAttributes; |
| import java.nio.file.attribute.FileOwnerAttributeView; |
| import java.nio.file.attribute.FileTime; |
| import java.nio.file.attribute.PosixFileAttributeView; |
| import java.nio.file.attribute.PosixFileAttributes; |
| import java.nio.file.attribute.PosixFilePermission; |
| import java.nio.file.attribute.UserDefinedFileAttributeView; |
| import java.time.Instant; |
| import java.util.ArrayList; |
| import java.util.Arrays; |
| import java.util.HashSet; |
| import java.util.Iterator; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Set; |
| import java.util.stream.Stream; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.junit.runners.Parameterized; |
| import org.junit.runners.Parameterized.Parameters; |
| |
| @RunWith(Parameterized.class) |
| public class FilesTest extends DesugaredLibraryTestBase { |
| |
| private static final String EXPECTED_RESULT_FORMAT = |
| StringUtils.lines( |
| "bytes written: 11", |
| "String written: Hello World", |
| "bytes read: 11", |
| "String read: Hello World", |
| "bytes read: 11", |
| "String read: Hello World", |
| "true", |
| "%s", |
| "null", |
| "null", |
| "%s", |
| "null", |
| "true", |
| "%s", |
| "unsupported", |
| "tmp", |
| "/", |
| "true", |
| "tmpFile", |
| "This", |
| "is", |
| "fun!", |
| "%s", |
| "%s", |
| "%s", |
| "%s"); |
| private static final List<String> EXPECTED_RESULT_POSIX = |
| ImmutableList.of( |
| "true", |
| "true", |
| "true", |
| "Succeeded with POSIX RO:false", |
| "Successfully set RO with POSIX", |
| "Succeeded with POSIX RO:true"); |
| private static final List<String> EXPECTED_RESULT_DESUGARING_NON_POSIX = |
| ImmutableList.of( |
| "null", |
| "null", |
| "unsupported", |
| "Fail to understand if the file is read-only: class" |
| + " java.lang.UnsupportedOperationException", |
| "Fail to set file as read-only: class java.lang.UnsupportedOperationException", |
| "NotSet"); |
| |
| private final TestParameters parameters; |
| private final LibraryDesugaringSpecification libraryDesugaringSpecification; |
| private final CompilationSpecification compilationSpecification; |
| |
| @Parameters(name = "{0}, spec: {1}, {2}") |
| public static List<Object[]> data() { |
| return buildParameters( |
| // Skip Android 4.4.4 due to missing libjavacrypto. |
| getTestParameters() |
| .withDexRuntime(Version.V4_0_4) |
| .withDexRuntimesStartingFromIncluding(Version.V5_1_1) |
| .withAllApiLevels() |
| .build(), |
| ImmutableList.of(JDK11_PATH), |
| DEFAULT_SPECIFICATIONS); |
| } |
| |
| public FilesTest( |
| TestParameters parameters, |
| LibraryDesugaringSpecification libraryDesugaringSpecification, |
| CompilationSpecification compilationSpecification) { |
| this.parameters = parameters; |
| this.libraryDesugaringSpecification = libraryDesugaringSpecification; |
| this.compilationSpecification = compilationSpecification; |
| } |
| |
| private static String computeExpectedResult(boolean supportPosix, boolean j$nioClasses) { |
| List<String> strings = |
| new ArrayList<>( |
| supportPosix ? EXPECTED_RESULT_POSIX : EXPECTED_RESULT_DESUGARING_NON_POSIX); |
| strings.add(j$nioClasses ? "j$.nio.file.attribute" : "java.nio.file.attribute"); |
| return String.format(EXPECTED_RESULT_FORMAT, strings.toArray()); |
| } |
| |
| private String getExpectedResult() { |
| if (libraryDesugaringSpecification.usesPlatformFileSystem(parameters)) { |
| return libraryDesugaringSpecification.hasNioFileDesugaring(parameters) |
| ? computeExpectedResult(true, true) |
| : computeExpectedResult(true, false); |
| } |
| return computeExpectedResult(false, true); |
| } |
| |
| @Test |
| public void test() throws Throwable { |
| testForDesugaredLibrary(parameters, libraryDesugaringSpecification, compilationSpecification) |
| .addInnerClasses(getClass()) |
| .addKeepMainRule(TestClass.class) |
| .compile() |
| .withArt6Plus64BitsLib() |
| .run(parameters.getRuntime(), TestClass.class) |
| .assertSuccessWithOutput(getExpectedResult()); |
| } |
| |
| public static class TestClass { |
| |
| public static void main(String[] args) throws Throwable { |
| Path path = Files.createTempFile("example", ".txt"); |
| readWriteThroughFilesAPI(path); |
| readThroughFileChannelAPI(path); |
| attributeViewAccess(path); |
| attributeAccess(path); |
| Files.setAttribute(path, "basic:lastModifiedTime", FileTime.from(Instant.EPOCH)); |
| pathGeneric(); |
| lines(path); |
| readOnlyTest(path); |
| fspMethodsWithGeneric(path); |
| } |
| |
| private static void readOnlyTest(Path path) { |
| isReadOnly(path); |
| if (setReadOnly(path)) { |
| isReadOnly(path); |
| } else { |
| System.out.println("NotSet"); |
| } |
| } |
| |
| private static boolean isReadOnly(Path path) { |
| try { |
| // DOS attempt. |
| try { |
| DosFileAttributeView dosFileAttributeView = |
| Files.getFileAttributeView(path, DosFileAttributeView.class); |
| if (dosFileAttributeView != null && dosFileAttributeView.readAttributes() != null) { |
| boolean readOnly = dosFileAttributeView.readAttributes().isReadOnly(); |
| System.out.println("Succeeded with DOS RO:" + readOnly); |
| return readOnly; |
| } |
| } catch (IOException ignored) { |
| } |
| // Posix attempt. |
| Set<PosixFilePermission> posixFilePermissions = Files.getPosixFilePermissions(path); |
| boolean readOnly = |
| posixFilePermissions.contains(OWNER_READ) |
| && !posixFilePermissions.contains(OWNER_WRITE); |
| System.out.println("Succeeded with POSIX RO:" + readOnly); |
| return readOnly; |
| |
| } catch (Throwable t) { |
| System.out.println("Fail to understand if the file is read-only: " + t.getClass()); |
| return false; |
| } |
| } |
| |
| /** Common pattern to set a file as read-only: Try on Dos, on failure, retry on Posix. */ |
| private static boolean setReadOnly(Path path) { |
| try { |
| |
| // DOS attempt. |
| try { |
| DosFileAttributeView dosFileAttributeView = |
| Files.getFileAttributeView(path, DosFileAttributeView.class); |
| if (dosFileAttributeView != null) { |
| dosFileAttributeView.setReadOnly(true); |
| System.out.println("Successfully set RO with DOS"); |
| return true; |
| } |
| } catch (IOException ignored) { |
| } |
| |
| // Posix attempt. |
| Set<PosixFilePermission> permissions = Files.getPosixFilePermissions(path); |
| List<PosixFilePermission> readPermissions = Arrays.asList(OWNER_READ); |
| List<PosixFilePermission> writePermissions = Arrays.asList(PosixFilePermission.OWNER_WRITE); |
| permissions.addAll(readPermissions); |
| permissions.removeAll(writePermissions); |
| Files.setPosixFilePermissions(path, permissions); |
| System.out.println("Successfully set RO with POSIX"); |
| return true; |
| |
| } catch (Throwable t) { |
| System.out.println("Fail to set file as read-only: " + t.getClass()); |
| return false; |
| } |
| } |
| |
| private static void pathGeneric() throws IOException { |
| Path tmpDict = Files.createTempDirectory("tmpDict"); |
| Path tmpFile = Files.createFile(tmpDict.resolve("tmpFile")); |
| Iterator<Path> iterator = tmpDict.iterator(); |
| System.out.println(iterator.next()); |
| Iterable<Path> rootDirectories = tmpFile.getFileSystem().getRootDirectories(); |
| System.out.println(rootDirectories.iterator().next()); |
| DirectoryStream<Path> paths = Files.newDirectoryStream(tmpDict); |
| Iterator<Path> theIterator = paths.iterator(); |
| System.out.println(theIterator.hasNext()); |
| System.out.println(theIterator.next().getFileName()); |
| } |
| |
| private static void fspMethodsWithGeneric(Path path) throws IOException { |
| Map<String, Object> mapping = Files.readAttributes(path, "lastModifiedTime"); |
| System.out.println(mapping.values().iterator().next().getClass().getPackage().getName()); |
| } |
| |
| private static void attributeViewAccess(Path path) throws IOException { |
| BasicFileAttributeView basicView = |
| Files.getFileAttributeView(path, BasicFileAttributeView.class); |
| if (basicView != null) { |
| System.out.println(basicView.readAttributes().isRegularFile()); |
| } else { |
| System.out.println("null"); |
| } |
| |
| PosixFileAttributeView posixView = |
| Files.getFileAttributeView(path, PosixFileAttributeView.class); |
| if (posixView != null) { |
| System.out.println(posixView.readAttributes().permissions().contains(OWNER_READ)); |
| } else { |
| System.out.println("null"); |
| } |
| |
| try { |
| DosFileAttributeView dosView = Files.getFileAttributeView(path, DosFileAttributeView.class); |
| if (dosView != null) { |
| System.out.println(dosView.readAttributes().isReadOnly()); |
| } else { |
| System.out.println("null"); |
| } |
| } catch (UnsupportedOperationException e) { |
| System.out.println("unsupported"); |
| } |
| |
| try { |
| AclFileAttributeView aclView = Files.getFileAttributeView(path, AclFileAttributeView.class); |
| if (aclView != null) { |
| System.out.println(aclView.getAcl().isEmpty()); |
| } else { |
| System.out.println("null"); |
| } |
| } catch (UnsupportedOperationException e) { |
| System.out.println("unsupported"); |
| } |
| |
| try { |
| FileOwnerAttributeView foView = |
| Files.getFileAttributeView(path, FileOwnerAttributeView.class); |
| if (foView != null) { |
| System.out.println(foView.getOwner() != null); |
| } else { |
| System.out.println("null"); |
| } |
| } catch (UnsupportedOperationException e) { |
| System.out.println("unsupported"); |
| } |
| |
| try { |
| UserDefinedFileAttributeView udView = |
| Files.getFileAttributeView(path, UserDefinedFileAttributeView.class); |
| if (udView != null) { |
| System.out.println(udView.name()); |
| } else { |
| System.out.println("null"); |
| } |
| } catch (UnsupportedOperationException e) { |
| System.out.println("unsupported"); |
| } |
| } |
| |
| private static void attributeAccess(Path path) throws IOException { |
| BasicFileAttributes attributes = Files.readAttributes(path, BasicFileAttributes.class); |
| if (attributes != null) { |
| System.out.println(attributes.isRegularFile()); |
| } else { |
| System.out.println("null"); |
| } |
| |
| try { |
| PosixFileAttributes posixAttributes = Files.readAttributes(path, PosixFileAttributes.class); |
| if (posixAttributes != null) { |
| System.out.println(posixAttributes.permissions().contains(OWNER_READ)); |
| } else { |
| System.out.println("null"); |
| } |
| } catch (UnsupportedOperationException e) { |
| System.out.println("unsupported"); |
| } |
| |
| try { |
| DosFileAttributes dosFileAttributes = Files.readAttributes(path, DosFileAttributes.class); |
| if (dosFileAttributes != null) { |
| System.out.println(dosFileAttributes.isReadOnly()); |
| } else { |
| System.out.println("null"); |
| } |
| } catch (UnsupportedOperationException e) { |
| System.out.println("unsupported"); |
| } |
| } |
| |
| private static void lines(Path path) throws IOException { |
| Files.write(path, "This\nis\nfun!".getBytes(StandardCharsets.UTF_8)); |
| Stream<String> lines = Files.lines(path, StandardCharsets.UTF_8); |
| lines.forEach(System.out::println); |
| } |
| |
| private static void readWriteThroughFilesAPI(Path path) throws IOException { |
| try (SeekableByteChannel channel = |
| Files.newByteChannel(path, StandardOpenOption.READ, StandardOpenOption.WRITE)) { |
| String toWrite = "Hello World"; |
| |
| // Write the String toWrite into the channel. |
| ByteBuffer byteBuffer = ByteBuffer.wrap(toWrite.getBytes()); |
| int write = channel.write(byteBuffer); |
| System.out.println("bytes written: " + write); |
| System.out.println("String written: " + toWrite); |
| |
| // Read the String toWrite from the channel. |
| channel.position(0); |
| ByteBuffer byteBuffer2 = ByteBuffer.allocate(write); |
| int read = channel.read(byteBuffer2); |
| System.out.println("bytes read: " + read); |
| System.out.println("String read: " + new String(byteBuffer2.array())); |
| } |
| } |
| |
| private static void readThroughFileChannelAPI(Path path) throws IOException { |
| try { |
| Set<OpenOption> openOptions = new HashSet<>(); |
| openOptions.add(LinkOption.NOFOLLOW_LINKS); |
| try (FileChannel channel = FileChannel.open(path, openOptions)) { |
| String toWrite = "Hello World"; |
| |
| // Read the String toWrite from the channel. |
| channel.position(0); |
| ByteBuffer byteBuffer2 = ByteBuffer.allocate(toWrite.length()); |
| int read = channel.read(byteBuffer2); |
| System.out.println("bytes read: " + read); |
| System.out.println("String read: " + new String(byteBuffer2.array())); |
| } |
| } catch (NoClassDefFoundError err) { |
| // TODO(b/222647019): FileChannel#open is not supported in between 24 and 26. |
| System.out.println("unsupported"); |
| System.out.println("unsupported"); |
| } |
| } |
| } |
| } |