178 lines
4.2 KiB
Go
178 lines
4.2 KiB
Go
package osvfs
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
"time"
|
|
"unicode"
|
|
|
|
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/tspath"
|
|
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/vfs"
|
|
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/vfs/internal"
|
|
)
|
|
|
|
// FS creates a new FS from the OS file system.
|
|
func FS() vfs.FS {
|
|
return osVFS
|
|
}
|
|
|
|
var osVFS vfs.FS = &osFS{
|
|
common: internal.Common{
|
|
RootFor: os.DirFS,
|
|
Realpath: osFSRealpath,
|
|
},
|
|
}
|
|
|
|
type osFS struct {
|
|
common internal.Common
|
|
}
|
|
|
|
// We do this right at startup to minimize the chance that executable gets moved or deleted.
|
|
var isFileSystemCaseSensitive = func() bool {
|
|
// win32/win64 are case insensitive platforms
|
|
if runtime.GOOS == "windows" {
|
|
return false
|
|
}
|
|
|
|
if runtime.GOARCH == "wasm" {
|
|
// !!! Who knows; this depends on the host implementation.
|
|
return true
|
|
}
|
|
|
|
// As a proxy for case-insensitivity, we check if the current executable exists under a different case.
|
|
// This is not entirely correct, since different OSs can have differing case sensitivity in different paths,
|
|
// but this is largely good enough for our purposes (and what sys.ts used to do with __filename).
|
|
exe, err := os.Executable()
|
|
if err != nil {
|
|
panic(fmt.Sprintf("vfs: failed to get executable path: %v", err))
|
|
}
|
|
|
|
// If the current executable exists under a different case, we must be case-insensitive.
|
|
swapped := swapCase(exe)
|
|
if _, err := os.Stat(swapped); err != nil {
|
|
if os.IsNotExist(err) {
|
|
return true
|
|
}
|
|
panic(fmt.Sprintf("vfs: failed to stat %q: %v", swapped, err))
|
|
}
|
|
return false
|
|
}()
|
|
|
|
// Convert all lowercase chars to uppercase, and vice-versa
|
|
func swapCase(str string) string {
|
|
return strings.Map(func(r rune) rune {
|
|
upper := unicode.ToUpper(r)
|
|
if upper == r {
|
|
return unicode.ToLower(r)
|
|
} else {
|
|
return upper
|
|
}
|
|
}, str)
|
|
}
|
|
|
|
func (vfs *osFS) UseCaseSensitiveFileNames() bool {
|
|
return isFileSystemCaseSensitive
|
|
}
|
|
|
|
var readSema = make(chan struct{}, 128)
|
|
|
|
func (vfs *osFS) ReadFile(path string) (contents string, ok bool) {
|
|
// Limit ourselves to fewer open files, which greatly reduces IO contention.
|
|
readSema <- struct{}{}
|
|
defer func() { <-readSema }()
|
|
|
|
return vfs.common.ReadFile(path)
|
|
}
|
|
|
|
func (vfs *osFS) DirectoryExists(path string) bool {
|
|
return vfs.common.DirectoryExists(path)
|
|
}
|
|
|
|
func (vfs *osFS) FileExists(path string) bool {
|
|
return vfs.common.FileExists(path)
|
|
}
|
|
|
|
func (vfs *osFS) GetAccessibleEntries(path string) vfs.Entries {
|
|
return vfs.common.GetAccessibleEntries(path)
|
|
}
|
|
|
|
func (vfs *osFS) Stat(path string) vfs.FileInfo {
|
|
return vfs.common.Stat(path)
|
|
}
|
|
|
|
func (vfs *osFS) WalkDir(root string, walkFn vfs.WalkDirFunc) error {
|
|
return vfs.common.WalkDir(root, walkFn)
|
|
}
|
|
|
|
func (vfs *osFS) Realpath(path string) string {
|
|
return osFSRealpath(path)
|
|
}
|
|
|
|
func osFSRealpath(path string) string {
|
|
_ = internal.RootLength(path) // Assert path is rooted
|
|
|
|
orig := path
|
|
path = filepath.FromSlash(path)
|
|
path, err := realpath(path)
|
|
if err != nil {
|
|
return orig
|
|
}
|
|
path, err = filepath.Abs(path)
|
|
if err != nil {
|
|
return orig
|
|
}
|
|
return tspath.NormalizeSlashes(path)
|
|
}
|
|
|
|
var writeSema = make(chan struct{}, 32)
|
|
|
|
func (vfs *osFS) writeFile(path string, content string, writeByteOrderMark bool) error {
|
|
writeSema <- struct{}{}
|
|
defer func() { <-writeSema }()
|
|
|
|
file, err := os.Create(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer file.Close()
|
|
|
|
if writeByteOrderMark {
|
|
if _, err := file.WriteString("\uFEFF"); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if _, err := file.WriteString(content); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (vfs *osFS) ensureDirectoryExists(directoryPath string) error {
|
|
return os.MkdirAll(directoryPath, 0o777)
|
|
}
|
|
|
|
func (vfs *osFS) WriteFile(path string, content string, writeByteOrderMark bool) error {
|
|
_ = internal.RootLength(path) // Assert path is rooted
|
|
if err := vfs.writeFile(path, content, writeByteOrderMark); err == nil {
|
|
return nil
|
|
}
|
|
if err := vfs.ensureDirectoryExists(tspath.GetDirectoryPath(tspath.NormalizePath(path))); err != nil {
|
|
return err
|
|
}
|
|
return vfs.writeFile(path, content, writeByteOrderMark)
|
|
}
|
|
|
|
func (vfs *osFS) Remove(path string) error {
|
|
// todo: #701 add retry mechanism?
|
|
return os.RemoveAll(path)
|
|
}
|
|
|
|
func (vfs *osFS) Chtimes(path string, aTime time.Time, mTime time.Time) error {
|
|
return os.Chtimes(path, aTime, mTime)
|
|
}
|