// File system interface package fs import ( "fmt" "io" "log" "regexp" "time" ) // Globals var ( // Filesystem registry fsRegistry []*FsInfo ) // Filesystem info type FsInfo struct { Name string // name of this fs NewFs func(string, string) (Fs, error) // create a new file system Config func(string) // function to call to help with config Options []Option } // An options for a Fs type Option struct { Name string Help string Optional bool Examples []OptionExample } // An example for an option type OptionExample struct { Value string Help string } // Choose an option func (o *Option) Choose() string { fmt.Println(o.Help) if len(o.Examples) > 0 { var values []string var help []string for _, example := range o.Examples { values = append(values, example.Value) help = append(help, example.Help) } return Choose(o.Name, values, help, true) } fmt.Printf("%s> ", o.Name) return ReadLine() } // Register a filesystem // // Fs modules should use this in an init() function func Register(info *FsInfo) { fsRegistry = append(fsRegistry, info) } // A Filesystem, describes the local filesystem and the remote object store type Fs interface { // String returns a description of the FS String() string // List the Fs into a channel List() ObjectsChan // List the Fs directories/buckets/containers into a channel ListDir() DirChan // Find the Object at remote. Returns nil if can't be found NewFsObject(remote string) Object // Put in to the remote path with the modTime given of the given size // // 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 Put(in io.Reader, remote string, modTime time.Time, size int64) (Object, error) // Make the directory (container, bucket) Mkdir() error // Remove the directory (container, bucket) if empty Rmdir() error // Precision of the ModTimes in this Fs Precision() time.Duration } // FIXME make f.Debugf... // A filesystem like object which can either be a remote object or a // local file/directory type Object interface { // Remote returns the remote path Remote() string // Md5sum returns the md5 checksum of the file Md5sum() (string, error) // ModTime returns the modification date of the file ModTime() time.Time // SetModTime sets the metadata on the object to set the modification date SetModTime(time.Time) // Size returns the size of the file Size() int64 // Open opens the file for read. Call Close() on the returned io.ReadCloser Open() (io.ReadCloser, error) // Storable says whether this object can be stored Storable() bool // Removes this object Remove() error } // Optional interfaces type Purger interface { // Purge all files in the root and the root directory // // Implement this if you have a way of deleting all the files // quicker than just running Remove() on the result of List() Purge() error } // A channel of Objects type ObjectsChan chan Object // A slice of Objects type Objects []Object // A structure of directory/container/bucket lists type Dir struct { Name string // name of the directory When time.Time // modification or creation time - IsZero for unknown Bytes int64 // size of directory and contents -1 for unknown Count int64 // number of objects -1 for unknown } // A channel of Dir objects type DirChan chan *Dir // Pattern to match a url var matcher = regexp.MustCompile(`^([\w_-]+)://(.*)$`) // Finds a FsInfo object for the name passed in // // Services are looked up in the config file func Find(name string) (*FsInfo, error) { for _, item := range fsRegistry { if item.Name == name { return item, nil } } return nil, fmt.Errorf("Didn't find filing system for %q", name) } // NewFs makes a new Fs object from the path // // The path is of the form service://path // // Services are looked up in the config file func NewFs(path string) (Fs, error) { parts := matcher.FindStringSubmatch(path) fsName, configName, fsPath := "local", "local", path if parts != nil { configName, fsPath = parts[1], parts[2] var err error fsName, err = ConfigFile.GetValue(configName, "type") if err != nil { return nil, fmt.Errorf("Didn't find section in config file for %q", configName) } } fs, err := Find(fsName) if err != nil { return nil, err } return fs.NewFs(configName, fsPath) } // Write debuging output for this Object func Debug(fs Object, text string, args ...interface{}) { if Config.Verbose { out := fmt.Sprintf(text, args...) log.Printf("%s: %s", fs.Remote(), out) } } // Write log output for this Object func Log(fs Object, text string, args ...interface{}) { if !Config.Quiet { out := fmt.Sprintf(text, args...) log.Printf("%s: %s", fs.Remote(), out) } } // checkClose is a utility function used to check the return from // Close in a defer statement. func checkClose(c io.Closer, err *error) { cerr := c.Close() if *err == nil { *err = cerr } } // Work out modify window for fses passed in - sets Config.ModifyWindow // // This is the largest modify window of all the fses in use, and the // user configured value func CalculateModifyWindow(fs ...Fs) { for _, f := range fs { if f != nil { precision := f.Precision() if precision > Config.ModifyWindow { Config.ModifyWindow = precision } } } if Config.Verbose { log.Printf("Modify window is %s\n", Config.ModifyWindow) } } // Check the two files to see if the MD5sums are the same // // May return an error which will already have been logged // // If an error is returned it will return false func CheckMd5sums(src, dst Object) (bool, error) { srcMd5, err := src.Md5sum() if err != nil { Stats.Error() Log(src, "Failed to calculate src md5: %s", err) return false, err } dstMd5, err := dst.Md5sum() if err != nil { Stats.Error() Log(dst, "Failed to calculate dst md5: %s", err) return false, err } // Debug("Src MD5 %s", srcMd5) // Debug("Dst MD5 %s", obj.Hash) return srcMd5 == dstMd5, nil } // Checks to see if the src and dst objects are equal by looking at // size, mtime and MD5SUM // // If the src and dst size are different then it is considered to be // not equal. // // If the size is the same and the mtime is the same then it is // considered to be equal. This is the heuristic rsync uses when // not using --checksum. // // If the size is the same and and mtime is different or unreadable // and the MD5SUM is the same then the file is considered to be equal. // In this case the mtime on the dst is updated. // // Otherwise the file is considered to be not equal including if there // were errors reading info. func Equal(src, dst Object) bool { if src.Size() != dst.Size() { Debug(src, "Sizes differ") return false } // Size the same so check the mtime srcModTime := src.ModTime() dstModTime := dst.ModTime() dt := dstModTime.Sub(srcModTime) ModifyWindow := Config.ModifyWindow if dt >= ModifyWindow || dt <= -ModifyWindow { Debug(src, "Modification times differ by %s: %v, %v", dt, srcModTime, dstModTime) } else { Debug(src, "Size and modification time differ by %s (within %s)", dt, ModifyWindow) return true } // mtime is unreadable or different but size is the same so // check the MD5SUM same, _ := CheckMd5sums(src, dst) if !same { Debug(src, "Md5sums differ") return false } // Size and MD5 the same but mtime different so update the // mtime of the dst object here dst.SetModTime(srcModTime) Debug(src, "Size and MD5SUM of src and dst objects identical") return true } // Copy src object to f func Copy(f Fs, src Object) { in0, err := src.Open() if err != nil { Stats.Error() Log(src, "Failed to open: %s", err) return } in := NewAccount(in0) // account the transfer dst, err := f.Put(in, src.Remote(), src.ModTime(), src.Size()) inErr := in.Close() if err == nil { err = inErr } if err != nil { Stats.Error() Log(src, "Failed to copy: %s", err) if dst != nil { Debug(dst, "Removing failed copy") removeErr := dst.Remove() if removeErr != nil { Stats.Error() Log(dst, "Failed to remove failed copy: %s", removeErr) } } return } Debug(src, "Copied") }