File Operations
Perfect brings file system operations into your sever-side Swift environment to control how data is stored and retrieved in an accessible way.
First, ensure the PerfectLib is imported in your Swift file:
import PerfectLib
You are now able to use the File object to query and manipulate the file system.
Setting Up a File Object Reference
Specify the absolute or relative path to the file:
let thisFile = File("/path/to/file/helloWorld.txt")
If you are not familiar with the file path, please read Directory Operations first.
Opening a File for Read or Write Access
Important: Before writing to a file — even if it is a new file — it must be opened with the appropriate permissions.
To open a file:
try thisFile.open(<OpenMode>,permissions:<PermissionMode>)
For example, to write a file:
let thisFile = File("helloWorld.txt")
try thisFile.open(.readWrite)
try thisFile.write(string: "Hello, World!")
thisFile.close()
For full outlines of OpenMode and PermissionMode values, see their definitions later in this document.
Checking If a File Exists
Use the exists method to return a Boolean value.
thisFile.exists
Get the Modification Time for a File
Return the modification date for the file in the standard UNIX format of seconds since 1970/01/01 00:00:00 GMT, as an integer using:
thisFile.modificationTime
File Paths
Regardless of how a file reference was defined, both the absolute (real) and internal path can be returned.
Return the "internal reference" file path:
thisFile.path
Return the "real" file path. If the file is a symbolic link, the link will be resolved:
thisFile.realPath
Closing a File
Once a file has been opened for read or write, it is advisable to either close it at a specific place within the code, or by using defer:
let thisFile = File("/path/to/file/helloWorld.txt")
// Your processing here
thisFile.close()
Deleting a File
To remove a file from the file system, use the delete() method. 
thisFile.delete()
This also closes the file, so there is no need to invoke an additional close() method.
Returning the Size of a File
size returns the size of the file in bytes, as an integer.
thisFile.size
Determining if the File is a Symbolic Link
If the file is a symbolic link, the method will return Boolean true, otherwise false.
thisFile.isLink
Determining if the File Object is a Directory
If the file object refers instead to a directory, isDir will return either a Boolean true or false value.
thisFile.isDir
Returning the File's UNIX Permissions
perms returns the UNIX style permissions for the file as a PermissionMode object.
thisFile.perms
For example:
print(thisFile.perms) >> PermissionMode(rawValue: 29092)
Reading Contents of a File
readSomeBytes
Reads up to the indicated number of bytes from the file:
let thisFile = File("/path/to/file/helloWorld.txt")
let contents = try thisFile.readSomeBytes(count: <Int>)
Parameters
To read a specific byte range of a file's contents, enter the number of bytes you wish to read. For example, to read the first 10 bytes of a file:
let thisFile = File("/path/to/file/helloWorld.txt")
let contents = try thisFile.readSomeBytes(count: 10)
print(contents)
>> [35, 32, 80, 101, 114, 102, 101, 99, 116, 84]
readString
readString reads the entire file as a string:
let thisFile = File("/path/to/file/helloWorld.txt")
let contents = try thisFile.readString()
Writing, Copying, and Moving Files
Important: Before writing to a file — even if it is a new file — it must be opened with the appropriate permissions.
Writing a String to a File
Use write to create or rewrite a string to the file using UTF-8 encoding. The method returns an integer which is the number of bytes written.
Note that this method uses the @discardableResult property, so it can be used without assignment if required.
let bytesWritten = try thisFile.write(string: <String>)
Writing a Bytes Array to a File
An array of bytes can also be written directly to a file. The method returns an integer which is the number of bytes written.
Note that this method uses the @discardableResult property, so it can be used without assignment if required.
let bytesWritten = try thisFile.write(
    bytes: <[UInt8]>, 
    dataPosition: <Int>, 
    length: <Int>
    )
Parameters
- bytes: The array of UInt8 to write
- dataPosition: Optional. The offset within bytesat which to begin writing
- length: Optional. The number of bytes to write
Moving a File
Once a file is defined, the moveto method can be used to relocate the file to a new location in the file system. This can also be used to rename a file if desired. The operation returns a new file object representing the new location.
let newFile = thisFile.moveTo(path: <String>, overWrite: <Bool>)
Parameters
- path: The path to move the file to
- overWrite: Optional. Indicates that any existing file at the destination path should first be deleted. Default is false
Error Handling
The method throws PerfectError.FileError on error.
let thisFile = File("/path/to/file/helloWorld.txt")
let newFile = try thisFile.moveTo(path: "/path/to/file/goodbyeWorld.txt")
Copying a File
Similar to moveTo, copyTo copies the file to the new location, optionally overwriting any existing file. However, it does not delete the original file. A new file object is returned representing the new location.
Note that this method uses the @discardableResult property, so it can be used without assignment if required.
let newFile = thisFile.copyTo(path: <String>, overWrite: <Bool>)
Parameters
- path: The path to copy the file to
- overWrite: Optional. Indicates that any existing file at the destination path should first be deleted. Default is false
Error Handling
The method throws PerfectError.FileError on error.
let thisFile = File("/path/to/file/helloWorld.txt")
let newFile = try thisFile.copyTo(path: "/path/to/file/goodbyeWorld.txt")
File Locking Functions
The file locking functions allow sections of a file to be locked with advisory-mode locks.
All the locks for a file are removed when the file is closed or the process terminates.
Note: These are not file system locks, and do not prevent others from performing write operations on the affected files: The locks are "advisory-mode locks".
Locking a File
Attempts to place an advisory lock starting from the current position marker up to the indicated byte count. This function will block the current thread until the lock can be performed.
let result = try thisFile.lock(byteCount: <Int>)
Unlocking a File
Unlocks the number of bytes starting from the current position marker up to the indicated byte count.
let result = try thisFile.unlock(byteCount: <Int>)
Attempt to Lock a File
Attempts to place an advisory lock starting from the current position marker up to the indicated byte count. This function will throw an exception if the file is already locked, but will not block the current thread.
let result = try thisFile.tryLock(byteCount: <Int>)
Testing a Lock
Tests if the indicated bytes are locked. Returns a Boolean true or false.
let isLocked = try thisFile.testLock(byteCount: <Int>)
File OpenMode
The OpenMode of a file is defined as an enum:
- .read: Opens the file for read-only access
- .write: Opens the file for write-only access, creating the file if it did not exist
- .readWrite: Opens the file for read-write access, creating the file if it did not exist
- .append: Opens the file for read-write access, creating the file if it did not exist and moving the file marker to the end
- .truncate: Opens the file for read-write access, creating the file if it did not exist and setting the file's size to zero
For example, to write a file:
let thisFile = File("helloWorld.txt")
try thisFile.open(.readWrite)
try thisFile.write(string: "Hello, World!")
thisFile.close()
File PermissionMode
The PermissionMode for a directory or file is provided as a single option or as an array of options.
For example, to create a directory with specific permissions:
let thisDir = Dir("/path/to/dir/")
do {
    try thisDir.create(perms: [.rwxUser, .rxGroup, .rxOther])
} catch {
    print("error")
}
//or
do {
    try thisDir.create(perms: .rwxUser)
} catch {
    print("error")
}
PermissionMode Options
- .readUser: Readable by user
- .writeUser: Writable by user
- .executeUser: Executable by user
- .readGroup: Readable by group
- .writeGroup: Writable by group
- .executeGroup: Executable by group
- .readOther: Readable by others
- .writeOther: Writable by others
- .executeOther: Executable by others
- .rwxUser: Read, write, execute by user
- .rwUserGroup: Read, write by user and group
- .rxGroup: Read, execute by group
- .rxOther: Read, execute by other
 
					


