Thanks to visit codestin.com
Credit goes to pkg.go.dev

afero

package module
v1.15.0 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Sep 8, 2025 License: Apache-2.0 Imports: 24 Imported by: 8,061

README

afero logo-sm

GitHub Workflow Status GoDoc Go Report Card Go Version

Afero: The Universal Filesystem Abstraction for Go

Afero is a powerful and extensible filesystem abstraction system for Go. It provides a single, unified API for interacting with diverse filesystems—including the local disk, memory, archives, and network storage.

Afero acts as a drop-in replacement for the standard os package, enabling you to write modular code that is agnostic to the underlying storage, dramatically simplifies testing, and allows for sophisticated architectural patterns through filesystem composition.

Why Afero?

Afero elevates filesystem interaction beyond simple file reading and writing, offering solutions for testability, flexibility, and advanced architecture.

🔑 Key Features:

  • Universal API: Write your code once. Run it against the local OS, in-memory storage, ZIP/TAR archives, or remote systems (SFTP, GCS).
  • Ultimate Testability: Utilize MemMapFs, a fully concurrent-safe, read/write in-memory filesystem. Write fast, isolated, and reliable unit tests without touching the physical disk or worrying about cleanup.
  • Powerful Composition: Afero's hidden superpower. Layer filesystems on top of each other to create sophisticated behaviors:
    • Sandboxing: Use CopyOnWriteFs to create temporary scratch spaces that isolate changes from the base filesystem.
    • Caching: Use CacheOnReadFs to automatically layer a fast cache (like memory) over a slow backend (like a network drive).
    • Security Jails: Use BasePathFs to restrict application access to a specific subdirectory (chroot).
  • os Package Compatibility: Afero mirrors the functions in the standard os package, making adoption and refactoring seamless.
  • io/fs Compatibility: Fully compatible with the Go standard library's io/fs interfaces.

Installation

go get github.com/spf13/afero
import "github.com/spf13/afero"

Quick Start: The Power of Abstraction

The core of Afero is the afero.Fs interface. By designing your functions to accept this interface rather than calling os.* functions directly, your code instantly becomes more flexible and testable.

1. Refactor Your Code

Change functions that rely on the os package to accept afero.Fs.

// Before: Coupled to the OS and difficult to test
// func ProcessConfiguration(path string) error {
//     data, err := os.ReadFile(path)
//     ...
// }

import "github.com/spf13/afero"

// After: Decoupled, flexible, and testable
func ProcessConfiguration(fs afero.Fs, path string) error {
    // Use Afero utility functions which mirror os/ioutil
    data, err := afero.ReadFile(fs, path)
    // ... process the data
    return err
}
2. Usage in Production

In your production environment, inject the OsFs backend, which wraps the standard operating system calls.

func main() {
    // Use the real OS filesystem
    AppFs := afero.NewOsFs()
    ProcessConfiguration(AppFs, "/etc/myapp.conf")
}
3. Usage in Testing

In your tests, inject MemMapFs. This provides a blazing-fast, isolated, in-memory filesystem that requires no disk I/O and no cleanup.

func TestProcessConfiguration(t *testing.T) {
    // Use the in-memory filesystem
    AppFs := afero.NewMemMapFs()
    
    // Pre-populate the memory filesystem for the test
    configPath := "/test/config.json"
    afero.WriteFile(AppFs, configPath, []byte(`{"feature": true}`), 0644)

    // Run the test entirely in memory
    err := ProcessConfiguration(AppFs, configPath)
    if err != nil {
        t.Fatal(err)
    }
}

Afero's Superpower: Composition

Afero's most unique feature is its ability to combine filesystems. This allows you to build complex behaviors out of simple components, keeping your application logic clean.

Example 1: Sandboxing with Copy-on-Write

Create a temporary environment where an application can "modify" system files without affecting the actual disk.

// 1. The base layer is the real OS, made read-only for safety.
baseFs := afero.NewReadOnlyFs(afero.NewOsFs())

// 2. The overlay layer is a temporary in-memory filesystem for changes.
overlayFs := afero.NewMemMapFs()

// 3. Combine them. Reads fall through to the base; writes only hit the overlay.
sandboxFs := afero.NewCopyOnWriteFs(baseFs, overlayFs)

// The application can now "modify" /etc/hosts, but the changes are isolated in memory.
afero.WriteFile(sandboxFs, "/etc/hosts", []byte("127.0.0.1 sandboxed-app"), 0644)

// The real /etc/hosts on disk is untouched.
Example 2: Caching a Slow Filesystem

Improve performance by layering a fast cache (like memory) over a slow backend (like a network drive or cloud storage).

import "time"

// Assume 'remoteFs' is a slow backend (e.g., SFTP or GCS)
var remoteFs afero.Fs 

// 'cacheFs' is a fast in-memory backend
cacheFs := afero.NewMemMapFs()

// Create the caching layer. Cache items for 5 minutes upon first read.
cachedFs := afero.NewCacheOnReadFs(remoteFs, cacheFs, 5*time.Minute)

// The first read is slow (fetches from remote, then caches)
data1, _ := afero.ReadFile(cachedFs, "data.json")

// The second read is instant (serves from memory cache)
data2, _ := afero.ReadFile(cachedFs, "data.json")
Example 3: Security Jails (chroot)

Restrict an application component's access to a specific subdirectory.

osFs := afero.NewOsFs()

// Create a filesystem rooted at /home/user/public
// The application cannot access anything above this directory.
jailedFs := afero.NewBasePathFs(osFs, "/home/user/public")

// To the application, this is reading "/"
// In reality, it's reading "/home/user/public/"
dirInfo, err := afero.ReadDir(jailedFs, "/")

// Attempts to access parent directories fail
_, err = jailedFs.Open("../secrets.txt") // Returns an error

Real-World Use Cases

Build Cloud-Agnostic Applications

Write applications that seamlessly work with different storage backends:

type DocumentProcessor struct {
    fs afero.Fs
}

func NewDocumentProcessor(fs afero.Fs) *DocumentProcessor {
    return &DocumentProcessor{fs: fs}
}

