Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/itchio/screw
🔩 Filesystem abstraction
https://github.com/itchio/screw
Last synced: about 2 months ago
JSON representation
🔩 Filesystem abstraction
- Host: GitHub
- URL: https://github.com/itchio/screw
- Owner: itchio
- Created: 2019-10-09T18:22:08.000Z (about 5 years ago)
- Default Branch: master
- Last Pushed: 2023-02-25T05:21:50.000Z (almost 2 years ago)
- Last Synced: 2024-06-19T17:50:10.027Z (7 months ago)
- Language: Go
- Size: 42 KB
- Stars: 4
- Watchers: 3
- Forks: 0
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# screw
[![Build Status](https://travis-ci.org/itchio/screw.svg?branch=master)](https://travis-ci.org/itchio/screw)
Requires Go 1.13+
**What do we have?** case-preserving case-insensitive filesystems on Windows & macOS.
**What do we want?** case-*sensible* semantics on Windows, macOS & Linux.
Also, retry in case renaming/removing files fails because they're locked by an Antivirus.
## Intro
Throughout this documentation and code, we make the following assumptions:
* that on Linux, we have a case-sensitive filesystem
* that on Windows and macOS, we have a case-preserving, case-insensitive filesystem.`screw` makes no effort to support [case-insensitive Linux
folders][ci-linux], or [case-sensitive Windows folders][cs-win], or [case-sensitive macOS partitions][cs-mac][ci-linux]: https://lwn.net/Articles/784041/
[cs-win]: https://devblogs.microsoft.com/commandline/per-directory-case-sensitivity-and-wsl/
[cs-mac]: https://apple.stackexchange.com/questions/8016/hfs-case-sensitive-or-case-insensitive-which-type-to-use-for-the-primary-driNow that we're clear on the scope of `screw`, let's agree on some common vocabulary.
## Case-sensitive filesystems (CS)
CS filesystems allow multiple files to exist, whose names differ only in casing.
For example, the following files may coexist:
```
- apricot
- Apricot
- APRICOT
```They are three separate files, they can have separate contents, separate permissions, etc.
Deleting one of them doesn't touch the others.Additionally, if only `apricot` exist, trying to access `APRICOT` will result in an error, since,
well, it doesn't exist.Case-sensitive semantics are the easiest to remember.
## Case-preserving case-insensitive filesystems (CPCI)
The idea behind CPCI filesystems is that if a user creates file
named `apricot`, and later tries to access it as `APRICOT`, *we know what
they mean*, so we should just give them `apricot`.The following properties follow:
- Only *a single casing* of a name may exist at a time.
- `apricot` and `APRICOT` may not both exist at the same time
- Opening `APRICOT` opens `apricot`
- Stat'ing `APRICOT` stats `apricot` *but it returns `APRICOT` as the name*
- Deleting `APRICOT` deletes `apricot`
- Creating `APRICOT` truncates `apricot`
- Renaming `banana` to `APRICOT` will *replace* `apricot`
- Renaming `apricot` to `APRICOT` is possible
- But it may take extra steps, depending on the OS API## Case-sensible semantics (CSBL)
`screw` tries to provide a *mostly case-sensitive* interface over CPCI filesystems.
If `APRICOT` exists, `screw` can pretend `apricot` doesn't - even though CPCI semantics
would say it does.However, if `APRICOT` exists, creating `apricot` cannot be magically made to work.
Those two can't coexist under CPCI. In that case, `apricot` throws an error,
`screw.ErrCaseConflict`.In short:
* `screw` only lets you `Stat`, `Lstat`, `Open` or `Remove` a file **if you give its actual casing**
* `screw` only lets you `Create` a file **if no casing variant exists**
* `screw` has `RemoveAll` do nothing if **the exact casing you passed doesn't exist**In the rest of the text, CPCI (case-preserving case-insensitive) semantics will be referred
to as "undesirable", and CSSB (case-sensible) semantics will be referred to as "desirable".If you disagree, `screw` probably isn't for you!
## Changes from `os` package
First, note that core `screw` operations are implemented in `OpenFile`, so when we mean
`Open` or `Create`, we actually mean these:| Shorthand | Actual call |
|-----------------------|--------------------------------------------------|
| Open(name) | OpenFile(name, O_RDONLY, 0) |
| Create(name) | OpenFile(name, O_RDWR\|O_CREATE\|O_TRUNC, 0o666) |The tables in the rest of this document attribute meaning to some emojis:
| Emoji | Meaning |
|-------|--------------------------|
| ✅ | Desirable return value |
| ⭕ | Undesirable return value |
| ❎ | Desirable error |
| ❌ | Undesirable error |The table assume `"apricot"` is passed to all those functions, and that the code
runs on a case-preserving, case-insensitive filesystem (Windows, macOS).| Operation | Existing file name | `os` package (CPCI) | `screw` package (CSBL)
|-------------|-----------------------|------------------------|-----------------------
| Stat, Lstat | (none) | ❎ os.ErrNotExist |
| | "apricot" | ✅ stat "apricot" |
| | "APRICOT" | ⭕ stat "APRICOT" | ❎ os.ErrNotExist
| Open | (none) | ❎ os.ErrNotExist |
| | "apricot" | ✅ open "apricot" |
| | "APRICOT" | ⭕ open "APRICOT" | ❎ os.ErrNotExist
| Create | (none) | ✅ create "apricot" |
| | "apricot" | ✅ truncate "apricot" |
| | "APRICOT" | ⭕ truncate "APRICOT" | ❎ screw.ErrCaseConflict
| Truncate | (none) | ✅ create "apricot" |
| | "apricot" | ✅ truncate "apricot" |
| | "APRICOT" | ⭕ truncate "APRICOT" | ❎ screw.ErrCaseConflictDestructive operations also behave differently:
| Operation | Existing file name | `os` package (CPCI) | `screw` package (CSBL)
|-------------|-----------------------|------------------------|-----------------------
| Remove | (none) | ❎ os.ErrNotExist |
| | "apricot" | ✅ removes "apricot" |
| | "APRICOT" | ⭕ removes "APRICOT" | ❎ os.ErrNotExist
| RemoveAll | (none) | ✅ does nothing |
| | "apricot" | ✅ removes "apricot" |
| | "APRICOT" | ⭕ removes "APRICOT" | ✅ does nothingThe mkdir family behaves differently:
| Operation | Existing dir name | `os` package (CPCI) | `screw` package (CSBL)
|-------------|-----------------------|------------------------|-----------------------
| Mkdir | (none) | ❎ mkdir "apricot" |
| | "apricot/" | ✅ os.ErrExist |
| | "APRICOT/" | ❌ os.ErrExist | ❎ screw.ErrCaseConflict
| MkdirAll | (none) | ✅ mkdir "apricot" |
| | "apricot/" | ✅ does nothing |
| | "APRICOT/" | ⭕ does nothing | ❎ screw.ErrCaseConflict## Changes from `ioutil` package
`ioutil.ReadFile`, `ioutil.WriteFile` and `ioutil.ReadDir` are included in `screw`
This table passes "apricot" to `ReadFile` and `ReadDir`
| Operation | Existing file/dir name| `ioutil` package (CPCI) | `screw` package (CSBL)
|-------------|-----------------------|-------------------------|-----------------------
| ReadFile | (none) | ❎ os.NotExist |
| | "apricot" | ✅ read "apricot" |
| | "APRICOT" | ⭕ read "APRICOT" | ❎ os.NotExist
| WriteFile | (none) | ✅ create "apricot" |
| | "apricot" | ✅ overwrites "apricot" |
| | "APRICOT" | ⭕ overwrites "apricot" | ❎ screw.ErrCaseConflict
| ReadDir | (none) | ❎ os.NotExist |
| | "apricot/" | ✅ list "apricot/" |
| | "APRICOT/" | ⭕ list "APRICOT/" | ❎ os.NotExist## What about methods of *os.File ?
One of the undesired behaviors of CPCI is that the following code:
```go
func main() {
// Note: the file `APRICOT` exists
f, _ := os.Open("apricot")
stats, _ := f.Stat()
fmt.Println(stats.Name())
}
```...will print `APRICOT`, not `apricot`.
What about `screw`? Shouldn't it have its own `File` type to prevent that?
In `screw`, that's not a problem, because the above code sample fails at the first line
with `os.ErrNotFound`, so any `os.FileInfo` obtained via screw contains the exact casing.Similarly `file.Readdir()` and `file.Readdirname()` can only be called on a file
opened with its exact casing.## What if one of the *parent* folders has the wrong case?
`screw` only checks the last path component, ie. the "base name", for performance
and compatibility reasons.Checking the *entire path* has these problems:
* It's expensive to call `FindFirstFile` for every path element, like:
* `C:\Users\bob\AppData\Roaming\something`
* `C:\Users\bob\AppData\Roaming`
* `C:\Users\bob\AppData`
* `C:\Users\bob`
* `C:\Users`
* On macOS, the "reference path" might be different even if both have the true case, e.g.:
* `/tmp/foobar` has reference path:
* `/private/tmp/foobar`## API Additions
In addition to wrapping a lot of `os` functions, `screw` also provides these functions:
| Operation | Existing file name | Parameter | Result
|---------------|-----------------------|--------------------------|--------------------------
| TrueBaseName | (none) | "apricot" | ✅ ""
| | "apricot" | "apricot" | ✅ "apricot"
| | "apricot" | "APRICOT" | ✅ "apricot"
| | "apricot/seed" | "apricot/SEED" | ✅ "apricot/seed"
| | "apricot/seed" | "APRICOT/seed" | ✅ "aAPRICOT/seed"**Important note**: contrary to the simplified table above, `GetActualCase` returns absolute paths,
not relative ones.## Rename
On Windows, `screw.Rename` differs from `os.Rename` in two ways.
After a case-only rename (e.g. `apricot` => `APRICOT`), screw makes sure the file now has the expected casing.
If it doesn't, it attempts a two-step rename, ie.:* `apricot` => `apricot_rename_${pid}`
* `apricot_rename_${pid}` => `APRICOT`
This seems unnecessary on recent versions of Windows 10 (as of October 2019), but it is the author's recollection
that this wasn't always the case.Additionally, `screw.Rename` contains retry logic on Windows (to sidestep spurious AV file locking),
and logic for older versions of Windows that don't support case-only renames.## UNC paths
UNC paths (like `\\?\C:\Windows\`, `\\SOMEHOST\\Share`) are untested and unsupported in `screw` at the time of this writing.
## Error wrapping
`screw` always wraps errors in a `*os.PathError`, to provide additional information as to which
operation caused the error, and on which file.Comparing errors with `==` is a **bad idea**.
Using `errors.Is(e, screw.ErrCaseConflict)` works.
Using `os.IsNotExist(e)` also works with `screw`-returned errors.
## Dependencies
On Windows, `screw` depends on `golang.org/x/sys/windows` to make the `FindFirstFile` syscall, instead of the legacy `syscall` package.
## Performance
Performance isn't a goal of `screw`, correctness is.