mirror of
https://github.com/SquidDev-CC/CC-Tweaked
synced 2025-08-29 08:42:17 +00:00
Add isReadOnly to fs.attributes (#639)
This commit is contained in:
parent
19054684c6
commit
0b6dbe7778
@ -396,7 +396,8 @@ public class FSAPI implements ILuaAPI {
|
||||
/**
|
||||
* Get attributes about a specific file or folder.
|
||||
*
|
||||
* The returned attributes table contains information about the size of the file, whether it is a directory, and when it was created and last modified.
|
||||
* The returned attributes table contains information about the size of the file, whether it is a directory,
|
||||
* when it was created and last modified, and whether it is read only.
|
||||
*
|
||||
* The creation and modification times are given as the number of milliseconds since the UNIX epoch. This may be given to {@link OSAPI#date} in order to
|
||||
* convert it to more usable form.
|
||||
@ -404,7 +405,7 @@ public class FSAPI implements ILuaAPI {
|
||||
* @param path The path to get attributes for.
|
||||
* @return The resulting attributes.
|
||||
* @throws LuaException If the path does not exist.
|
||||
* @cc.treturn { size = number, isDir = boolean, created = number, modified = number } The resulting attributes.
|
||||
* @cc.treturn { size = number, isDir = boolean, isReadOnly = boolean, created = number, modified = number } The resulting attributes.
|
||||
* @see #getSize If you only care about the file's size.
|
||||
* @see #isDir If you only care whether a path is a directory or not.
|
||||
*/
|
||||
@ -413,11 +414,12 @@ public class FSAPI implements ILuaAPI {
|
||||
try {
|
||||
BasicFileAttributes attributes = this.fileSystem.getAttributes(path);
|
||||
Map<String, Object> result = new HashMap<>();
|
||||
result.put("modification", getFileTime(attributes.lastModifiedTime()));
|
||||
result.put("modified", getFileTime(attributes.lastModifiedTime()));
|
||||
result.put("created", getFileTime(attributes.creationTime()));
|
||||
result.put("size", attributes.isDirectory() ? 0 : attributes.size());
|
||||
result.put("isDir", attributes.isDirectory());
|
||||
result.put( "modification", getFileTime( attributes.lastModifiedTime() ) );
|
||||
result.put( "modified", getFileTime( attributes.lastModifiedTime() ) );
|
||||
result.put( "created", getFileTime( attributes.creationTime() ) );
|
||||
result.put( "size", attributes.isDirectory() ? 0 : attributes.size() );
|
||||
result.put( "isDir", attributes.isDirectory() );
|
||||
result.put( "isReadOnly", fileSystem.isReadOnly( path ) );
|
||||
return result;
|
||||
} catch (FileSystemException e) {
|
||||
throw new LuaException(e.getMessage());
|
||||
|
@ -11,4 +11,212 @@ describe("The fs library", function()
|
||||
expect.error(fs.complete, "", "", true, 1):eq("bad argument #4 (expected boolean, got number)")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.isDriveRoot", function()
|
||||
it("validates arguments", function()
|
||||
fs.isDriveRoot("")
|
||||
|
||||
expect.error(fs.isDriveRoot, nil):eq("bad argument #1 (expected string, got nil)")
|
||||
end)
|
||||
|
||||
it("correctly identifies drive roots", function()
|
||||
expect(fs.isDriveRoot("/rom")):eq(true)
|
||||
expect(fs.isDriveRoot("/")):eq(true)
|
||||
expect(fs.isDriveRoot("/rom/startup.lua")):eq(false)
|
||||
expect(fs.isDriveRoot("/rom/programs/delete.lua")):eq(false)
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.list", function()
|
||||
it("fails on files", function()
|
||||
expect.error(fs.list, "rom/startup.lua"):eq("/rom/startup.lua: Not a directory")
|
||||
expect.error(fs.list, "startup.lua"):eq("/startup.lua: Not a directory")
|
||||
end)
|
||||
|
||||
it("fails on non-existent nodes", function()
|
||||
expect.error(fs.list, "rom/x"):eq("/rom/x: Not a directory")
|
||||
expect.error(fs.list, "x"):eq("/x: Not a directory")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.combine", function()
|
||||
it("removes . and ..", function()
|
||||
expect(fs.combine("./a/b")):eq("a/b")
|
||||
expect(fs.combine("a/b", "../c")):eq("a/c")
|
||||
expect(fs.combine("a", "../c")):eq("c")
|
||||
expect(fs.combine("a", "../../c")):eq("../c")
|
||||
end)
|
||||
|
||||
it("combines empty paths", function()
|
||||
expect(fs.combine("a")):eq("a")
|
||||
expect(fs.combine("a", "")):eq("a")
|
||||
expect(fs.combine("", "a")):eq("a")
|
||||
expect(fs.combine("a", "", "b", "c")):eq("a/b/c")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.getSize", function()
|
||||
it("fails on non-existent nodes", function()
|
||||
expect.error(fs.getSize, "rom/x"):eq("/rom/x: No such file")
|
||||
expect.error(fs.getSize, "x"):eq("/x: No such file")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.open", function()
|
||||
describe("reading", function()
|
||||
it("fails on directories", function()
|
||||
expect { fs.open("rom", "r") }:same { nil, "/rom: No such file" }
|
||||
expect { fs.open("", "r") }:same { nil, "/: No such file" }
|
||||
end)
|
||||
|
||||
it("fails on non-existent nodes", function()
|
||||
expect { fs.open("rom/x", "r") }:same { nil, "/rom/x: No such file" }
|
||||
expect { fs.open("x", "r") }:same { nil, "/x: No such file" }
|
||||
end)
|
||||
|
||||
it("errors when closing twice", function()
|
||||
local handle = fs.open("rom/startup.lua", "r")
|
||||
handle.close()
|
||||
expect.error(handle.close):eq("attempt to use a closed file")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("reading in binary mode", function()
|
||||
it("errors when closing twice", function()
|
||||
local handle = fs.open("rom/startup.lua", "rb")
|
||||
handle.close()
|
||||
expect.error(handle.close):eq("attempt to use a closed file")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("writing", function()
|
||||
it("fails on directories", function()
|
||||
expect { fs.open("", "w") }:same { nil, "/: Cannot write to directory" }
|
||||
end)
|
||||
|
||||
it("fails on read-only mounts", function()
|
||||
expect { fs.open("rom/x", "w") }:same { nil, "/rom/x: Access denied" }
|
||||
end)
|
||||
|
||||
it("errors when closing twice", function()
|
||||
local handle = fs.open("test-files/out.txt", "w")
|
||||
handle.close()
|
||||
expect.error(handle.close):eq("attempt to use a closed file")
|
||||
end)
|
||||
|
||||
it("fails gracefully when opening 'CON' on Windows", function()
|
||||
local ok, err = fs.open("test-files/con", "w")
|
||||
if ok then fs.delete("test-files/con") return end
|
||||
|
||||
-- On my Windows/Java version the message appears to be "Incorrect function.". It may not be
|
||||
-- consistent though, and honestly doesn't matter too much.
|
||||
expect(err):str_match("^/test%-files/con: .*")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("writing in binary mode", function()
|
||||
it("errors when closing twice", function()
|
||||
local handle = fs.open("test-files/out.txt", "wb")
|
||||
handle.close()
|
||||
expect.error(handle.close):eq("attempt to use a closed file")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("appending", function()
|
||||
it("fails on directories", function()
|
||||
expect { fs.open("", "a") }:same { nil, "/: Cannot write to directory" }
|
||||
end)
|
||||
|
||||
it("fails on read-only mounts", function()
|
||||
expect { fs.open("rom/x", "a") }:same { nil, "/rom/x: Access denied" }
|
||||
end)
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.makeDir", function()
|
||||
it("fails on files", function()
|
||||
expect.error(fs.makeDir, "startup.lua"):eq("/startup.lua: File exists")
|
||||
end)
|
||||
|
||||
it("fails on read-only mounts", function()
|
||||
expect.error(fs.makeDir, "rom/x"):eq("/rom/x: Access denied")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.delete", function()
|
||||
it("fails on read-only mounts", function()
|
||||
expect.error(fs.delete, "rom/x"):eq("/rom/x: Access denied")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.copy", function()
|
||||
it("fails on read-only mounts", function()
|
||||
expect.error(fs.copy, "rom", "rom/startup"):eq("/rom/startup: Access denied")
|
||||
end)
|
||||
|
||||
it("fails to copy a folder inside itself", function()
|
||||
fs.makeDir("some-folder")
|
||||
expect.error(fs.copy, "some-folder", "some-folder/x"):eq("/some-folder: Can't copy a directory inside itself")
|
||||
expect.error(fs.copy, "some-folder", "Some-Folder/x"):eq("/some-folder: Can't copy a directory inside itself")
|
||||
end)
|
||||
|
||||
it("copies folders", function()
|
||||
fs.delete("some-folder")
|
||||
fs.delete("another-folder")
|
||||
|
||||
fs.makeDir("some-folder")
|
||||
fs.copy("some-folder", "another-folder")
|
||||
expect(fs.isDir("another-folder")):eq(true)
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.move", function()
|
||||
it("fails on read-only mounts", function()
|
||||
expect.error(fs.move, "rom", "rom/move"):eq("Access denied")
|
||||
expect.error(fs.move, "test-files", "rom/move"):eq("Access denied")
|
||||
expect.error(fs.move, "rom", "test-files"):eq("Access denied")
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.getCapacity", function()
|
||||
it("returns nil on read-only mounts", function()
|
||||
expect(fs.getCapacity("rom")):eq(nil)
|
||||
end)
|
||||
|
||||
it("returns the capacity on the root mount", function()
|
||||
expect(fs.getCapacity("")):eq(10000000)
|
||||
end)
|
||||
end)
|
||||
|
||||
describe("fs.attributes", function()
|
||||
it("errors on non-existent files", function()
|
||||
expect.error(fs.attributes, "xuxu_nao_existe"):eq("/xuxu_nao_existe: No such file")
|
||||
end)
|
||||
|
||||
it("returns information about read-only mounts", function()
|
||||
expect(fs.attributes("rom")):matches { isDir = true, size = 0, isReadOnly = true }
|
||||
end)
|
||||
|
||||
it("returns information about files", function()
|
||||
local now = os.epoch("utc")
|
||||
|
||||
fs.delete("/tmp/basic-file")
|
||||
local h = fs.open("/tmp/basic-file", "w")
|
||||
h.write("A reasonably sized string")
|
||||
h.close()
|
||||
|
||||
local attributes = fs.attributes("tmp/basic-file")
|
||||
expect(attributes):matches { isDir = false, size = 25, isReadOnly = false }
|
||||
|
||||
if attributes.created - now >= 1000 then
|
||||
fail(("Expected created time (%d) to be within 1000ms of now (%d"):format(attributes.created, now))
|
||||
end
|
||||
|
||||
if attributes.modified - now >= 1000 then
|
||||
fail(("Expected modified time (%d) to be within 1000ms of now (%d"):format(attributes.modified, now))
|
||||
end
|
||||
|
||||
expect(attributes.modification):eq(attributes.modified)
|
||||
end)
|
||||
end)
|
||||
end)
|
||||
|
Loading…
x
Reference in New Issue
Block a user