1
0
mirror of https://github.com/SquidDev-CC/CC-Tweaked synced 2025-10-25 02:47:39 +00:00

Make mount error messages a bit more consistent

- Move most error message constants to a new MountHelpers class.
 - Be a little more consistent in when we throw "No such file" vs "Not a
   file/directory" messages.
This commit is contained in:
Jonathan Coates
2023-10-22 13:13:07 +01:00
parent cab66a2d6e
commit 09e521727f
16 changed files with 197 additions and 86 deletions

View File

@@ -21,6 +21,8 @@ import java.io.IOException;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import static dan200.computercraft.core.filesystem.MountHelpers.NO_SUCH_FILE;
/** /**
* A mount backed by Minecraft's {@link ResourceManager}. * A mount backed by Minecraft's {@link ResourceManager}.
* *

View File

@@ -17,14 +17,14 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.function.Function; import java.util.function.Function;
import static dan200.computercraft.core.filesystem.MountHelpers.*;
/** /**
* An abstract mount which stores its file tree in memory. * An abstract mount which stores its file tree in memory.
* *
* @param <T> The type of file. * @param <T> The type of file.
*/ */
public abstract class AbstractInMemoryMount<T extends AbstractInMemoryMount.FileEntry<T>> implements Mount { public abstract class AbstractInMemoryMount<T extends AbstractInMemoryMount.FileEntry<T>> implements Mount {
protected static final String NO_SUCH_FILE = "No such file";
@Nullable @Nullable
protected T root; protected T root;
@@ -57,7 +57,8 @@ public abstract class AbstractInMemoryMount<T extends AbstractInMemoryMount.File
@Override @Override
public final void list(String path, List<String> contents) throws IOException { public final void list(String path, List<String> contents) throws IOException {
var file = get(path); var file = get(path);
if (file == null || file.children == null) throw new FileOperationException(path, "Not a directory"); if (file == null) throw new FileOperationException(path, NO_SUCH_FILE);
if (file.children == null) throw new FileOperationException(path, NOT_A_DIRECTORY);
contents.addAll(file.children.keySet()); contents.addAll(file.children.keySet());
} }
@@ -82,7 +83,8 @@ public abstract class AbstractInMemoryMount<T extends AbstractInMemoryMount.File
@Override @Override
public final SeekableByteChannel openForRead(String path) throws IOException { public final SeekableByteChannel openForRead(String path) throws IOException {
var file = get(path); var file = get(path);
if (file == null || file.isDirectory()) throw new FileOperationException(path, NO_SUCH_FILE); if (file == null) throw new FileOperationException(path, NO_SUCH_FILE);
if (file.isDirectory()) throw new FileOperationException(path, NOT_A_FILE);
return openForRead(path, file); return openForRead(path, file);
} }

View File

@@ -21,8 +21,6 @@ import java.util.concurrent.TimeUnit;
* @param <T> The type of file. * @param <T> The type of file.
*/ */
public abstract class ArchiveMount<T extends ArchiveMount.FileEntry<T>> extends AbstractInMemoryMount<T> { public abstract class ArchiveMount<T extends ArchiveMount.FileEntry<T>> extends AbstractInMemoryMount<T> {
protected static final String NO_SUCH_FILE = "No such file";
/** /**
* Limit the entire cache to 64MiB. * Limit the entire cache to 64MiB.
*/ */

View File

@@ -17,6 +17,9 @@ import java.nio.file.attribute.BasicFileAttributes;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Set;
import static dan200.computercraft.core.filesystem.MountHelpers.NOT_A_FILE;
import static dan200.computercraft.core.filesystem.MountHelpers.NO_SUCH_FILE;
/** /**
* A {@link Mount} implementation which provides read-only access to a directory. * A {@link Mount} implementation which provides read-only access to a directory.
*/ */
@@ -84,7 +87,9 @@ public class FileMount implements Mount {
@Override @Override
public SeekableByteChannel openForRead(String path) throws FileOperationException { public SeekableByteChannel openForRead(String path) throws FileOperationException {
var file = resolvePath(path); var file = resolvePath(path);
if (!Files.isRegularFile(file)) throw new FileOperationException(path, "No such file"); if (!Files.isRegularFile(file)) {
throw new FileOperationException(path, Files.exists(file) ? NOT_A_FILE : NO_SUCH_FILE);
}
try { try {
return Files.newByteChannel(file, READ_OPTIONS); return Files.newByteChannel(file, READ_OPTIONS);
@@ -103,7 +108,7 @@ public class FileMount implements Mount {
protected FileOperationException remapException(String fallbackPath, IOException exn) { protected FileOperationException remapException(String fallbackPath, IOException exn) {
return exn instanceof FileSystemException fsExn return exn instanceof FileSystemException fsExn
? remapException(fallbackPath, fsExn) ? remapException(fallbackPath, fsExn)
: new FileOperationException(fallbackPath, exn.getMessage() == null ? "Access denied" : exn.getMessage()); : new FileOperationException(fallbackPath, exn.getMessage() == null ? MountHelpers.ACCESS_DENIED : exn.getMessage());
} }
/** /**
@@ -115,7 +120,7 @@ public class FileMount implements Mount {
* @return The wrapped exception. * @return The wrapped exception.
*/ */
protected FileOperationException remapException(String fallbackPath, FileSystemException exn) { protected FileOperationException remapException(String fallbackPath, FileSystemException exn) {
var reason = getReason(exn); var reason = MountHelpers.getReason(exn);
var failedFile = exn.getFile(); var failedFile = exn.getFile();
if (failedFile == null) return new FileOperationException(fallbackPath, reason); if (failedFile == null) return new FileOperationException(fallbackPath, reason);
@@ -125,20 +130,4 @@ public class FileMount implements Mount {
? new FileOperationException(Joiner.on('/').join(root.relativize(failedPath)), reason) ? new FileOperationException(Joiner.on('/').join(root.relativize(failedPath)), reason)
: new FileOperationException(fallbackPath, reason); : new FileOperationException(fallbackPath, reason);
} }
/**
* Get the user-friendly reason for a {@link FileSystemException}.
*
* @param exn The exception that occurred.
* @return The friendly reason for this exception.
*/
protected String getReason(FileSystemException exn) {
if (exn instanceof FileAlreadyExistsException) return "File exists";
if (exn instanceof NoSuchFileException) return "No such file";
if (exn instanceof NotDirectoryException) return "Not a directory";
if (exn instanceof AccessDeniedException) return "Access denied";
var reason = exn.getReason();
return reason != null ? reason.trim() : "Operation failed";
}
} }

View File

@@ -24,6 +24,8 @@ import java.util.*;
import java.util.function.Function; import java.util.function.Function;
import java.util.regex.Pattern; import java.util.regex.Pattern;
import static dan200.computercraft.core.filesystem.MountHelpers.*;
public class FileSystem { public class FileSystem {
/** /**
* Maximum depth that {@link #copyRecursive(String, MountWrapper, String, MountWrapper, int)} will descend into. * Maximum depth that {@link #copyRecursive(String, MountWrapper, String, MountWrapper, int)} will descend into.
@@ -206,9 +208,9 @@ public class FileSystem {
sourcePath = sanitizePath(sourcePath); sourcePath = sanitizePath(sourcePath);
destPath = sanitizePath(destPath); destPath = sanitizePath(destPath);
if (isReadOnly(sourcePath) || isReadOnly(destPath)) throw new FileSystemException("Access denied"); if (isReadOnly(sourcePath) || isReadOnly(destPath)) throw new FileSystemException(ACCESS_DENIED);
if (!exists(sourcePath)) throw new FileSystemException("No such file"); if (!exists(sourcePath)) throw new FileSystemException(NO_SUCH_FILE);
if (exists(destPath)) throw new FileSystemException("File exists"); if (exists(destPath)) throw new FileSystemException(FILE_EXISTS);
if (contains(sourcePath, destPath)) throw new FileSystemException("Can't move a directory inside itself"); if (contains(sourcePath, destPath)) throw new FileSystemException("Can't move a directory inside itself");
var mount = getMount(sourcePath); var mount = getMount(sourcePath);
@@ -223,15 +225,9 @@ public class FileSystem {
public synchronized void copy(String sourcePath, String destPath) throws FileSystemException { public synchronized void copy(String sourcePath, String destPath) throws FileSystemException {
sourcePath = sanitizePath(sourcePath); sourcePath = sanitizePath(sourcePath);
destPath = sanitizePath(destPath); destPath = sanitizePath(destPath);
if (isReadOnly(destPath)) { if (isReadOnly(destPath)) throw new FileSystemException("/" + destPath + ": " + ACCESS_DENIED);
throw new FileSystemException("/" + destPath + ": Access denied"); if (!exists(sourcePath)) throw new FileSystemException("/" + sourcePath + ": " + NO_SUCH_FILE);
} if (exists(destPath)) throw new FileSystemException("/" + destPath + ": " + FILE_EXISTS);
if (!exists(sourcePath)) {
throw new FileSystemException("/" + sourcePath + ": No such file");
}
if (exists(destPath)) {
throw new FileSystemException("/" + destPath + ": File exists");
}
if (contains(sourcePath, destPath)) { if (contains(sourcePath, destPath)) {
throw new FileSystemException("/" + sourcePath + ": Can't copy a directory inside itself"); throw new FileSystemException("/" + sourcePath + ": Can't copy a directory inside itself");
} }
@@ -264,7 +260,7 @@ public class FileSystem {
// Copy bytes as fast as we can // Copy bytes as fast as we can
ByteStreams.copy(source, destination); ByteStreams.copy(source, destination);
} catch (AccessDeniedException e) { } catch (AccessDeniedException e) {
throw new FileSystemException("Access denied"); throw new FileSystemException(ACCESS_DENIED);
} catch (IOException e) { } catch (IOException e) {
throw FileSystemException.of(e); throw FileSystemException.of(e);
} }

View File

@@ -7,6 +7,8 @@ package dan200.computercraft.core.filesystem;
import java.io.IOException; import java.io.IOException;
import java.io.Serial; import java.io.Serial;
import static dan200.computercraft.core.filesystem.MountHelpers.ACCESS_DENIED;
public class FileSystemException extends Exception { public class FileSystemException extends Exception {
@Serial @Serial
private static final long serialVersionUID = -2500631644868104029L; private static final long serialVersionUID = -2500631644868104029L;
@@ -20,6 +22,6 @@ public class FileSystemException extends Exception {
} }
public static String getMessage(IOException e) { public static String getMessage(IOException e) {
return e.getMessage() == null ? "Access denied" : e.getMessage(); return e.getMessage() == null ? ACCESS_DENIED : e.getMessage();
} }
} }

View File

@@ -14,11 +14,12 @@ import java.io.FileNotFoundException;
import java.io.IOException; import java.io.IOException;
import java.nio.file.attribute.BasicFileAttributes; import java.nio.file.attribute.BasicFileAttributes;
import java.nio.file.attribute.FileTime; import java.nio.file.attribute.FileTime;
import java.time.Instant;
import java.util.HashMap; import java.util.HashMap;
import java.util.zip.ZipEntry; import java.util.zip.ZipEntry;
import java.util.zip.ZipFile; import java.util.zip.ZipFile;
import static dan200.computercraft.core.filesystem.MountHelpers.NO_SUCH_FILE;
/** /**
* A mount which reads zip/jar files. * A mount which reads zip/jar files.
*/ */
@@ -95,9 +96,7 @@ public final class JarMount extends ArchiveMount<JarMount.FileEntry> implements
} }
} }
private static final FileTime EPOCH = FileTime.from(Instant.EPOCH);
private static FileTime orEpoch(@Nullable FileTime time) { private static FileTime orEpoch(@Nullable FileTime time) {
return time == null ? EPOCH : time; return time == null ? MountHelpers.EPOCH : time;
} }
} }

View File

@@ -22,14 +22,13 @@ import java.nio.file.attribute.FileTime;
import java.time.Instant; import java.time.Instant;
import java.util.*; import java.util.*;
import static dan200.computercraft.core.filesystem.WritableFileMount.MINIMUM_FILE_SIZE; import static dan200.computercraft.core.filesystem.MountHelpers.*;
/** /**
* A basic {@link Mount} which stores files and directories in-memory. * A basic {@link Mount} which stores files and directories in-memory.
*/ */
public final class MemoryMount extends AbstractInMemoryMount<MemoryMount.FileEntry> implements WritableMount { public final class MemoryMount extends AbstractInMemoryMount<MemoryMount.FileEntry> implements WritableMount {
private static final byte[] EMPTY = new byte[0]; private static final byte[] EMPTY = new byte[0];
private static final FileTime EPOCH = FileTime.from(Instant.EPOCH);
private final long capacity; private final long capacity;
@@ -80,7 +79,7 @@ public final class MemoryMount extends AbstractInMemoryMount<MemoryMount.FileEnt
@Override @Override
protected SeekableByteChannel openForRead(String path, FileEntry file) throws IOException { protected SeekableByteChannel openForRead(String path, FileEntry file) throws IOException {
if (file.contents == null) throw new FileOperationException(path, "File is a directory"); if (file.contents == null) throw new FileOperationException(path, NOT_A_FILE);
return new EntryChannel(file, 0); return new EntryChannel(file, 0);
} }
@@ -119,7 +118,7 @@ public final class MemoryMount extends AbstractInMemoryMount<MemoryMount.FileEnt
if (nextEntry == null) { if (nextEntry == null) {
lastEntry.children.put(part, nextEntry = FileEntry.newDir()); lastEntry.children.put(part, nextEntry = FileEntry.newDir());
} else if (nextEntry.children == null) { } else if (nextEntry.children == null) {
throw new FileOperationException(path, "File exists"); throw new FileOperationException(path, FILE_EXISTS);
} }
lastEntry = nextEntry; lastEntry = nextEntry;
@@ -129,7 +128,7 @@ public final class MemoryMount extends AbstractInMemoryMount<MemoryMount.FileEnt
@Override @Override
public void delete(String path) throws IOException { public void delete(String path) throws IOException {
if (path.isEmpty()) throw new AccessDeniedException("Access denied"); if (path.isEmpty()) throw new AccessDeniedException(ACCESS_DENIED);
var node = getParentAndName(path); var node = getParentAndName(path);
if (node != null) node.parent().remove(node.name()); if (node != null) node.parent().remove(node.name());
} }
@@ -139,23 +138,23 @@ public final class MemoryMount extends AbstractInMemoryMount<MemoryMount.FileEnt
if (dest.startsWith(source)) throw new FileOperationException(source, "Cannot move a directory inside itself"); if (dest.startsWith(source)) throw new FileOperationException(source, "Cannot move a directory inside itself");
var sourceParent = getParentAndName(source); var sourceParent = getParentAndName(source);
if (sourceParent == null || !sourceParent.exists()) throw new FileOperationException(source, "No such file"); if (sourceParent == null || !sourceParent.exists()) throw new FileOperationException(source, NO_SUCH_FILE);
var destParent = getParentAndName(dest); var destParent = getParentAndName(dest);
if (destParent == null) throw new FileOperationException(dest, "Parent directory does not exist"); if (destParent == null) throw new FileOperationException(dest, "Parent directory does not exist");
if (destParent.exists()) throw new FileOperationException(dest, "File exists"); if (destParent.exists()) throw new FileOperationException(dest, FILE_EXISTS);
destParent.put(sourceParent.parent().remove(sourceParent.name())); destParent.put(sourceParent.parent().remove(sourceParent.name()));
} }
private FileEntry getForWrite(String path) throws FileOperationException { private FileEntry getForWrite(String path) throws FileOperationException {
if (path.isEmpty()) throw new FileOperationException(path, "Cannot write to directory"); if (path.isEmpty()) throw new FileOperationException(path, CANNOT_WRITE_TO_DIRECTORY);
var parent = getParentAndName(path); var parent = getParentAndName(path);
if (parent == null) throw new FileOperationException(path, "Parent directory does not exist"); if (parent == null) throw new FileOperationException(path, "Parent directory does not exist");
var file = parent.get(); var file = parent.get();
if (file != null && file.isDirectory()) throw new FileOperationException(path, "Cannot write to directory"); if (file != null && file.isDirectory()) throw new FileOperationException(path, CANNOT_WRITE_TO_DIRECTORY);
if (file == null) parent.put(file = FileEntry.newFile()); if (file == null) parent.put(file = FileEntry.newFile());
return file; return file;

View File

@@ -0,0 +1,88 @@
// SPDX-FileCopyrightText: 2023 The CC: Tweaked Developers
//
// SPDX-License-Identifier: MPL-2.0
package dan200.computercraft.core.filesystem;
import dan200.computercraft.api.filesystem.Mount;
import dan200.computercraft.api.filesystem.WritableMount;
import java.nio.file.FileSystemException;
import java.nio.file.*;
import java.nio.file.attribute.BasicFileAttributes;
import java.nio.file.attribute.FileTime;
import java.time.Instant;
import java.util.List;
/**
* Useful constants and helper functions for working with mounts.
*/
public final class MountHelpers {
/**
* A {@link FileTime} set to the Unix EPOCH, intended for {@link BasicFileAttributes}'s file times.
*/
public static final FileTime EPOCH = FileTime.from(Instant.EPOCH);
/**
* The minimum size of a file for file {@linkplain WritableMount#getCapacity() capacity calculations}.
*/
public static final long MINIMUM_FILE_SIZE = 500;
/**
* The error message used when the file does not exist.
*/
public static final String NO_SUCH_FILE = "No such file";
/**
* The error message used when trying to use a file as a directory (for instance when
* {@linkplain Mount#list(String, List) listing its contents}).
*/
public static final String NOT_A_DIRECTORY = "Not a directory";
/**
* The error message used when trying to use a directory as a file (for instance when
* {@linkplain Mount#openForRead(String) opening for reading}).
*/
public static final String NOT_A_FILE = "Not a file";
/**
* The error message used when attempting to modify a read-only file or mount.
*/
public static final String ACCESS_DENIED = "Access denied";
/**
* The error message used when trying to overwrite a file (for instance when
* {@linkplain WritableMount#rename(String, String) renaming files} or {@linkplain WritableMount#makeDirectory(String)
* creating directories}).
*/
public static final String FILE_EXISTS = "File exists";
/**
* The error message used when trying to {@linkplain WritableMount#openForWrite(String) opening a directory to read}.
*/
public static final String CANNOT_WRITE_TO_DIRECTORY = "Cannot write to directory";
/**
* The error message used when the mount runs out of space.
*/
public static final String OUT_OF_SPACE = "Out of space";
private MountHelpers() {
}
/**
* Get the user-friendly reason for a {@link java.nio.file.FileSystemException}.
*
* @param exn The exception that occurred.
* @return The friendly reason for this exception.
*/
public static String getReason(FileSystemException exn) {
if (exn instanceof FileAlreadyExistsException) return FILE_EXISTS;
if (exn instanceof NoSuchFileException) return NO_SUCH_FILE;
if (exn instanceof NotDirectoryException) return NOT_A_DIRECTORY;
if (exn instanceof AccessDeniedException) return ACCESS_DENIED;
var reason = exn.getReason();
return reason != null ? reason.trim() : "Operation failed";
}
}

View File

@@ -15,6 +15,8 @@ import java.nio.file.attribute.BasicFileAttributes;
import java.util.List; import java.util.List;
import java.util.OptionalLong; import java.util.OptionalLong;
import static dan200.computercraft.core.filesystem.MountHelpers.*;
class MountWrapper { class MountWrapper {
private final String label; private final String label;
private final String location; private final String location;
@@ -87,9 +89,8 @@ class MountWrapper {
public void list(String path, List<String> contents) throws FileSystemException { public void list(String path, List<String> contents) throws FileSystemException {
path = toLocal(path); path = toLocal(path);
try { try {
if (!mount.exists(path) || !mount.isDirectory(path)) { if (!mount.exists(path)) throw localExceptionOf(path, NO_SUCH_FILE);
throw localExceptionOf(path, "Not a directory"); if (!mount.isDirectory(path)) throw localExceptionOf(path, NOT_A_DIRECTORY);
}
mount.list(path, contents); mount.list(path, contents);
} catch (IOException e) { } catch (IOException e) {
@@ -125,7 +126,7 @@ class MountWrapper {
} }
public void makeDirectory(String path) throws FileSystemException { public void makeDirectory(String path) throws FileSystemException {
if (writableMount == null) throw exceptionOf(path, "Access denied"); if (writableMount == null) throw exceptionOf(path, ACCESS_DENIED);
path = toLocal(path); path = toLocal(path);
try { try {
@@ -136,7 +137,7 @@ class MountWrapper {
} }
public void delete(String path) throws FileSystemException { public void delete(String path) throws FileSystemException {
if (writableMount == null) throw exceptionOf(path, "Access denied"); if (writableMount == null) throw exceptionOf(path, ACCESS_DENIED);
path = toLocal(path); path = toLocal(path);
try { try {
@@ -147,7 +148,7 @@ class MountWrapper {
} }
public void rename(String source, String dest) throws FileSystemException { public void rename(String source, String dest) throws FileSystemException {
if (writableMount == null) throw exceptionOf(source, "Access denied"); if (writableMount == null) throw exceptionOf(source, ACCESS_DENIED);
source = toLocal(source); source = toLocal(source);
dest = toLocal(dest); dest = toLocal(dest);
@@ -164,12 +165,12 @@ class MountWrapper {
} }
public SeekableByteChannel openForWrite(String path) throws FileSystemException { public SeekableByteChannel openForWrite(String path) throws FileSystemException {
if (writableMount == null) throw exceptionOf(path, "Access denied"); if (writableMount == null) throw exceptionOf(path, ACCESS_DENIED);
path = toLocal(path); path = toLocal(path);
try { try {
if (mount.exists(path) && mount.isDirectory(path)) { if (mount.exists(path) && mount.isDirectory(path)) {
throw localExceptionOf(path, "Cannot write to directory"); throw localExceptionOf(path, CANNOT_WRITE_TO_DIRECTORY);
} else { } else {
if (!path.isEmpty()) { if (!path.isEmpty()) {
var dir = FileSystem.getDirectory(path); var dir = FileSystem.getDirectory(path);
@@ -185,7 +186,7 @@ class MountWrapper {
} }
public SeekableByteChannel openForAppend(String path) throws FileSystemException { public SeekableByteChannel openForAppend(String path) throws FileSystemException {
if (writableMount == null) throw exceptionOf(path, "Access denied"); if (writableMount == null) throw exceptionOf(path, ACCESS_DENIED);
path = toLocal(path); path = toLocal(path);
try { try {
@@ -198,7 +199,7 @@ class MountWrapper {
} }
return writableMount.openForWrite(path); return writableMount.openForWrite(path);
} else if (mount.isDirectory(path)) { } else if (mount.isDirectory(path)) {
throw localExceptionOf(path, "Cannot write to directory"); throw localExceptionOf(path, CANNOT_WRITE_TO_DIRECTORY);
} else { } else {
return writableMount.openForAppend(path); return writableMount.openForAppend(path);
} }
@@ -220,7 +221,7 @@ class MountWrapper {
// This error will contain the absolute path, leaking information about where MC is installed. We drop that, // This error will contain the absolute path, leaking information about where MC is installed. We drop that,
// just taking the reason. We assume that the error refers to the input path. // just taking the reason. We assume that the error refers to the input path.
var message = ex.getReason(); var message = ex.getReason();
if (message == null) message = "Access denied"; if (message == null) message = ACCESS_DENIED;
return localExceptionOf(localPath, message); return localExceptionOf(localPath, message);
} }

View File

@@ -20,13 +20,14 @@ import java.nio.file.*;
import java.nio.file.attribute.BasicFileAttributes; import java.nio.file.attribute.BasicFileAttributes;
import java.util.Set; import java.util.Set;
import static dan200.computercraft.core.filesystem.MountHelpers.*;
/** /**
* A {@link WritableFileMount} implementation which provides read-write access to a directory. * A {@link WritableFileMount} implementation which provides read-write access to a directory.
*/ */
public class WritableFileMount extends FileMount implements WritableMount { public class WritableFileMount extends FileMount implements WritableMount {
private static final Logger LOG = LoggerFactory.getLogger(WritableFileMount.class); private static final Logger LOG = LoggerFactory.getLogger(WritableFileMount.class);
static final long MINIMUM_FILE_SIZE = 500;
private static final Set<OpenOption> WRITE_OPTIONS = Set.of(StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING); private static final Set<OpenOption> WRITE_OPTIONS = Set.of(StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);
private static final Set<OpenOption> APPEND_OPTIONS = Set.of(StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.APPEND); private static final Set<OpenOption> APPEND_OPTIONS = Set.of(StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.APPEND);
@@ -49,7 +50,7 @@ public class WritableFileMount extends FileMount implements WritableMount {
try { try {
Files.createDirectories(root); Files.createDirectories(root);
} catch (IOException e) { } catch (IOException e) {
throw new FileOperationException("Access denied"); throw new FileOperationException(ACCESS_DENIED);
} }
} }
@@ -78,7 +79,7 @@ public class WritableFileMount extends FileMount implements WritableMount {
create(); create();
var file = resolveFile(path); var file = resolveFile(path);
if (file.exists()) { if (file.exists()) {
if (!file.isDirectory()) throw new FileOperationException(path, "File exists"); if (!file.isDirectory()) throw new FileOperationException(path, FILE_EXISTS);
return; return;
} }
@@ -90,19 +91,19 @@ public class WritableFileMount extends FileMount implements WritableMount {
} }
if (getRemainingSpace() < dirsToCreate * MINIMUM_FILE_SIZE) { if (getRemainingSpace() < dirsToCreate * MINIMUM_FILE_SIZE) {
throw new FileOperationException(path, "Out of space"); throw new FileOperationException(path, OUT_OF_SPACE);
} }
if (file.mkdirs()) { if (file.mkdirs()) {
usedSpace += dirsToCreate * MINIMUM_FILE_SIZE; usedSpace += dirsToCreate * MINIMUM_FILE_SIZE;
} else { } else {
throw new FileOperationException(path, "Access denied"); throw new FileOperationException(path, ACCESS_DENIED);
} }
} }
@Override @Override
public void delete(String path) throws IOException { public void delete(String path) throws IOException {
if (path.isEmpty()) throw new FileOperationException(path, "Access denied"); if (path.isEmpty()) throw new FileOperationException(path, ACCESS_DENIED);
if (created()) { if (created()) {
var file = resolveFile(path); var file = resolveFile(path);
@@ -125,7 +126,7 @@ public class WritableFileMount extends FileMount implements WritableMount {
if (success) { if (success) {
usedSpace -= Math.max(MINIMUM_FILE_SIZE, fileSize); usedSpace -= Math.max(MINIMUM_FILE_SIZE, fileSize);
} else { } else {
throw new IOException("Access denied"); throw new IOException(ACCESS_DENIED);
} }
} }
@@ -133,8 +134,8 @@ public class WritableFileMount extends FileMount implements WritableMount {
public void rename(String source, String dest) throws FileOperationException { public void rename(String source, String dest) throws FileOperationException {
var sourceFile = resolvePath(source); var sourceFile = resolvePath(source);
var destFile = resolvePath(dest); var destFile = resolvePath(dest);
if (!Files.exists(sourceFile)) throw new FileOperationException(source, "No such file"); if (!Files.exists(sourceFile)) throw new FileOperationException(source, NO_SUCH_FILE);
if (Files.exists(destFile)) throw new FileOperationException(dest, "File exists"); if (Files.exists(destFile)) throw new FileOperationException(dest, FILE_EXISTS);
if (destFile.startsWith(sourceFile)) { if (destFile.startsWith(sourceFile)) {
throw new FileOperationException(source, "Cannot move a directory inside itself"); throw new FileOperationException(source, "Cannot move a directory inside itself");
@@ -164,9 +165,9 @@ public class WritableFileMount extends FileMount implements WritableMount {
var file = resolvePath(path); var file = resolvePath(path);
var attributes = tryGetAttributes(path, file); var attributes = tryGetAttributes(path, file);
if (attributes == null) { if (attributes == null) {
if (getRemainingSpace() < MINIMUM_FILE_SIZE) throw new FileOperationException(path, "Out of space"); if (getRemainingSpace() < MINIMUM_FILE_SIZE) throw new FileOperationException(path, OUT_OF_SPACE);
} else if (attributes.isDirectory()) { } else if (attributes.isDirectory()) {
throw new FileOperationException(path, "Cannot write to directory"); throw new FileOperationException(path, CANNOT_WRITE_TO_DIRECTORY);
} else { } else {
usedSpace -= Math.max(attributes.size(), MINIMUM_FILE_SIZE); usedSpace -= Math.max(attributes.size(), MINIMUM_FILE_SIZE);
} }
@@ -187,9 +188,9 @@ public class WritableFileMount extends FileMount implements WritableMount {
var file = resolvePath(path); var file = resolvePath(path);
var attributes = tryGetAttributes(path, file); var attributes = tryGetAttributes(path, file);
if (attributes == null) { if (attributes == null) {
if (getRemainingSpace() < MINIMUM_FILE_SIZE) throw new FileOperationException(path, "Out of space"); if (getRemainingSpace() < MINIMUM_FILE_SIZE) throw new FileOperationException(path, OUT_OF_SPACE);
} else if (attributes.isDirectory()) { } else if (attributes.isDirectory()) {
throw new FileOperationException(path, "Cannot write to directory"); throw new FileOperationException(path, CANNOT_WRITE_TO_DIRECTORY);
} }
// Allowing seeking when appending is not recommended, so we use a separate channel. // Allowing seeking when appending is not recommended, so we use a separate channel.
@@ -228,7 +229,7 @@ public class WritableFileMount extends FileMount implements WritableMount {
ignoredBytesLeft = 0; ignoredBytesLeft = 0;
var bytesLeft = capacity - usedSpace; var bytesLeft = capacity - usedSpace;
if (newBytes > bytesLeft) throw new IOException("Out of space"); if (newBytes > bytesLeft) throw new IOException(OUT_OF_SPACE);
usedSpace += newBytes; usedSpace += newBytes;
} }
} }

View File

@@ -10,6 +10,8 @@ import dan200.computercraft.test.core.filesystem.MountContract;
import dan200.computercraft.test.core.filesystem.WritableMountContract; import dan200.computercraft.test.core.filesystem.WritableMountContract;
import org.opentest4j.TestAbortedException; import org.opentest4j.TestAbortedException;
import static dan200.computercraft.core.filesystem.MountHelpers.EPOCH;
public class MemoryMountTest implements MountContract, WritableMountContract { public class MemoryMountTest implements MountContract, WritableMountContract {
@Override @Override
public Mount createSkeleton() { public Mount createSkeleton() {

View File

@@ -82,8 +82,8 @@ describe("The fs library", function()
end) end)
it("fails on non-existent nodes", function() it("fails on non-existent nodes", function()
expect.error(fs.list, "rom/x"):eq("/rom/x: Not a directory") expect.error(fs.list, "rom/x"):eq("/rom/x: No such file")
expect.error(fs.list, "x"):eq("/x: Not a directory") expect.error(fs.list, "x"):eq("/x: No such file")
end) end)
end) end)
@@ -160,8 +160,8 @@ describe("The fs library", function()
describe("fs.open", function() describe("fs.open", function()
describe("reading", function() describe("reading", function()
it("fails on directories", function() it("fails on directories", function()
expect { fs.open("rom", "r") }:same { nil, "/rom: No such file" } expect { fs.open("rom", "r") }:same { nil, "/rom: Not a file" }
expect { fs.open("", "r") }:same { nil, "/: No such file" } expect { fs.open("", "r") }:same { nil, "/: Not a file" }
end) end)
it("fails on non-existent nodes", function() it("fails on non-existent nodes", function()

View File

@@ -75,7 +75,7 @@ describe("The io library", function()
describe("io.lines", function() describe("io.lines", function()
it("validates arguments", function() it("validates arguments", function()
io.lines(nil) io.lines(nil)
expect.error(io.lines, ""):eq("/: No such file") expect.error(io.lines, ""):eq("/: Not a file")
expect.error(io.lines, false):eq("bad argument #1 (string expected, got boolean)") expect.error(io.lines, false):eq("bad argument #1 (string expected, got boolean)")
end) end)

View File

@@ -6,6 +6,8 @@ package dan200.computercraft.test.core.filesystem;
import dan200.computercraft.api.filesystem.FileOperationException; import dan200.computercraft.api.filesystem.FileOperationException;
import dan200.computercraft.api.filesystem.Mount; import dan200.computercraft.api.filesystem.Mount;
import dan200.computercraft.test.core.ReplaceUnderscoresDisplayNameGenerator;
import org.junit.jupiter.api.DisplayNameGeneration;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import java.io.IOException; import java.io.IOException;
@@ -19,6 +21,7 @@ import java.util.ArrayList;
import java.util.Comparator; import java.util.Comparator;
import java.util.List; import java.util.List;
import static dan200.computercraft.core.filesystem.MountHelpers.*;
import static org.junit.jupiter.api.Assertions.*; import static org.junit.jupiter.api.Assertions.*;
/** /**
@@ -26,8 +29,8 @@ import static org.junit.jupiter.api.Assertions.*;
* *
* @see WritableMountContract * @see WritableMountContract
*/ */
@DisplayNameGeneration(ReplaceUnderscoresDisplayNameGenerator.class)
public interface MountContract { public interface MountContract {
FileTime EPOCH = FileTime.from(Instant.EPOCH);
FileTime MODIFY_TIME = FileTime.from(Instant.EPOCH.plus(2, ChronoUnit.DAYS)); FileTime MODIFY_TIME = FileTime.from(Instant.EPOCH.plus(2, ChronoUnit.DAYS));
/** /**
@@ -84,6 +87,24 @@ public interface MountContract {
assertEquals(List.of("dir", "f.lua"), list); assertEquals(List.of("dir", "f.lua"), list);
} }
@Test
default void testListMissing() throws IOException {
var mount = createSkeleton();
var error = assertThrows(FileOperationException.class, () -> mount.list("no_such_file", new ArrayList<>()));
assertEquals("no_such_file", error.getFilename());
assertEquals(NO_SUCH_FILE, error.getMessage());
}
@Test
default void testListFile() throws IOException {
var mount = createSkeleton();
var error = assertThrows(FileOperationException.class, () -> mount.list("dir/file.lua", new ArrayList<>()));
assertEquals("dir/file.lua", error.getFilename());
assertEquals(NOT_A_DIRECTORY, error.getMessage());
}
@Test @Test
default void testOpenFile() throws IOException { default void testOpenFile() throws IOException {
var mount = createSkeleton(); var mount = createSkeleton();
@@ -97,7 +118,7 @@ public interface MountContract {
} }
@Test @Test
default void testOpenFileFails() throws IOException { default void openForRead_fails_on_missing_file() throws IOException {
var mount = createSkeleton(); var mount = createSkeleton();
var exn = assertThrows(FileOperationException.class, () -> mount.openForRead("doesnt/exist")); var exn = assertThrows(FileOperationException.class, () -> mount.openForRead("doesnt/exist"));
@@ -105,6 +126,16 @@ public interface MountContract {
assertEquals("No such file", exn.getMessage()); assertEquals("No such file", exn.getMessage());
} }
@Test
default void openForRead_fails_on_directory() throws IOException {
var mount = createSkeleton();
var error = assertThrows(FileOperationException.class, () -> mount.openForRead("dir").close());
assertEquals("dir", error.getFilename());
assertEquals(NOT_A_FILE, error.getMessage());
}
@Test @Test
default void testSize() throws IOException { default void testSize() throws IOException {
var mount = createSkeleton(); var mount = createSkeleton();

View File

@@ -16,6 +16,7 @@ import org.opentest4j.TestAbortedException;
import java.io.IOException; import java.io.IOException;
import java.util.stream.Stream; import java.util.stream.Stream;
import static dan200.computercraft.core.filesystem.MountHelpers.MINIMUM_FILE_SIZE;
import static org.junit.jupiter.api.Assertions.*; import static org.junit.jupiter.api.Assertions.*;
/** /**
@@ -74,7 +75,7 @@ public interface WritableMountContract {
assertTrue(mount.isDirectory("a/b/c")); assertTrue(mount.isDirectory("a/b/c"));
assertEquals(CAPACITY - 500 * 3, mount.getRemainingSpace()); assertEquals(CAPACITY - MINIMUM_FILE_SIZE * 3, mount.getRemainingSpace());
assertEquals(access.computeRemainingSpace(), access.mount().getRemainingSpace(), "Free space is inconsistent"); assertEquals(access.computeRemainingSpace(), access.mount().getRemainingSpace(), "Free space is inconsistent");
} }
@@ -108,7 +109,7 @@ public interface WritableMountContract {
Mounts.writeFile(mount, "hello.txt", ""); Mounts.writeFile(mount, "hello.txt", "");
assertEquals(0, mount.getSize("hello.txt")); assertEquals(0, mount.getSize("hello.txt"));
assertEquals(CAPACITY - 500, mount.getRemainingSpace()); assertEquals(CAPACITY - MINIMUM_FILE_SIZE, mount.getRemainingSpace());
assertEquals(access.computeRemainingSpace(), access.mount().getRemainingSpace(), "Free space is inconsistent"); assertEquals(access.computeRemainingSpace(), access.mount().getRemainingSpace(), "Free space is inconsistent");
} }