2019-06-26 20:39:01 +02:00
|
|
|
|
package fichier
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"net/http"
|
|
|
|
|
"strconv"
|
|
|
|
|
"strings"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2019-07-28 19:47:38 +02:00
|
|
|
|
"github.com/rclone/rclone/fs"
|
2020-01-14 18:33:35 +01:00
|
|
|
|
"github.com/rclone/rclone/fs/config"
|
2019-07-28 19:47:38 +02:00
|
|
|
|
"github.com/rclone/rclone/fs/config/configmap"
|
|
|
|
|
"github.com/rclone/rclone/fs/config/configstruct"
|
|
|
|
|
"github.com/rclone/rclone/fs/fshttp"
|
|
|
|
|
"github.com/rclone/rclone/fs/hash"
|
|
|
|
|
"github.com/rclone/rclone/lib/dircache"
|
2020-01-14 18:33:35 +01:00
|
|
|
|
"github.com/rclone/rclone/lib/encoder"
|
2019-07-28 19:47:38 +02:00
|
|
|
|
"github.com/rclone/rclone/lib/pacer"
|
|
|
|
|
"github.com/rclone/rclone/lib/rest"
|
2019-06-26 20:39:01 +02:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
2020-04-24 03:15:52 +02:00
|
|
|
|
rootID = "0"
|
|
|
|
|
apiBaseURL = "https://api.1fichier.com/v1"
|
|
|
|
|
minSleep = 400 * time.Millisecond // api is extremely rate limited now
|
|
|
|
|
maxSleep = 5 * time.Second
|
|
|
|
|
decayConstant = 2 // bigger for slower decay, exponential
|
|
|
|
|
attackConstant = 0 // start with max sleep
|
2019-06-26 20:39:01 +02:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
|
fs.Register(&fs.RegInfo{
|
|
|
|
|
Name: "fichier",
|
|
|
|
|
Description: "1Fichier",
|
|
|
|
|
Config: func(name string, config configmap.Mapper) {
|
|
|
|
|
},
|
|
|
|
|
NewFs: NewFs,
|
2020-01-14 18:33:35 +01:00
|
|
|
|
Options: []fs.Option{{
|
|
|
|
|
Help: "Your API Key, get it from https://1fichier.com/console/params.pl",
|
|
|
|
|
Name: "api_key",
|
|
|
|
|
}, {
|
|
|
|
|
Help: "If you want to download a shared folder, add this parameter",
|
|
|
|
|
Name: "shared_folder",
|
|
|
|
|
Required: false,
|
|
|
|
|
Advanced: true,
|
|
|
|
|
}, {
|
|
|
|
|
Name: config.ConfigEncoding,
|
|
|
|
|
Help: config.ConfigEncodingHelp,
|
|
|
|
|
Advanced: true,
|
2020-01-14 22:51:49 +01:00
|
|
|
|
// Characters that need escaping
|
|
|
|
|
//
|
|
|
|
|
// '\\': '\', // FULLWIDTH REVERSE SOLIDUS
|
|
|
|
|
// '<': '<', // FULLWIDTH LESS-THAN SIGN
|
|
|
|
|
// '>': '>', // FULLWIDTH GREATER-THAN SIGN
|
|
|
|
|
// '"': '"', // FULLWIDTH QUOTATION MARK - not on the list but seems to be reserved
|
|
|
|
|
// '\'': ''', // FULLWIDTH APOSTROPHE
|
|
|
|
|
// '$': '$', // FULLWIDTH DOLLAR SIGN
|
|
|
|
|
// '`': '`', // FULLWIDTH GRAVE ACCENT
|
|
|
|
|
//
|
|
|
|
|
// Leading space and trailing space
|
|
|
|
|
Default: (encoder.Display |
|
|
|
|
|
encoder.EncodeBackSlash |
|
|
|
|
|
encoder.EncodeSingleQuote |
|
|
|
|
|
encoder.EncodeBackQuote |
|
|
|
|
|
encoder.EncodeDoubleQuote |
|
|
|
|
|
encoder.EncodeLtGt |
|
|
|
|
|
encoder.EncodeDollar |
|
|
|
|
|
encoder.EncodeLeftSpace |
|
|
|
|
|
encoder.EncodeRightSpace |
|
|
|
|
|
encoder.EncodeInvalidUtf8),
|
2020-01-14 18:33:35 +01:00
|
|
|
|
}},
|
2019-06-26 20:39:01 +02:00
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Options defines the configuration for this backend
|
|
|
|
|
type Options struct {
|
2020-01-14 18:33:35 +01:00
|
|
|
|
APIKey string `config:"api_key"`
|
|
|
|
|
SharedFolder string `config:"shared_folder"`
|
|
|
|
|
Enc encoder.MultiEncoder `config:"encoding"`
|
2019-06-26 20:39:01 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Fs is the interface a cloud storage system must provide
|
|
|
|
|
type Fs struct {
|
|
|
|
|
root string
|
|
|
|
|
name string
|
|
|
|
|
features *fs.Features
|
2020-01-14 18:33:35 +01:00
|
|
|
|
opt Options
|
2019-06-26 20:39:01 +02:00
|
|
|
|
dirCache *dircache.DirCache
|
|
|
|
|
baseClient *http.Client
|
|
|
|
|
pacer *fs.Pacer
|
|
|
|
|
rest *rest.Client
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
|
|
|
|
func (f *Fs) FindLeaf(ctx context.Context, pathID, leaf string) (pathIDOut string, found bool, err error) {
|
2019-07-20 01:50:57 +02:00
|
|
|
|
folderID, err := strconv.Atoi(pathID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", false, err
|
|
|
|
|
}
|
2019-09-04 21:00:37 +02:00
|
|
|
|
folders, err := f.listFolders(ctx, folderID)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return "", false, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, folder := range folders.SubFolders {
|
|
|
|
|
if folder.Name == leaf {
|
|
|
|
|
pathIDOut := strconv.Itoa(folder.ID)
|
|
|
|
|
return pathIDOut, true, nil
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return "", false, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// CreateDir makes a directory with pathID as parent and name leaf
|
|
|
|
|
func (f *Fs) CreateDir(ctx context.Context, pathID, leaf string) (newID string, err error) {
|
2019-07-20 01:50:57 +02:00
|
|
|
|
folderID, err := strconv.Atoi(pathID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
2019-09-04 21:00:37 +02:00
|
|
|
|
resp, err := f.makeFolder(ctx, leaf, folderID)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return strconv.Itoa(resp.FolderID), err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Name of the remote (as passed into NewFs)
|
|
|
|
|
func (f *Fs) Name() string {
|
|
|
|
|
return f.name
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Root of the remote (as passed into NewFs)
|
|
|
|
|
func (f *Fs) Root() string {
|
|
|
|
|
return f.root
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// String returns a description of the FS
|
|
|
|
|
func (f *Fs) String() string {
|
|
|
|
|
return fmt.Sprintf("1Fichier root '%s'", f.root)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Precision of the ModTimes in this Fs
|
|
|
|
|
func (f *Fs) Precision() time.Duration {
|
|
|
|
|
return fs.ModTimeNotSupported
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Hashes returns the supported hash types of the filesystem
|
|
|
|
|
func (f *Fs) Hashes() hash.Set {
|
|
|
|
|
return hash.Set(hash.Whirlpool)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Features returns the optional features of this Fs
|
|
|
|
|
func (f *Fs) Features() *fs.Features {
|
|
|
|
|
return f.features
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewFs makes a new Fs object from the path
|
|
|
|
|
//
|
|
|
|
|
// The path is of the form remote:path
|
|
|
|
|
//
|
|
|
|
|
// Remotes are looked up in the config file. If the remote isn't
|
|
|
|
|
// found then NotFoundInConfigFile will be returned.
|
|
|
|
|
//
|
|
|
|
|
// On Windows avoid single character remote names as they can be mixed
|
|
|
|
|
// up with drive letters.
|
2019-10-01 16:30:51 +02:00
|
|
|
|
func NewFs(name string, root string, config configmap.Mapper) (fs.Fs, error) {
|
2019-06-26 20:39:01 +02:00
|
|
|
|
opt := new(Options)
|
|
|
|
|
err := configstruct.Set(config, opt)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If using a Shared Folder override root
|
|
|
|
|
if opt.SharedFolder != "" {
|
|
|
|
|
root = ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//workaround for wonky parser
|
|
|
|
|
root = strings.Trim(root, "/")
|
|
|
|
|
|
|
|
|
|
f := &Fs{
|
|
|
|
|
name: name,
|
|
|
|
|
root: root,
|
2020-01-14 18:33:35 +01:00
|
|
|
|
opt: *opt,
|
2020-04-24 03:15:52 +02:00
|
|
|
|
pacer: fs.NewPacer(pacer.NewDefault(pacer.MinSleep(minSleep), pacer.MaxSleep(maxSleep), pacer.DecayConstant(decayConstant), pacer.AttackConstant(attackConstant))),
|
2019-06-26 20:39:01 +02:00
|
|
|
|
baseClient: &http.Client{},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f.features = (&fs.Features{
|
|
|
|
|
DuplicateFiles: true,
|
|
|
|
|
CanHaveEmptyDirectories: true,
|
|
|
|
|
}).Fill(f)
|
|
|
|
|
|
|
|
|
|
client := fshttp.NewClient(fs.Config)
|
|
|
|
|
|
|
|
|
|
f.rest = rest.NewClient(client).SetRoot(apiBaseURL)
|
|
|
|
|
|
2020-01-14 18:33:35 +01:00
|
|
|
|
f.rest.SetHeader("Authorization", "Bearer "+f.opt.APIKey)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
|
|
|
|
|
f.dirCache = dircache.New(root, rootID, f)
|
|
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
|
|
// Find the current root
|
|
|
|
|
err = f.dirCache.FindRoot(ctx, false)
|
|
|
|
|
if err != nil {
|
|
|
|
|
// Assume it is a file
|
|
|
|
|
newRoot, remote := dircache.SplitPath(root)
|
|
|
|
|
tempF := *f
|
|
|
|
|
tempF.dirCache = dircache.New(newRoot, rootID, &tempF)
|
|
|
|
|
tempF.root = newRoot
|
|
|
|
|
// Make new Fs which is the parent
|
|
|
|
|
err = tempF.dirCache.FindRoot(ctx, false)
|
|
|
|
|
if err != nil {
|
|
|
|
|
// No root so return old f
|
|
|
|
|
return f, nil
|
|
|
|
|
}
|
|
|
|
|
_, err := tempF.NewObject(ctx, remote)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if err == fs.ErrorObjectNotFound {
|
|
|
|
|
// File doesn't exist so return old f
|
|
|
|
|
return f, nil
|
|
|
|
|
}
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
f.features.Fill(&tempF)
|
|
|
|
|
// XXX: update the old f here instead of returning tempF, since
|
|
|
|
|
// `features` were already filled with functions having *f as a receiver.
|
2019-07-28 19:47:38 +02:00
|
|
|
|
// See https://github.com/rclone/rclone/issues/2182
|
2019-06-26 20:39:01 +02:00
|
|
|
|
f.dirCache = tempF.dirCache
|
|
|
|
|
f.root = tempF.root
|
|
|
|
|
// return an error with an fs which points to the parent
|
|
|
|
|
return f, fs.ErrorIsFile
|
|
|
|
|
}
|
|
|
|
|
return f, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// List the objects and directories in dir into entries. The
|
|
|
|
|
// entries can be returned in any order but should be for a
|
|
|
|
|
// complete directory.
|
|
|
|
|
//
|
|
|
|
|
// dir should be "" to list the root, and should not have
|
|
|
|
|
// trailing slashes.
|
|
|
|
|
//
|
|
|
|
|
// This should return ErrDirNotFound if the directory isn't
|
|
|
|
|
// found.
|
|
|
|
|
func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err error) {
|
2020-01-14 18:33:35 +01:00
|
|
|
|
if f.opt.SharedFolder != "" {
|
|
|
|
|
return f.listSharedFiles(ctx, f.opt.SharedFolder)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dirContent, err := f.listDir(ctx, dir)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return dirContent, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewObject finds the Object at remote. If it can't be found
|
|
|
|
|
// it returns the error ErrorObjectNotFound.
|
|
|
|
|
func (f *Fs) NewObject(ctx context.Context, remote string) (fs.Object, error) {
|
|
|
|
|
leaf, directoryID, err := f.dirCache.FindRootAndPath(ctx, remote, false)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if err == fs.ErrorDirNotFound {
|
|
|
|
|
return nil, fs.ErrorObjectNotFound
|
|
|
|
|
}
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-20 01:50:57 +02:00
|
|
|
|
folderID, err := strconv.Atoi(directoryID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2019-09-04 21:00:37 +02:00
|
|
|
|
files, err := f.listFiles(ctx, folderID)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, file := range files.Items {
|
|
|
|
|
if file.Filename == leaf {
|
|
|
|
|
path, ok := f.dirCache.GetInv(directoryID)
|
|
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
|
return nil, errors.New("Cannot find dir in dircache")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return f.newObjectFromFile(ctx, path, file), nil
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil, fs.ErrorObjectNotFound
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Put in to the remote path with the modTime given of the given size
|
|
|
|
|
//
|
2020-05-25 08:05:53 +02:00
|
|
|
|
// When called from outside an Fs by rclone, src.Size() will always be >= 0.
|
2019-06-26 20:39:01 +02:00
|
|
|
|
// But for unknown-sized objects (indicated by src.Size() == -1), Put should either
|
|
|
|
|
// return an error or upload it properly (rather than e.g. calling panic).
|
|
|
|
|
//
|
|
|
|
|
// May create the object even if it returns an error - if so
|
|
|
|
|
// will return the object and the error, otherwise will return
|
|
|
|
|
// nil and the error
|
|
|
|
|
func (f *Fs) Put(ctx context.Context, in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) (fs.Object, error) {
|
|
|
|
|
exisitingObj, err := f.NewObject(ctx, src.Remote())
|
|
|
|
|
switch err {
|
|
|
|
|
case nil:
|
|
|
|
|
return exisitingObj, exisitingObj.Update(ctx, in, src, options...)
|
|
|
|
|
case fs.ErrorObjectNotFound:
|
|
|
|
|
// Not found so create it
|
|
|
|
|
return f.PutUnchecked(ctx, in, src, options...)
|
|
|
|
|
default:
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// putUnchecked uploads the object with the given name and size
|
|
|
|
|
//
|
|
|
|
|
// This will create a duplicate if we upload a new file without
|
|
|
|
|
// checking to see if there is one already - use Put() for that.
|
|
|
|
|
func (f *Fs) putUnchecked(ctx context.Context, in io.Reader, remote string, size int64, options ...fs.OpenOption) (fs.Object, error) {
|
2019-09-05 14:59:06 +02:00
|
|
|
|
if size > int64(100e9) {
|
2019-06-26 20:39:01 +02:00
|
|
|
|
return nil, errors.New("File too big, cant upload")
|
|
|
|
|
} else if size == 0 {
|
|
|
|
|
return nil, fs.ErrorCantUploadEmptyFiles
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-04 21:00:37 +02:00
|
|
|
|
nodeResponse, err := f.getUploadNode(ctx)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leaf, directoryID, err := f.dirCache.FindRootAndPath(ctx, remote, true)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-21 23:11:02 +01:00
|
|
|
|
_, err = f.uploadFile(ctx, in, size, leaf, directoryID, nodeResponse.ID, nodeResponse.URL, options...)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-04 21:00:37 +02:00
|
|
|
|
fileUploadResponse, err := f.endUpload(ctx, nodeResponse.ID, nodeResponse.URL)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(fileUploadResponse.Links) != 1 {
|
|
|
|
|
return nil, errors.New("unexpected amount of files")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
link := fileUploadResponse.Links[0]
|
|
|
|
|
fileSize, err := strconv.ParseInt(link.Size, 10, 64)
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &Object{
|
|
|
|
|
fs: f,
|
|
|
|
|
remote: remote,
|
|
|
|
|
file: File{
|
|
|
|
|
ACL: 0,
|
|
|
|
|
CDN: 0,
|
|
|
|
|
Checksum: link.Whirlpool,
|
|
|
|
|
ContentType: "",
|
|
|
|
|
Date: time.Now().Format("2006-01-02 15:04:05"),
|
|
|
|
|
Filename: link.Filename,
|
|
|
|
|
Pass: 0,
|
2019-10-01 13:34:38 +02:00
|
|
|
|
Size: fileSize,
|
2019-06-26 20:39:01 +02:00
|
|
|
|
URL: link.Download,
|
|
|
|
|
},
|
|
|
|
|
}, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PutUnchecked uploads the object
|
|
|
|
|
//
|
|
|
|
|
// This will create a duplicate if we upload a new file without
|
|
|
|
|
// checking to see if there is one already - use Put() for that.
|
|
|
|
|
func (f *Fs) PutUnchecked(ctx context.Context, in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) (fs.Object, error) {
|
|
|
|
|
return f.putUnchecked(ctx, in, src.Remote(), src.Size(), options...)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Mkdir makes the directory (container, bucket)
|
|
|
|
|
//
|
|
|
|
|
// Shouldn't return an error if it already exists
|
|
|
|
|
func (f *Fs) Mkdir(ctx context.Context, dir string) error {
|
|
|
|
|
err := f.dirCache.FindRoot(ctx, true)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
if dir != "" {
|
|
|
|
|
_, err = f.dirCache.FindDir(ctx, dir, true)
|
|
|
|
|
}
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Rmdir removes the directory (container, bucket) if empty
|
|
|
|
|
//
|
|
|
|
|
// Return an error if it doesn't exist or isn't empty
|
|
|
|
|
func (f *Fs) Rmdir(ctx context.Context, dir string) error {
|
|
|
|
|
err := f.dirCache.FindRoot(ctx, false)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-20 01:50:57 +02:00
|
|
|
|
directoryID, err := f.dirCache.FindDir(ctx, dir, false)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
folderID, err := strconv.Atoi(directoryID)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-04 21:00:37 +02:00
|
|
|
|
_, err = f.removeFolder(ctx, dir, folderID)
|
2019-06-26 20:39:01 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f.dirCache.FlushDir(dir)
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check the interfaces are satisfied
|
|
|
|
|
var (
|
|
|
|
|
_ fs.Fs = (*Fs)(nil)
|
|
|
|
|
_ fs.PutUncheckeder = (*Fs)(nil)
|
|
|
|
|
_ dircache.DirCacher = (*Fs)(nil)
|
|
|
|
|
)
|