mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-05 05:57:39 +03:00
The intention is to keep to repo root clean while the list of packages is slowly growing. Additionally, a bunch of small (~30 LoC) files in the repo root is merged into a single maddy.go file, for the same reason. Most of the internal code is moved into the internal/ directory. Go toolchain will make it impossible to import these packages from external applications. Some packages are renamed and moved into the pkg/ directory in the root. According to https://github.com/golang-standards/project-layout this is the de-facto standard to place "library code that's ok to use by external applications" in. To clearly define the purpose of top-level directories, README.md files are added to each.
68 lines
1.7 KiB
Go
68 lines
1.7 KiB
Go
package buffer
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"fmt"
|
|
"io"
|
|
"math/rand"
|
|
"os"
|
|
"path/filepath"
|
|
)
|
|
|
|
// FileBuffer implements Buffer interface using file system.
|
|
type FileBuffer struct {
|
|
Path string
|
|
|
|
// LenHint is the size of the stored blob. It can
|
|
// be set to avoid the need to call os.Stat in the
|
|
// Len() method.
|
|
LenHint int
|
|
}
|
|
|
|
func (fb FileBuffer) Open() (io.ReadCloser, error) {
|
|
return os.Open(fb.Path)
|
|
}
|
|
|
|
func (fb FileBuffer) Len() int {
|
|
if fb.LenHint != 0 {
|
|
return fb.LenHint
|
|
}
|
|
|
|
info, err := os.Stat(fb.Path)
|
|
if err != nil {
|
|
// Any access to the file will probably fail too. So we can't return a
|
|
// sensible value.
|
|
return 0
|
|
}
|
|
|
|
return int(info.Size())
|
|
}
|
|
|
|
func (fb FileBuffer) Remove() error {
|
|
return os.Remove(fb.Path)
|
|
}
|
|
|
|
// BufferInFile is a convenience function which creates FileBuffer with underlying
|
|
// file created in the specified directory with the random name.
|
|
func BufferInFile(r io.Reader, dir string) (Buffer, error) {
|
|
// It is assumed that PRNG is initialized somewhere during program startup.
|
|
nameBytes := make([]byte, 32)
|
|
_, err := rand.Read(nameBytes)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("buffer: failed to generate randomness for file name: %v", err)
|
|
}
|
|
path := filepath.Join(dir, hex.EncodeToString(nameBytes))
|
|
f, err := os.Create(path)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("buffer: failed to create file: %v", err)
|
|
}
|
|
if _, err = io.Copy(f, r); err != nil {
|
|
return nil, fmt.Errorf("buffer: failed to write file: %v", err)
|
|
}
|
|
// TODO: Consider caching an *os.File object for faster first Open().
|
|
if err := f.Close(); err != nil {
|
|
return nil, fmt.Errorf("buffer: failed to close file: %v", err)
|
|
}
|
|
|
|
return FileBuffer{Path: path}, nil
|
|
}
|