2015-12-07 05:01:03 +01:00
|
|
|
// Package yandex provides an interface to the Yandex Disk storage.
|
|
|
|
//
|
|
|
|
// dibu28 <dibu28@gmail.com> github.com/dibu28
|
|
|
|
package yandex
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ncw/rclone/fs"
|
|
|
|
"github.com/ncw/rclone/oauthutil"
|
|
|
|
yandex "github.com/ncw/rclone/yandex/api"
|
|
|
|
"golang.org/x/oauth2"
|
|
|
|
)
|
|
|
|
|
|
|
|
//oAuth
|
|
|
|
const (
|
|
|
|
rcloneClientID = "ac39b43b9eba4cae8ffb788c06d816a8"
|
|
|
|
rcloneClientSecret = "k8jKzZnMmM+Wx5jAksPAwYKPgImOiN+FhNKD09KBg9A="
|
|
|
|
)
|
|
|
|
|
|
|
|
// Globals
|
|
|
|
var (
|
|
|
|
// Description of how to auth for this app
|
|
|
|
oauthConfig = &oauth2.Config{
|
|
|
|
Endpoint: oauth2.Endpoint{
|
|
|
|
AuthURL: "https://oauth.yandex.com/authorize", //same as https://oauth.yandex.ru/authorize
|
|
|
|
TokenURL: "https://oauth.yandex.com/token", //same as https://oauth.yandex.ru/token
|
|
|
|
},
|
|
|
|
ClientID: rcloneClientID,
|
|
|
|
ClientSecret: fs.Reveal(rcloneClientSecret),
|
|
|
|
RedirectURL: oauthutil.RedirectURL,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// Register with Fs
|
|
|
|
func init() {
|
|
|
|
fs.Register(&fs.Info{
|
|
|
|
Name: "yandex",
|
|
|
|
NewFs: NewFs,
|
|
|
|
Config: func(name string) {
|
|
|
|
err := oauthutil.Config(name, oauthConfig)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Failed to configure token: %v", err)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Options: []fs.Option{{
|
|
|
|
Name: oauthutil.ConfigClientID,
|
|
|
|
Help: "Yandex Client Id - leave blank normally.",
|
|
|
|
}, {
|
|
|
|
Name: oauthutil.ConfigClientSecret,
|
|
|
|
Help: "Yandex Client Secret - leave blank normally.",
|
|
|
|
}},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fs represents a remote yandex
|
|
|
|
type Fs struct {
|
|
|
|
name string
|
|
|
|
yd *yandex.Client // client for rest api
|
|
|
|
root string //root path
|
|
|
|
diskRoot string //root path with "disk:/" container name
|
|
|
|
mkdircache map[string]int
|
|
|
|
}
|
|
|
|
|
|
|
|
// Object describes a swift object
|
|
|
|
type Object struct {
|
|
|
|
fs *Fs // what this object is part of
|
|
|
|
remote string // The remote path
|
|
|
|
md5sum string // The MD5Sum of the object
|
|
|
|
bytes uint64 // Bytes in the object
|
|
|
|
modTime time.Time // Modified time of the object
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------------------------------
|
|
|
|
|
|
|
|
// 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 converts this Fs to a string
|
|
|
|
func (f *Fs) String() string {
|
|
|
|
return fmt.Sprintf("Yandex %s", f.root)
|
|
|
|
}
|
|
|
|
|
|
|
|
// read access token from ConfigFile string
|
|
|
|
func getAccessToken(name string) (*oauth2.Token, error) {
|
|
|
|
// Read the token from the config file
|
|
|
|
tokenConfig := fs.ConfigFile.MustValue(name, "token")
|
|
|
|
//Get access token from config string
|
|
|
|
decoder := json.NewDecoder(strings.NewReader(tokenConfig))
|
|
|
|
var result *oauth2.Token
|
|
|
|
err := decoder.Decode(&result)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFs constructs an Fs from the path, container:path
|
|
|
|
func NewFs(name, root string) (fs.Fs, error) {
|
|
|
|
//read access token from config
|
|
|
|
token, err := getAccessToken(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//create new client
|
2015-12-30 14:30:01 +01:00
|
|
|
yandexDisk := yandex.NewClient(token.AccessToken, fs.Config.Client())
|
2015-12-07 05:01:03 +01:00
|
|
|
|
|
|
|
f := &Fs{
|
|
|
|
yd: yandexDisk,
|
|
|
|
}
|
|
|
|
|
|
|
|
f.setRoot(root)
|
|
|
|
|
|
|
|
//limited fs
|
|
|
|
// Check to see if the object exists and is a file
|
|
|
|
//request object meta info
|
|
|
|
var opt2 yandex.ResourceInfoRequestOptions
|
|
|
|
if ResourceInfoResponse, err := yandexDisk.NewResourceInfoRequest(root, opt2).Exec(); err != nil {
|
|
|
|
//return err
|
|
|
|
} else {
|
|
|
|
if ResourceInfoResponse.ResourceType == "file" {
|
|
|
|
//limited fs
|
|
|
|
remote := path.Base(root)
|
|
|
|
f.setRoot(path.Dir(root))
|
|
|
|
|
|
|
|
obj := f.newFsObjectWithInfo(remote, ResourceInfoResponse)
|
|
|
|
// return a Fs Limited to this object
|
|
|
|
return fs.NewLimited(f, obj), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sets root in f
|
|
|
|
func (f *Fs) setRoot(root string) {
|
|
|
|
//Set root path
|
|
|
|
f.root = strings.Trim(root, "/")
|
|
|
|
//Set disk root path.
|
|
|
|
//Adding "disk:" to root path as all paths on disk start with it
|
|
|
|
var diskRoot = ""
|
|
|
|
if f.root == "" {
|
|
|
|
diskRoot = "disk:/"
|
|
|
|
} else {
|
|
|
|
diskRoot = "disk:/" + f.root + "/"
|
|
|
|
}
|
|
|
|
f.diskRoot = diskRoot
|
|
|
|
}
|
|
|
|
|
|
|
|
// list the objects into the function supplied
|
|
|
|
//
|
|
|
|
// If directories is set it only sends directories
|
|
|
|
func (f *Fs) list(directories bool, fn func(string, yandex.ResourceInfoResponse)) {
|
|
|
|
//request files list. list is divided into pages. We send request for each page
|
|
|
|
//items per page is limited by limit
|
|
|
|
//TODO may be add config parameter for the items per page limit
|
|
|
|
var limit uint32 = 1000 // max number of object per request
|
|
|
|
var itemsCount uint32 //number of items per page in response
|
|
|
|
var offset uint32 //for the next page of request
|
|
|
|
// yandex disk api request options
|
|
|
|
var opt yandex.FlatFileListRequestOptions
|
|
|
|
opt.Limit = &limit
|
|
|
|
opt.Offset = &offset
|
|
|
|
//query each page of list until itemCount is less then limit
|
|
|
|
for {
|
|
|
|
//send request
|
|
|
|
info, err := f.yd.NewFlatFileListRequest(opt).Exec()
|
|
|
|
if err != nil {
|
|
|
|
fs.Stats.Error()
|
|
|
|
fs.ErrorLog(f, "Couldn't list: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
itemsCount = uint32(len(info.Items))
|
|
|
|
|
|
|
|
//list files
|
|
|
|
for _, item := range info.Items {
|
|
|
|
// filter file list and get only files we need
|
|
|
|
if strings.HasPrefix(item.Path, f.diskRoot) {
|
|
|
|
//trim root folder from filename
|
|
|
|
var name = strings.TrimPrefix(item.Path, f.diskRoot)
|
|
|
|
fn(name, item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//offset for the next page of items
|
|
|
|
offset += itemsCount
|
|
|
|
//check if we reached end of list
|
|
|
|
if itemsCount < limit {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// List walks the path returning a channel of FsObjects
|
|
|
|
func (f *Fs) List() fs.ObjectsChan {
|
|
|
|
out := make(fs.ObjectsChan, fs.Config.Checkers)
|
|
|
|
// List the objects
|
|
|
|
go func() {
|
|
|
|
defer close(out)
|
|
|
|
f.list(false, func(remote string, object yandex.ResourceInfoResponse) {
|
|
|
|
if fs := f.newFsObjectWithInfo(remote, &object); fs != nil {
|
|
|
|
out <- fs
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}()
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFsObject returns an Object from a path
|
|
|
|
//
|
|
|
|
// May return nil if an error occurred
|
|
|
|
func (f *Fs) NewFsObject(remote string) fs.Object {
|
|
|
|
return f.newFsObjectWithInfo(remote, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return an FsObject from a path
|
|
|
|
//
|
|
|
|
// May return nil if an error occurred
|
|
|
|
func (f *Fs) newFsObjectWithInfo(remote string, info *yandex.ResourceInfoResponse) fs.Object {
|
|
|
|
o := &Object{
|
|
|
|
fs: f,
|
|
|
|
remote: remote,
|
|
|
|
}
|
|
|
|
if info != nil {
|
|
|
|
o.setMetaData(info)
|
|
|
|
} else {
|
|
|
|
err := o.readMetaData()
|
|
|
|
if err != nil {
|
|
|
|
fs.ErrorLog(f, "Couldn't get object '%s' metadata: %s", o.remotePath(), err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return o
|
|
|
|
}
|
|
|
|
|
|
|
|
// setMetaData sets the fs data from a storage.Object
|
|
|
|
func (o *Object) setMetaData(info *yandex.ResourceInfoResponse) {
|
|
|
|
o.bytes = info.Size
|
|
|
|
o.md5sum = info.Md5
|
|
|
|
|
|
|
|
if info.CustomProperties["rclone_modified"] == nil {
|
|
|
|
//read modTime from Modified property of object
|
|
|
|
t, err := time.Parse(time.RFC3339Nano, info.Modified)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
o.modTime = t
|
|
|
|
} else {
|
|
|
|
// interface{} to string type assertion
|
|
|
|
if modtimestr, ok := info.CustomProperties["rclone_modified"].(string); ok {
|
|
|
|
//read modTime from rclone_modified custom_property of object
|
|
|
|
t, err := time.Parse(time.RFC3339Nano, modtimestr)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
o.modTime = t
|
|
|
|
} else {
|
|
|
|
return //if it is not a string
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// readMetaData gets the info if it hasn't already been fetched
|
|
|
|
func (o *Object) readMetaData() (err error) {
|
|
|
|
//request meta info
|
|
|
|
var opt2 yandex.ResourceInfoRequestOptions
|
|
|
|
ResourceInfoResponse, err := o.fs.yd.NewResourceInfoRequest(o.remotePath(), opt2).Exec()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
o.setMetaData(ResourceInfoResponse)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListDir walks the path returning a channel of FsObjects
|
|
|
|
func (f *Fs) ListDir() fs.DirChan {
|
|
|
|
out := make(fs.DirChan, fs.Config.Checkers)
|
|
|
|
go func() {
|
|
|
|
defer close(out)
|
|
|
|
|
|
|
|
//request object meta info
|
|
|
|
var opt yandex.ResourceInfoRequestOptions
|
|
|
|
ResourceInfoResponse, err := f.yd.NewResourceInfoRequest(f.diskRoot, opt).Exec()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if ResourceInfoResponse.ResourceType == "dir" {
|
|
|
|
//list all subdirs
|
|
|
|
for _, element := range ResourceInfoResponse.Embedded.Items {
|
|
|
|
if element.ResourceType == "dir" {
|
|
|
|
t, err := time.Parse(time.RFC3339Nano, element.Modified)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
out <- &fs.Dir{
|
|
|
|
Name: element.Name,
|
|
|
|
When: t,
|
|
|
|
Bytes: int64(element.Size),
|
|
|
|
Count: -1,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}()
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put the object
|
|
|
|
//
|
|
|
|
// Copy the reader in to the new object which is returned
|
|
|
|
//
|
|
|
|
// The new object may have been created if an error is returned
|
|
|
|
func (f *Fs) Put(in io.Reader, remote string, modTime time.Time, size int64) (fs.Object, error) {
|
|
|
|
o := &Object{
|
|
|
|
fs: f,
|
|
|
|
remote: remote,
|
|
|
|
bytes: uint64(size),
|
|
|
|
modTime: modTime,
|
|
|
|
}
|
|
|
|
//TODO maybe read metadata after upload to check if file uploaded successfully
|
|
|
|
return o, o.Update(in, modTime, size)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mkdir creates the container if it doesn't exist
|
|
|
|
func (f *Fs) Mkdir() error {
|
|
|
|
return mkDirFullPath(f.yd, f.diskRoot)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rmdir deletes the container
|
|
|
|
//
|
|
|
|
// Returns an error if it isn't empty
|
|
|
|
func (f *Fs) Rmdir() error {
|
|
|
|
return f.purgeCheck(true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// purgeCheck remotes the root directory, if check is set then it
|
|
|
|
// refuses to do so if it has anything in
|
|
|
|
func (f *Fs) purgeCheck(check bool) error {
|
|
|
|
if check {
|
|
|
|
//to comply with rclone logic we check if the directory is empty before delete.
|
|
|
|
//send request to get list of objects in this directory.
|
|
|
|
var opt yandex.ResourceInfoRequestOptions
|
|
|
|
ResourceInfoResponse, err := f.yd.NewResourceInfoRequest(f.diskRoot, opt).Exec()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Rmdir failed: %s", err)
|
|
|
|
}
|
|
|
|
if len(ResourceInfoResponse.Embedded.Items) != 0 {
|
|
|
|
return fmt.Errorf("Rmdir failed: Directory not empty")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//delete directory
|
|
|
|
return f.yd.Delete(f.diskRoot, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Precision return the precision of this Fs
|
|
|
|
func (f *Fs) Precision() time.Duration {
|
2015-12-30 21:47:44 +01:00
|
|
|
return time.Nanosecond
|
2015-12-07 05:01:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Purge deletes all the files and the container
|
|
|
|
//
|
|
|
|
// Optional interface: Only implement this if you have a way of
|
|
|
|
// deleting all the files quicker than just running Remove() on the
|
|
|
|
// result of List()
|
|
|
|
func (f *Fs) Purge() error {
|
|
|
|
return f.purgeCheck(false)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------------------------------
|
|
|
|
|
|
|
|
// Fs returns the parent Fs
|
|
|
|
func (o *Object) Fs() fs.Fs {
|
|
|
|
return o.fs
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a string version
|
|
|
|
func (o *Object) String() string {
|
|
|
|
if o == nil {
|
|
|
|
return "<nil>"
|
|
|
|
}
|
|
|
|
return o.remote
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remote returns the remote path
|
|
|
|
func (o *Object) Remote() string {
|
|
|
|
return o.remote
|
|
|
|
}
|
|
|
|
|
|
|
|
// Md5sum returns the Md5sum of an object returning a lowercase hex string
|
|
|
|
func (o *Object) Md5sum() (string, error) {
|
|
|
|
return o.md5sum, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Size returns the size of an object in bytes
|
|
|
|
func (o *Object) Size() int64 {
|
|
|
|
var size = int64(o.bytes) //need to cast from uint64 in yandex disk to int64 in rclone. can cause overflow
|
|
|
|
return size
|
|
|
|
}
|
|
|
|
|
|
|
|
// ModTime returns the modification time of the object
|
|
|
|
//
|
|
|
|
// It attempts to read the objects mtime and if that isn't present the
|
|
|
|
// LastModified returned in the http headers
|
|
|
|
func (o *Object) ModTime() time.Time {
|
|
|
|
err := o.readMetaData()
|
|
|
|
if err != nil {
|
|
|
|
fs.Log(o, "Failed to read metadata: %s", err)
|
|
|
|
return time.Now()
|
|
|
|
}
|
|
|
|
return o.modTime
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open an object for read
|
|
|
|
func (o *Object) Open() (in io.ReadCloser, err error) {
|
|
|
|
return o.fs.yd.Download(o.remotePath())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove an object
|
|
|
|
func (o *Object) Remove() error {
|
|
|
|
return o.fs.yd.Delete(o.remotePath(), true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetModTime sets the modification time of the local fs object
|
|
|
|
//
|
|
|
|
// Commits the datastore
|
|
|
|
func (o *Object) SetModTime(modTime time.Time) {
|
|
|
|
remote := o.remotePath()
|
|
|
|
//set custom_property 'rclone_modified' of object to modTime
|
|
|
|
err := o.fs.yd.SetCustomProperty(remote, "rclone_modified", modTime.Format(time.RFC3339Nano))
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Storable returns whether this object is storable
|
|
|
|
func (o *Object) Storable() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the remote path for the object
|
|
|
|
func (o *Object) remotePath() string {
|
|
|
|
return o.fs.diskRoot + o.remote
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the already existing object
|
|
|
|
//
|
|
|
|
// Copy the reader into the object updating modTime and size
|
|
|
|
//
|
|
|
|
// The new object may have been created if an error is returned
|
|
|
|
func (o *Object) Update(in io.Reader, modTime time.Time, size int64) error {
|
|
|
|
remote := o.remotePath()
|
|
|
|
//create full path to file before upload.
|
|
|
|
err1 := mkDirFullPath(o.fs.yd, remote)
|
|
|
|
if err1 != nil {
|
|
|
|
return err1
|
|
|
|
}
|
|
|
|
//upload file
|
|
|
|
overwrite := true //overwrite existing file
|
|
|
|
err := o.fs.yd.Upload(in, remote, overwrite)
|
|
|
|
if err == nil {
|
|
|
|
//if file uploaded sucessfuly then return metadata
|
|
|
|
o.bytes = uint64(size)
|
|
|
|
o.modTime = modTime
|
|
|
|
o.md5sum = "" // according to unit tests after put the md5 is empty.
|
|
|
|
//and set modTime of uploaded file
|
|
|
|
o.SetModTime(modTime)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// utility funcs-------------------------------------------------------------------
|
|
|
|
|
|
|
|
// mkDirExecute execute mkdir
|
|
|
|
func mkDirExecute(client *yandex.Client, path string) (int, string, error) {
|
|
|
|
statusCode, jsonErrorString, err := client.Mkdir(path)
|
|
|
|
if statusCode == 409 { // dir already exist
|
|
|
|
return statusCode, jsonErrorString, err
|
|
|
|
}
|
|
|
|
if statusCode == 201 { // dir was created
|
|
|
|
return statusCode, jsonErrorString, err
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
// error creating directory
|
2015-12-30 22:07:42 +01:00
|
|
|
log.Printf("Failed to create folder: %v", err)
|
2015-12-07 05:01:03 +01:00
|
|
|
return statusCode, jsonErrorString, err
|
|
|
|
}
|
|
|
|
return 0, "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
//mkDirFullPath Creates Each Directory in the path if needed. Send request once for every directory in the path.
|
|
|
|
func mkDirFullPath(client *yandex.Client, path string) error {
|
|
|
|
//trim filename from path
|
|
|
|
dirString := strings.TrimSuffix(path, filepath.Base(path))
|
|
|
|
//trim "disk:/" from path
|
|
|
|
dirString = strings.TrimPrefix(dirString, "disk:/")
|
|
|
|
|
|
|
|
//1 Try to create directory first
|
|
|
|
if _, jsonErrorString, err := mkDirExecute(client, dirString); err != nil {
|
|
|
|
er2, _ := client.ParseAPIError(jsonErrorString)
|
|
|
|
if er2 != "DiskPathPointsToExistentDirectoryError" {
|
|
|
|
//2 if it fails then create all directories in the path from root.
|
|
|
|
dirs := strings.Split(dirString, "/") //path separator /
|
|
|
|
var mkdirpath = "/" //path separator /
|
|
|
|
for _, element := range dirs {
|
|
|
|
if element != "" {
|
|
|
|
mkdirpath += element + "/" //path separator /
|
|
|
|
_, _, err2 := mkDirExecute(client, mkdirpath)
|
|
|
|
if err2 != nil {
|
|
|
|
//we continue even if some directories exist.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the interfaces are satisfied
|
|
|
|
var (
|
|
|
|
_ fs.Fs = (*Fs)(nil)
|
|
|
|
_ fs.Purger = (*Fs)(nil)
|
|
|
|
//_ fs.Copier = (*Fs)(nil)
|
|
|
|
_ fs.Object = (*Object)(nil)
|
|
|
|
)
|