func (p *DocumentProcessor) Process(inputPath, outputPath string) error {
    // This code works whether fs is local disk, cloud storage, or memory
    content, err := afero.ReadFile(p.fs, inputPath)
    if err != nil {
        return err
    }
    
    processed := processContent(content)
    return afero.WriteFile(p.fs, outputPath, processed, 0644)
}

// Use with local filesystem
processor := NewDocumentProcessor(afero.NewOsFs())

// Use with Google Cloud Storage
processor := NewDocumentProcessor(gcsFS)

// Use with in-memory filesystem for testing
processor := NewDocumentProcessor(afero.NewMemMapFs())
Treating Archives as Filesystems

Read files directly from .zip or .tar archives without unpacking them to disk first.

import (
    "archive/zip"
    "github.com/spf13/afero/zipfs"
)

// Assume 'zipReader' is a *zip.Reader initialized from a file or memory
var zipReader *zip.Reader 

// Create a read-only ZipFs
archiveFS := zipfs.New(zipReader)

// Read a file from within the archive using the standard Afero API
content, err := afero.ReadFile(archiveFS, "/docs/readme.md")
Serving Any Filesystem over HTTP

Use HttpFs to expose any Afero filesystem—even one created dynamically in memory—through a standard Go web server.

import (
    "net/http"
    "github.com/spf13/afero"
)

func main() {
    memFS := afero.NewMemMapFs()
    afero.WriteFile(memFS, "index.html", []byte("<h1>Hello from Memory!</h1>"), 0644)

    // Wrap the memory filesystem to make it compatible with http.FileServer.
    httpFS := afero.NewHttpFs(memFS)

    http.Handle("/", http.FileServer(httpFS.Dir("/")))
    http.ListenAndServe(":8080", nil)
}
Testing Made Simple

One of Afero's greatest strengths is making filesystem-dependent code easily testable:

func SaveUserData(fs afero.Fs, userID string, data []byte) error {
    filename := fmt.Sprintf("users/%s.json", userID)
    return afero.WriteFile(fs, filename, data, 0644)
}

func TestSaveUserData(t *testing.T) {
    // Create a clean, fast, in-memory filesystem for testing
    testFS := afero.NewMemMapFs()
    
    userData := []byte(`{"name": "John", "email": "[email protected]"}`)
    err := SaveUserData(testFS, "123", userData)
    
    if err != nil {
        t.Fatalf("SaveUserData failed: %v", err)
    }
    
    // Verify the file was saved correctly
    saved, err := afero.ReadFile(testFS, "users/123.json")
    if err != nil {
        t.Fatalf("Failed to read saved file: %v", err)
    }
    
    if string(saved) != string(userData) {
        t.Errorf("Data mismatch: got %s, want %s", saved, userData)
    }
}

Benefits of testing with Afero:

  • Fast - No disk I/O, tests run in memory
  • 🔄 Reliable - Each test starts with a clean slate
  • 🧹 No cleanup - Memory is automatically freed
  • 🔒 Safe - Can't accidentally modify real files
  • 🏃 Parallel - Tests can run concurrently without conflicts

Backend Reference

Type Backend Constructor Description Status
Core OsFs afero.NewOsFs() Interacts with the real operating system filesystem. Use in production. ✅ Official
MemMapFs afero.NewMemMapFs() A fast, atomic, concurrent-safe, in-memory filesystem. Ideal for testing. ✅ Official
Composition CopyOnWriteFs afero.NewCopyOnWriteFs(base, overlay) A read-only base with a writable overlay. Ideal for sandboxing. ✅ Official
CacheOnReadFs afero.NewCacheOnReadFs(base, cache, ttl) Lazily caches files from a slow base into a fast layer on first read. ✅ Official
BasePathFs afero.NewBasePathFs(source, path) Restricts operations to a subdirectory (chroot/jail). ✅ Official
ReadOnlyFs afero.NewReadOnlyFs(source) Provides a read-only view, preventing any modifications. ✅ Official
RegexpFs afero.NewRegexpFs(source, regexp) Filters a filesystem, only showing files that match a regex. ✅ Official
Utility HttpFs afero.NewHttpFs(source) Wraps any Afero filesystem to be served via http.FileServer. ✅ Official
Archives ZipFs zipfs.New(zipReader) Read-only access to files within a ZIP archive. ✅ Official
TarFs tarfs.New(tarReader) Read-only access to files within a TAR archive. ✅ Official
Network GcsFs gcsfs.NewGcsFs(...) Google Cloud Storage backend. ⚡ Experimental
SftpFs sftpfs.New(...) SFTP backend. ⚡ Experimental
3rd Party Cloud S3Fs fclairamb/afero-s3 Production-ready S3 backend built on official AWS SDK. 🔹 3rd Party
MinioFs cpyun/afero-minio MinIO object storage backend with S3 compatibility. 🔹 3rd Party
DriveFs fclairamb/afero-gdrive Google Drive backend with streaming support. 🔹 3rd Party
DropboxFs fclairamb/afero-dropbox Dropbox backend with streaming support. 🔹 3rd Party
3rd Party Specialized GitFs tobiash/go-gitfs Git repository filesystem (read-only, Afero compatible). 🔹 3rd Party
DockerFs unmango/aferox Docker container filesystem access. 🔹 3rd Party
GitHubFs unmango/aferox GitHub repository and releases filesystem. 🔹 3rd Party
FilterFs unmango/aferox Filesystem filtering with predicates. 🔹 3rd Party
IgnoreFs unmango/aferox .gitignore-aware filtering filesystem. 🔹 3rd Party
FUSEFs JakWai01/sile-fystem Generic FUSE implementation using any Afero backend. 🔹 3rd Party

Afero vs. io/fs (Go 1.16+)

Go 1.16 introduced the io/fs package, which provides a standard abstraction for read-only filesystems.

Afero complements io/fs by focusing on different needs:

  • Use io/fs when: You only need to read files and want to conform strictly to the standard library interfaces.
  • Use Afero when:
    • Your application needs to create, write, modify, or delete files.
    • You need to test complex read/write interactions (e.g., renaming, concurrent writes).
    • You need advanced compositional features (Copy-on-Write, Caching, etc.).

Afero is fully compatible with io/fs. You can wrap any Afero filesystem to satisfy the fs.FS interface using afero.NewIOFS:

import "io/fs"

// Create an Afero filesystem (writable)
var myAferoFs afero.Fs = afero.NewMemMapFs()

// Convert it to a standard library fs.FS (read-only view)
var myIoFs fs.FS = afero.NewIOFS(myAferoFs)

Third-Party Backends & Ecosystem

The Afero community has developed numerous backends and tools that extend the library's capabilities. Below are curated, well-maintained options organized by maturity and reliability.

These are mature, reliable backends that we can confidently recommend for production use:

Amazon S3 - fclairamb/afero-s3

Production-ready S3 backend built on the official AWS SDK for Go.

import "github.com/fclairamb/afero-s3"

s3fs := s3.NewFs(bucket, session)
MinIO - cpyun/afero-minio

MinIO object storage backend providing S3-compatible object storage with deduplication and optimization features.

import "github.com/cpyun/afero-minio"

minioFs := miniofs.NewMinioFs(ctx, "minio://endpoint/bucket")
Community & Specialized Backends
Cloud Storage
Version Control Systems
  • Git Repositories - tobiash/go-gitfs
    Read-only filesystem abstraction for Git repositories. Works with bare repositories and provides filesystem view of any git reference. Uses go-git for repository access.
Container and Remote Systems
  • Docker Containers - unmango/aferox
    Access Docker container filesystems as if they were local filesystems

  • GitHub API - unmango/aferox
    Turn GitHub repositories, releases, and assets into browsable filesystems

FUSE Integration
  • Generic FUSE - JakWai01/sile-fystem
    Mount any Afero filesystem as a FUSE filesystem, allowing any Afero backend to be used as a real mounted filesystem
Specialized Filesystems
  • FAT32 Support - aligator/GoFAT
    Pure Go FAT filesystem implementation (currently read-only)
Interface Adapters & Utilities

Cross-Interface Compatibility:

Working Directory Management:

Advanced Filtering:

  • unmango/aferox includes multiple specialized filesystems:
    • FilterFs - Predicate-based file filtering
    • IgnoreFs - .gitignore-aware filtering
    • WriterFs - Dump writes to io.Writer for debugging
Developer Tools & Utilities

nhatthm Utility Suite - Essential tools for Afero development:

Ecosystem Showcase

Windows Virtual Drives - balazsgrill/potatodrive
Mount any Afero filesystem as a Windows drive letter. Brilliant demonstration of Afero's power!

Modern Asset Embedding (Go 1.16+)

Instead of third-party tools, use Go's native //go:embed with Afero:

import (
    "embed"
    "github.com/spf13/afero"
)

//go:embed assets/*
var assetsFS embed.FS

func main() {
    // Convert embedded files to Afero filesystem
    fs := afero.FromIOFS(assetsFS)
    
    // Use like any other Afero filesystem
    content, _ := afero.ReadFile(fs, "assets/config.json")
}

Contributing

We welcome contributions! The project is mature, but we are actively looking for contributors to help implement and stabilize network/cloud backends.

  • 🔥 Microsoft Azure Blob Storage
  • 🔒 Modern Encryption Backend - Built on secure, contemporary crypto (not legacy EncFS)
  • 🐙 Canonical go-git Adapter - Unified solution for Git integration
  • 📡 SSH/SCP Backend - Secure remote file operations
  • Stabilization of existing experimental backends (GCS, SFTP)

To contribute:

  1. Fork the repository
  2. Create your feature branch (git checkout -b my-new-feature)
  3. Commit your changes (git commit -am 'Add some feature')
  4. Push to the branch (git push origin my-new-feature)
  5. Create a new Pull Request

📄 License

Afero is released under the Apache 2.0 license. See LICENSE.txt for details.

🔗 Additional Resources


Afero comes from the Latin roots Ad-Facere, meaning "to make" or "to do" - fitting for a library that empowers you to make and do amazing things with filesystems.

Documentation

Index

Constants

View Source
const BADFD = syscall.EBADFD
View Source
const FilePathSeparator = string(filepath.Separator)

Filepath separator defined by os.Separator.

Variables

View Source
var (
	ErrFileClosed        = errors.New("File is closed")
	ErrOutOfRange        = errors.New("out of range")
	ErrTooLarge          = errors.New("too large")
	ErrFileNotFound      = os.ErrNotExist
	ErrFileExists        = os.ErrExist
	ErrDestinationExists = os.ErrExist
)
View Source
var ErrNoReadlink = errors.New("readlink not supported")

ErrNoReadlink is the error that will be wrapped in an os.Path if a file system does not support the readlink operation either directly or through its delegated filesystem. As expressed by support for the LinkReader interface.

View Source
var ErrNoSymlink = errors.New("symlink not supported")

ErrNoSymlink is the error that will be wrapped in an os.LinkError if a file system does not support Symlink's either directly or through its delegated filesystem. As expressed by support for the Linker interface.

Functions

func DirExists

func DirExists(fs Fs, path string) (bool, error)

DirExists checks if a path exists and is a directory.

func Exists

func Exists(fs Fs, path string) (bool, error)

Check if a file or directory exists.

func FileContainsAnyBytes

func FileContainsAnyBytes(fs Fs, filename string, subslices [][]byte) (bool, error)

Check if a file contains any of the specified byte slices.

func FileContainsBytes

func FileContainsBytes(fs Fs, filename string, subslice []byte) (bool, error)

Check if a file contains a specified byte slice.

func FullBaseFsPath

func FullBaseFsPath(basePathFs *BasePathFs, relativePath string) string

func GetTempDir

func GetTempDir(fs Fs, subPath string) string

GetTempDir returns the default temp directory with trailing slash if subPath is not empty then it will be created recursively with mode 777 rwx rwx rwx

func Glob

func Glob(fs Fs, pattern string) (matches []string, err error)

Glob returns the names of all files matching pattern or nil if there is no matching file. The syntax of patterns is the same as in Match. The pattern may describe hierarchical names such as /usr/*/bin/ed (assuming the Separator is '/').

Glob ignores file system errors such as I/O errors reading directories. The only possible returned error is ErrBadPattern, when pattern is malformed.

This was adapted from (http://golang.org/pkg/path/filepath) and uses several built-ins from that package.

func IsDir

func IsDir(fs Fs, path string) (bool, error)

IsDir checks if a given path is a directory.

func IsEmpty

func IsEmpty(fs Fs, path string) (bool, error)

IsEmpty checks if a given file or directory is empty.

func NeuterAccents

func NeuterAccents(s string) string

Transform characters with accents into plain forms.

func ReadAll

func ReadAll(r io.Reader) ([]byte, error)

ReadAll reads from r until an error or EOF and returns the data it read. A successful call returns err == nil, not err == EOF. Because ReadAll is defined to read from src until EOF, it does not treat an EOF from Read as an error to be reported.

func ReadDir

func ReadDir(fs Fs, dirname string) ([]os.FileInfo, error)

func ReadFile

func ReadFile(fs Fs, filename string) ([]byte, error)

func SafeWriteReader

func SafeWriteReader(fs Fs, path string, r io.Reader) (err error)

func TempDir

func TempDir(fs Fs, dir, prefix string) (name string, err error)

func UnicodeSanitize

func UnicodeSanitize(s string) string

Rewrite string to remove non-standard path characters

func Walk

func Walk(fs Fs, root string, walkFn filepath.WalkFunc) error

func WriteFile

func WriteFile(fs Fs, filename string, data []byte, perm os.FileMode) error

func WriteReader

func WriteReader(fs Fs, path string, r io.Reader) (err error)

Types

type Afero

type Afero struct {
	Fs
}

func (Afero) DirExists

func (a Afero) DirExists(path string) (bool, error)

func (Afero) Exists

func (a Afero) Exists(path string) (bool, error)

func (Afero) FileContainsAnyBytes

func (a Afero) FileContainsAnyBytes(filename string, subslices [][]byte) (bool, error)

func (Afero) FileContainsBytes

func (a Afero) FileContainsBytes(filename string, subslice []byte) (bool, error)

func (Afero) GetTempDir

func (a Afero) GetTempDir(subPath string) string

func (Afero) IsDir

func (a Afero) IsDir(path string) (bool, error)

func (Afero) IsEmpty

func (a Afero) IsEmpty(path string) (bool, error)

func (Afero) ReadDir

func (a Afero) ReadDir(dirname string) ([]os.FileInfo, error)

ReadDir reads the directory named by dirname and returns a list of sorted directory entries.

func (Afero) ReadFile

func (a Afero) ReadFile(filename string) ([]byte, error)

ReadFile reads the file named by filename and returns the contents. A successful call returns err == nil, not err == EOF. Because ReadFile reads the whole file, it does not treat an EOF from Read as an error to be reported.

func (Afero) SafeWriteReader

func (a Afero) SafeWriteReader(path string, r io.Reader) (err error)

Same as WriteReader but checks to see if file/directory already exists.

func (Afero) TempDir

func (a Afero) TempDir(dir, prefix string) (name string, err error)

TempDir creates a new temporary directory in the directory dir with a name beginning with prefix and returns the path of the new directory. If dir is the empty string, TempDir uses the default directory for temporary files (see os.TempDir). Multiple programs calling TempDir simultaneously will not choose the same directory. It is the caller's responsibility to remove the directory when no longer needed.

func (Afero) TempFile

func (a Afero) TempFile(dir, pattern string) (f File, err error)

TempFile creates a new temporary file in the directory dir, opens the file for reading and writing, and returns the resulting *os.File. The filename is generated by taking pattern and adding a random string to the end. If pattern includes a "*", the random string replaces the last "*". If dir is the empty string, TempFile uses the default directory for temporary files (see os.TempDir). Multiple programs calling TempFile simultaneously will not choose the same file. The caller can use f.Name() to find the pathname of the file. It is the caller's responsibility to remove the file when no longer needed.

func (Afero) Walk

func (a Afero) Walk(root string, walkFn filepath.WalkFunc) error

func (Afero) WriteFile

func (a Afero) WriteFile(filename string, data []byte, perm os.FileMode) error

WriteFile writes data to a file named by filename. If the file does not exist, WriteFile creates it with permissions perm; otherwise WriteFile truncates it before writing.

func (Afero) WriteReader

func (a Afero) WriteReader(path string, r io.Reader) (err error)

Takes a reader and a path and writes the content

type BasePathFile added in v1.1.0

type BasePathFile struct {
	File
	// contains filtered or unexported fields
}

func (*BasePathFile) Name added in v1.1.0

func (f *BasePathFile) Name() string

func (*BasePathFile) ReadDir added in v1.9.0

func (f *BasePathFile) ReadDir(n int) ([]fs.DirEntry, error)

type BasePathFs

type BasePathFs struct {
	// contains filtered or unexported fields
}

The BasePathFs restricts all operations to a given path within an Fs. The given file name to the operations on this Fs will be prepended with the base path before calling the base Fs. Any file name (after filepath.Clean()) outside this base path will be treated as non existing file.

Note that it does not clean the error messages on return, so you may reveal the real path on errors.

func (*BasePathFs) Chmod

func (b *BasePathFs) Chmod(name string, mode os.FileMode) (err error)

func (*BasePathFs) Chown added in v1.5.0

func (b *BasePathFs) Chown(name string, uid, gid int) (err error)

func (*BasePathFs) Chtimes

func (b *BasePathFs) Chtimes(name string, atime, mtime time.Time) (err error)

func (*BasePathFs) Create

func (b *BasePathFs) Create(name string) (f File, err error)

func (*BasePathFs) LstatIfPossible added in v1.1.0

func (b *BasePathFs) LstatIfPossible(name string) (os.FileInfo, bool, error)

func (*BasePathFs) Mkdir

func (b *BasePathFs) Mkdir(name string, mode os.FileMode) (err error)

func (*BasePathFs) MkdirAll

func (b *BasePathFs) MkdirAll(name string, mode os.FileMode) (err error)

func (*BasePathFs) Name

func (b *BasePathFs) Name() string

func (*BasePathFs) Open

func (b *BasePathFs) Open(name string) (f File, err error)

func (*BasePathFs) OpenFile

func (b *BasePathFs) OpenFile(name string, flag int, mode os.FileMode) (f File, err error)

func (*BasePathFs) ReadlinkIfPossible added in v1.3.0

func (b *BasePathFs) ReadlinkIfPossible(name string) (string, error)

func (*BasePathFs) RealPath

func (b *BasePathFs) RealPath(name string) (path string, err error)

on a file outside the base path it returns the given file name and an error, else the given file with the base path prepended

func (*BasePathFs) Remove

func (b *BasePathFs) Remove(name string) (err error)

func (*BasePathFs) RemoveAll

func (b *BasePathFs) RemoveAll(name string) (err error)

func (*BasePathFs) Rename

func (b *BasePathFs) Rename(oldname, newname string) (err error)

func (*BasePathFs) Stat

func (b *BasePathFs) Stat(name string) (fi os.FileInfo, err error)

func (*BasePathFs) SymlinkIfPossible added in v1.3.0

func (b *BasePathFs) SymlinkIfPossible(oldname, newname string) error

type CacheOnReadFs

type CacheOnReadFs struct {
	// contains filtered or unexported fields
}

If the cache duration is 0, cache time will be unlimited, i.e. once a file is in the layer, the base will never be read again for this file.

For cache times greater than 0, the modification time of a file is checked. Note that a lot of file system implementations only allow a resolution of a second for timestamps... or as the godoc for os.Chtimes() states: "The underlying filesystem may truncate or round the values to a less precise time unit."

This caching union will forward all write calls also to the base file system first. To prevent writing to the base Fs, wrap it in a read-only filter - Note: this will also make the overlay read-only, for writing files in the overlay, use the overlay Fs directly, not via the union Fs.

func (*CacheOnReadFs) Chmod

func (u *CacheOnReadFs) Chmod(name string, mode os.FileMode) error

func (*CacheOnReadFs) Chown added in v1.5.0

func (u *CacheOnReadFs) Chown(name string, uid, gid int) error

func (*CacheOnReadFs) Chtimes

func (u *CacheOnReadFs) Chtimes(name string, atime, mtime time.Time) error

func (*CacheOnReadFs) Create

func (u *CacheOnReadFs) Create(name string) (File, error)

func (*CacheOnReadFs) Mkdir

func (u *CacheOnReadFs) Mkdir(name string, perm os.FileMode) error

func (*CacheOnReadFs) MkdirAll

func (u *CacheOnReadFs) MkdirAll(name string, perm os.FileMode) error

func (*CacheOnReadFs) Name

func (u *CacheOnReadFs) Name() string

func (*CacheOnReadFs) Open

func (u *CacheOnReadFs) Open(name string) (File, error)

func (*CacheOnReadFs) OpenFile

func (u *CacheOnReadFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (*CacheOnReadFs) Remove

func (u *CacheOnReadFs) Remove(name string) error

func (*CacheOnReadFs) RemoveAll

func (u *CacheOnReadFs) RemoveAll(name string) error

func (*CacheOnReadFs) Rename

func (u *CacheOnReadFs) Rename(oldname, newname string) error

func (*CacheOnReadFs) Stat

func (u *CacheOnReadFs) Stat(name string) (os.FileInfo, error)

type CopyOnWriteFs

type CopyOnWriteFs struct {
	// contains filtered or unexported fields
}

The CopyOnWriteFs is a union filesystem: a read only base file system with a possibly writeable layer on top. Changes to the file system will only be made in the overlay: Changing an existing file in the base layer which is not present in the overlay will copy the file to the overlay ("changing" includes also calls to e.g. Chtimes(), Chmod() and Chown()).

Reading directories is currently only supported via Open(), not OpenFile().

func (*CopyOnWriteFs) Chmod

func (u *CopyOnWriteFs) Chmod(name string, mode os.FileMode) error

func (*CopyOnWriteFs) Chown added in v1.5.0

func (u *CopyOnWriteFs) Chown(name string, uid, gid int) error

func (*CopyOnWriteFs) Chtimes

func (u *CopyOnWriteFs) Chtimes(name string, atime, mtime time.Time) error

func (*CopyOnWriteFs) Create

func (u *CopyOnWriteFs) Create(name string) (File, error)

func (*CopyOnWriteFs) LstatIfPossible added in v1.1.0

func (u *CopyOnWriteFs) LstatIfPossible(name string) (os.FileInfo, bool, error)

func (*CopyOnWriteFs) Mkdir

func (u *CopyOnWriteFs) Mkdir(name string, perm os.FileMode) error

func (*CopyOnWriteFs) MkdirAll

func (u *CopyOnWriteFs) MkdirAll(name string, perm os.FileMode) error

func (*CopyOnWriteFs) Name

func (u *CopyOnWriteFs) Name() string

func (*CopyOnWriteFs) Open

func (u *CopyOnWriteFs) Open(name string) (File, error)

This function handles the 9 different possibilities caused by the union which are the intersection of the following...

layer: doesn't exist, exists as a file, and exists as a directory
base:  doesn't exist, exists as a file, and exists as a directory

func (*CopyOnWriteFs) OpenFile

func (u *CopyOnWriteFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (*CopyOnWriteFs) ReadlinkIfPossible added in v1.3.0

func (u *CopyOnWriteFs) ReadlinkIfPossible(name string) (string, error)

func (*CopyOnWriteFs) Remove

func (u *CopyOnWriteFs) Remove(name string) error

Removing files present only in the base layer is not permitted. If a file is present in the base layer and the overlay, only the overlay will be removed.

func (*CopyOnWriteFs) RemoveAll

func (u *CopyOnWriteFs) RemoveAll(name string) error

func (*CopyOnWriteFs) Rename

func (u *CopyOnWriteFs) Rename(oldname, newname string) error

Renaming files present only in the base layer is not permitted

func (*CopyOnWriteFs) Stat

func (u *CopyOnWriteFs) Stat(name string) (os.FileInfo, error)

func (*CopyOnWriteFs) SymlinkIfPossible added in v1.3.0

func (u *CopyOnWriteFs) SymlinkIfPossible(oldname, newname string) error

type DirsMerger added in v1.1.0

type DirsMerger func(lofi, bofi []os.FileInfo) ([]os.FileInfo, error)

DirsMerger is how UnionFile weaves two directories together. It takes the FileInfo slices from the layer and the base and returns a single view.

type File

type File interface {
	io.Closer
	io.Reader
	io.ReaderAt
	io.Seeker
	io.Writer
	io.WriterAt

	Name() string
	Readdir(count int) ([]os.FileInfo, error)
	Readdirnames(n int) ([]string, error)
	Stat() (os.FileInfo, error)
	Sync() error
	Truncate(size int64) error
	WriteString(s string) (ret int, err error)
}

File represents a file in the filesystem.

func TempFile

func TempFile(fs Fs, dir, pattern string) (f File, err error)

type FromIOFS added in v1.6.0

type FromIOFS struct {
	fs.FS
}

FromIOFS adopts io/fs.FS to use it as afero.Fs Note that io/fs.FS is read-only so all mutating methods will return fs.PathError with fs.ErrPermission To store modifications you may use afero.CopyOnWriteFs

func (FromIOFS) Chmod added in v1.6.0

func (f FromIOFS) Chmod(name string, mode os.FileMode) error

func (FromIOFS) Chown added in v1.6.0

func (f FromIOFS) Chown(name string, uid, gid int) error

func (FromIOFS) Chtimes added in v1.6.0

func (f FromIOFS) Chtimes(name string, atime time.Time, mtime time.Time) error

func (FromIOFS) Create added in v1.6.0

func (f FromIOFS) Create(name string) (File, error)

func (FromIOFS) Mkdir added in v1.6.0

func (f FromIOFS) Mkdir(
	name string,
	perm os.FileMode,
) error

func (FromIOFS) MkdirAll added in v1.6.0

func (f FromIOFS) MkdirAll(path string, perm os.FileMode) error

func (FromIOFS) Name added in v1.6.0

func (f FromIOFS) Name() string

func (FromIOFS) Open added in v1.6.0

func (f FromIOFS) Open(name string) (File, error)

func (FromIOFS) OpenFile added in v1.6.0

func (f FromIOFS) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (FromIOFS) Remove added in v1.6.0

func (f FromIOFS) Remove(name string) error

func (FromIOFS) RemoveAll added in v1.6.0

func (f FromIOFS) RemoveAll(path string) error

func (FromIOFS) Rename added in v1.6.0

func (f FromIOFS) Rename(oldname, newname string) error

func (FromIOFS) Stat added in v1.6.0

func (f FromIOFS) Stat(name string) (os.FileInfo, error)

type Fs

type Fs interface {
	// Create creates a file in the filesystem, returning the file and an
	// error, if any happens.
	Create(name string) (File, error)

	// Mkdir creates a directory in the filesystem, return an error if any
	// happens.
	Mkdir(name string, perm os.FileMode) error

	// MkdirAll creates a directory path and all parents that does not exist
	// yet.
	MkdirAll(path string, perm os.FileMode) error

	// Open opens a file, returning it or an error, if any happens.
	Open(name string) (File, error)

	// OpenFile opens a file using the given flags and the given mode.
	OpenFile(name string, flag int, perm os.FileMode) (File, error)

	// Remove removes a file identified by name, returning an error, if any
	// happens.
	Remove(name string) error

	// RemoveAll removes a directory path and any children it contains. It
	// does not fail if the path does not exist (return nil).
	RemoveAll(path string) error

	// Rename renames a file.
	Rename(oldname, newname string) error

	// Stat returns a FileInfo describing the named file, or an error, if any
	// happens.
	Stat(name string) (os.FileInfo, error)

	// The name of this FileSystem
	Name() string

	// Chmod changes the mode of the named file to mode.
	Chmod(name string, mode os.FileMode) error

	// Chown changes the uid and gid of the named file.
	Chown(name string, uid, gid int) error

	// Chtimes changes the access and modification times of the named file
	Chtimes(name string, atime time.Time, mtime time.Time) error
}

Fs is the filesystem interface.

Any simulated or real filesystem should implement this interface.

func NewBasePathFs

func NewBasePathFs(source Fs, path string) Fs

func NewCacheOnReadFs

func NewCacheOnReadFs(base Fs, layer Fs, cacheTime time.Duration) Fs

func NewCopyOnWriteFs

func NewCopyOnWriteFs(base Fs, layer Fs) Fs

func NewMemMapFs

func NewMemMapFs() Fs

func NewOsFs

func NewOsFs() Fs

func NewReadOnlyFs

func NewReadOnlyFs(source Fs) Fs

func NewRegexpFs

func NewRegexpFs(source Fs, re *regexp.Regexp) Fs

type HttpFs

type HttpFs struct {
	// contains filtered or unexported fields
}

func NewHttpFs

func NewHttpFs(source Fs) *HttpFs

func (HttpFs) Chmod

func (h HttpFs) Chmod(name string, mode os.FileMode) error

func (HttpFs) Chown added in v1.5.0

func (h HttpFs) Chown(name string, uid, gid int) error

func (HttpFs) Chtimes

func (h HttpFs) Chtimes(name string, atime time.Time, mtime time.Time) error

func (HttpFs) Create

func (h HttpFs) Create(name string) (File, error)

func (HttpFs) Dir

func (h HttpFs) Dir(s string) *httpDir

func (HttpFs) Mkdir

func (h HttpFs) Mkdir(name string, perm os.FileMode) error

func (HttpFs) MkdirAll

func (h HttpFs) MkdirAll(path string, perm os.FileMode) error

func (HttpFs) Name

func (h HttpFs) Name() string

func (HttpFs) Open

func (h HttpFs) Open(name string) (http.File, error)

func (HttpFs) OpenFile

func (h HttpFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (HttpFs) Remove

func (h HttpFs) Remove(name string) error

func (HttpFs) RemoveAll

func (h HttpFs) RemoveAll(path string) error

func (HttpFs) Rename

func (h HttpFs) Rename(oldname, newname string) error

func (HttpFs) Stat

func (h HttpFs) Stat(name string) (os.FileInfo, error)

type IOFS added in v1.6.0

type IOFS struct {
	Fs
}

IOFS adopts afero.Fs to stdlib io/fs.FS

func NewIOFS added in v1.6.0

func NewIOFS(fs Fs) IOFS

func (IOFS) Glob added in v1.6.0

func (iofs IOFS) Glob(pattern string) ([]string, error)

func (IOFS) Open added in v1.6.0

func (iofs IOFS) Open(name string) (fs.File, error)

func (IOFS) ReadDir added in v1.6.0

func (iofs IOFS) ReadDir(name string) ([]fs.DirEntry, error)

func (IOFS) ReadFile added in v1.6.0

func (iofs IOFS) ReadFile(name string) ([]byte, error)

func (IOFS) Sub added in v1.6.0

func (iofs IOFS) Sub(dir string) (fs.FS, error)

type LinkReader added in v1.3.0

type LinkReader interface {
	ReadlinkIfPossible(name string) (string, error)
}

LinkReader is an optional interface in Afero. It is only implemented by the filesystems saying so.

type Linker added in v1.3.0

type Linker interface {
	SymlinkIfPossible(oldname, newname string) error
}

Linker is an optional interface in Afero. It is only implemented by the filesystems saying so. It will call Symlink if the filesystem itself is, or it delegates to, the os filesystem, or the filesystem otherwise supports Symlink's.

type Lstater added in v1.1.0

type Lstater interface {
	LstatIfPossible(name string) (os.FileInfo, bool, error)
}

Lstater is an optional interface in Afero. It is only implemented by the filesystems saying so. It will call Lstat if the filesystem itself is, or it delegates to, the os filesystem. Else it will call Stat. In addition to the FileInfo, it will return a boolean telling whether Lstat was called or not.

type MemMapFs

type MemMapFs struct {
	// contains filtered or unexported fields
}

func (*MemMapFs) Chmod

func (m *MemMapFs) Chmod(name string, mode os.FileMode) error

func (*MemMapFs) Chown added in v1.5.0

func (m *MemMapFs) Chown(name string, uid, gid int) error

func (*MemMapFs) Chtimes

func (m *MemMapFs) Chtimes(name string, atime time.Time, mtime time.Time) error

func (*MemMapFs) Create

func (m *MemMapFs) Create(name string) (File, error)

func (*MemMapFs) List

func (m *MemMapFs) List()

func (*MemMapFs) LstatIfPossible added in v1.4.0

func (m *MemMapFs) LstatIfPossible(name string) (os.FileInfo, bool, error)

func (*MemMapFs) Mkdir

func (m *MemMapFs) Mkdir(name string, perm os.FileMode) error

func (*MemMapFs) MkdirAll

func (m *MemMapFs) MkdirAll(path string, perm os.FileMode) error

func (*MemMapFs) Name

func (*MemMapFs) Name() string

func (*MemMapFs) Open

func (m *MemMapFs) Open(name string) (File, error)

func (*MemMapFs) OpenFile

func (m *MemMapFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (*MemMapFs) Remove

func (m *MemMapFs) Remove(name string) error

func (*MemMapFs) RemoveAll

func (m *MemMapFs) RemoveAll(path string) error

func (*MemMapFs) Rename

func (m *MemMapFs) Rename(oldname, newname string) error

func (*MemMapFs) Stat

func (m *MemMapFs) Stat(name string) (os.FileInfo, error)

type OsFs

type OsFs struct{}

OsFs is a Fs implementation that uses functions provided by the os package.

For details in any method, check the documentation of the os package (http://golang.org/pkg/os/).

func (OsFs) Chmod

func (OsFs) Chmod(name string, mode os.FileMode) error

func (OsFs) Chown added in v1.5.0

func (OsFs) Chown(name string, uid, gid int) error

func (OsFs) Chtimes

func (OsFs) Chtimes(name string, atime time.Time, mtime time.Time) error

func (OsFs) Create

func (OsFs) Create(name string) (File, error)

func (OsFs) LstatIfPossible added in v1.1.0

func (OsFs) LstatIfPossible(name string) (os.FileInfo, bool, error)

func (OsFs) Mkdir

func (OsFs) Mkdir(name string, perm os.FileMode) error

func (OsFs) MkdirAll

func (OsFs) MkdirAll(path string, perm os.FileMode) error

func (OsFs) Name

func (OsFs) Name() string

func (OsFs) Open

func (OsFs) Open(name string) (File, error)

func (OsFs) OpenFile

func (OsFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (OsFs) ReadlinkIfPossible added in v1.3.0

func (OsFs) ReadlinkIfPossible(name string) (string, error)

func (OsFs) Remove

func (OsFs) Remove(name string) error

func (OsFs) RemoveAll

func (OsFs) RemoveAll(path string) error

func (OsFs) Rename

func (OsFs) Rename(oldname, newname string) error

func (OsFs) Stat

func (OsFs) Stat(name string) (os.FileInfo, error)

func (OsFs) SymlinkIfPossible added in v1.3.0

func (OsFs) SymlinkIfPossible(oldname, newname string) error

type ReadOnlyFs

type ReadOnlyFs struct {
	// contains filtered or unexported fields
}

func (*ReadOnlyFs) Chmod

func (r *ReadOnlyFs) Chmod(n string, m os.FileMode) error

func (*ReadOnlyFs) Chown added in v1.5.0

func (r *ReadOnlyFs) Chown(n string, uid, gid int) error

func (*ReadOnlyFs) Chtimes

func (r *ReadOnlyFs) Chtimes(n string, a, m time.Time) error

func (*ReadOnlyFs) Create

func (r *ReadOnlyFs) Create(n string) (File, error)

func (*ReadOnlyFs) LstatIfPossible added in v1.1.0

func (r *ReadOnlyFs) LstatIfPossible(name string) (os.FileInfo, bool, error)

func (*ReadOnlyFs) Mkdir

func (r *ReadOnlyFs) Mkdir(n string, p os.FileMode) error

func (*ReadOnlyFs) MkdirAll

func (r *ReadOnlyFs) MkdirAll(n string, p os.FileMode) error

func (*ReadOnlyFs) Name

func (r *ReadOnlyFs) Name() string

func (*ReadOnlyFs) Open

func (r *ReadOnlyFs) Open(n string) (File, error)

func (*ReadOnlyFs) OpenFile

func (r *ReadOnlyFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (*ReadOnlyFs) ReadDir

func (r *ReadOnlyFs) ReadDir(name string) ([]os.FileInfo, error)

func (*ReadOnlyFs) ReadlinkIfPossible added in v1.3.0

func (r *ReadOnlyFs) ReadlinkIfPossible(name string) (string, error)

func (*ReadOnlyFs) Remove

func (r *ReadOnlyFs) Remove(n string) error

func (*ReadOnlyFs) RemoveAll

func (r *ReadOnlyFs) RemoveAll(p string) error

func (*ReadOnlyFs) Rename

func (r *ReadOnlyFs) Rename(o, n string) error

func (*ReadOnlyFs) Stat

func (r *ReadOnlyFs) Stat(name string) (os.FileInfo, error)

func (*ReadOnlyFs) SymlinkIfPossible added in v1.3.0

func (r *ReadOnlyFs) SymlinkIfPossible(oldname, newname string) error

type RegexpFile

type RegexpFile struct {
	// contains filtered or unexported fields
}

func (*RegexpFile) Close

func (f *RegexpFile) Close() error

func (*RegexpFile) Name

func (f *RegexpFile) Name() string

func (*RegexpFile) Read

func (f *RegexpFile) Read(s []byte) (int, error)

func (*RegexpFile) ReadAt

func (f *RegexpFile) ReadAt(s []byte, o int64) (int, error)

func (*RegexpFile) Readdir

func (f *RegexpFile) Readdir(c int) (fi []os.FileInfo, err error)

func (*RegexpFile) Readdirnames

func (f *RegexpFile) Readdirnames(c int) (n []string, err error)

func (*RegexpFile) Seek

func (f *RegexpFile) Seek(o int64, w int) (int64, error)

func (*RegexpFile) Stat

func (f *RegexpFile) Stat() (os.FileInfo, error)

func (*RegexpFile) Sync

func (f *RegexpFile) Sync() error

func (*RegexpFile) Truncate

func (f *RegexpFile) Truncate(s int64) error

func (*RegexpFile) Write

func (f *RegexpFile) Write(s []byte) (int, error)

func (*RegexpFile) WriteAt

func (f *RegexpFile) WriteAt(s []byte, o int64) (int, error)

func (*RegexpFile) WriteString

func (f *RegexpFile) WriteString(s string) (int, error)

type RegexpFs

type RegexpFs struct {
	// contains filtered or unexported fields
}

The RegexpFs filters files (not directories) by regular expression. Only files matching the given regexp will be allowed, all others get a ENOENT error ( "No such file or directory").

func (*RegexpFs) Chmod

func (r *RegexpFs) Chmod(name string, mode os.FileMode) error

func (*RegexpFs) Chown added in v1.5.0

func (r *RegexpFs) Chown(name string, uid, gid int) error

func (*RegexpFs) Chtimes

func (r *RegexpFs) Chtimes(name string, a, m time.Time) error

func (*RegexpFs) Create

func (r *RegexpFs) Create(name string) (File, error)

func (*RegexpFs) Mkdir

func (r *RegexpFs) Mkdir(n string, p os.FileMode) error

func (*RegexpFs) MkdirAll

func (r *RegexpFs) MkdirAll(n string, p os.FileMode) error

func (*RegexpFs) Name

func (r *RegexpFs) Name() string

func (*RegexpFs) Open

func (r *RegexpFs) Open(name string) (File, error)

func (*RegexpFs) OpenFile

func (r *RegexpFs) OpenFile(name string, flag int, perm os.FileMode) (File, error)

func (*RegexpFs) Remove

func (r *RegexpFs) Remove(name string) error

func (*RegexpFs) RemoveAll

func (r *RegexpFs) RemoveAll(p string) error

func (*RegexpFs) Rename

func (r *RegexpFs) Rename(oldname, newname string) error

func (*RegexpFs) Stat

func (r *RegexpFs) Stat(name string) (os.FileInfo, error)

type Symlinker added in v1.3.0

type Symlinker interface {
	Lstater
	Linker
	LinkReader
}

Symlinker is an optional interface in Afero. It is only implemented by the filesystems saying so. It indicates support for 3 symlink related interfaces that implement the behaviors of the os methods:

  • Lstat
  • Symlink, and
  • Readlink

type UnionFile

type UnionFile struct {
	Base   File
	Layer  File
	Merger DirsMerger
	// contains filtered or unexported fields
}

The UnionFile implements the afero.File interface and will be returned when reading a directory present at least in the overlay or opening a file for writing.

The calls to Readdir() and Readdirnames() merge the file os.FileInfo / names from the base and the overlay - for files present in both layers, only those from the overlay will be used.

When opening files for writing (Create() / OpenFile() with the right flags) the operations will be done in both layers, starting with the overlay. A successful read in the overlay will move the cursor position in the base layer by the number of bytes read.

func (*UnionFile) Close

func (f *UnionFile) Close() error

func (*UnionFile) Name

func (f *UnionFile) Name() string

func (*UnionFile) Read

func (f *UnionFile) Read(s []byte) (int, error)

func (*UnionFile) ReadAt

func (f *UnionFile) ReadAt(s []byte, o int64) (int, error)

func (*UnionFile) Readdir

func (f *UnionFile) Readdir(c int) (ofi []os.FileInfo, err error)

Readdir will weave the two directories together and return a single view of the overlayed directories. At the end of the directory view, the error is io.EOF if c > 0.

func (*UnionFile) Readdirnames

func (f *UnionFile) Readdirnames(c int) ([]string, error)

func (*UnionFile) Seek

func (f *UnionFile) Seek(o int64, w int) (pos int64, err error)

func (*UnionFile) Stat

func (f *UnionFile) Stat() (os.FileInfo, error)

func (*UnionFile) Sync

func (f *UnionFile) Sync() (err error)

func (*UnionFile) Truncate

func (f *UnionFile) Truncate(s int64) (err error)

func (*UnionFile) Write

func (f *UnionFile) Write(s []byte) (n int, err error)

func (*UnionFile) WriteAt

func (f *UnionFile) WriteAt(s []byte, o int64) (n int, err error)

func (*UnionFile) WriteString

func (f *UnionFile) WriteString(s string) (n int, err error)

Directories

Path Synopsis
gcsfs module
internal
sftpfs module
package tarfs implements a read-only in-memory representation of a tar archive
package tarfs implements a read-only in-memory representation of a tar archive

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL