forked from TrueCloudLab/rclone
drive: restructure Object type
This commit is contained in:
parent
171e39b230
commit
0b2fc621fc
1 changed files with 363 additions and 281 deletions
|
@ -337,18 +337,27 @@ type Fs struct {
|
||||||
isTeamDrive bool // true if this is a team drive
|
isTeamDrive bool // true if this is a team drive
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type baseObject struct {
|
||||||
|
fs *Fs // what this object is part of
|
||||||
|
remote string // The remote path
|
||||||
|
id string // Drive Id of this object
|
||||||
|
modifiedDate string // RFC3339 time it was last modified
|
||||||
|
mimeType string // The object MIME type
|
||||||
|
bytes int64 // size of the object
|
||||||
|
}
|
||||||
|
type documentObject struct {
|
||||||
|
baseObject
|
||||||
|
url string // Download URL of this object
|
||||||
|
documentMimeType string // the original document MIME type
|
||||||
|
extLen int // The length of the added export extension
|
||||||
|
}
|
||||||
|
|
||||||
// Object describes a drive object
|
// Object describes a drive object
|
||||||
type Object struct {
|
type Object struct {
|
||||||
fs *Fs // what this object is part of
|
baseObject
|
||||||
remote string // The remote path
|
url string // Download URL of this object
|
||||||
id string // Drive Id of this object
|
md5sum string // md5sum of the object
|
||||||
url string // Download URL of this object
|
v2Download bool // generate v2 download link ondemand
|
||||||
md5sum string // md5sum of the object
|
|
||||||
bytes int64 // size of the object
|
|
||||||
modifiedDate string // RFC3339 time it was last modified
|
|
||||||
documentMimeType string // if set this is a Google doc
|
|
||||||
v2Download bool // generate v2 download link ondemand
|
|
||||||
mimeType string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
// ------------------------------------------------------------
|
||||||
|
@ -822,54 +831,138 @@ func NewFs(name, path string, m configmap.Mapper) (fs.Fs, error) {
|
||||||
return f, nil
|
return f, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return an Object from a path
|
func (f *Fs) newBaseObject(remote string, info *drive.File) baseObject {
|
||||||
//
|
modifiedDate := info.ModifiedTime
|
||||||
// If it can't be found it returns the error fs.ErrorObjectNotFound.
|
if f.opt.UseCreatedDate {
|
||||||
func (f *Fs) newObjectWithInfo(remote string, info *drive.File) (fs.Object, error) {
|
modifiedDate = info.CreatedTime
|
||||||
o := &Object{
|
|
||||||
fs: f,
|
|
||||||
remote: remote,
|
|
||||||
}
|
}
|
||||||
if info != nil {
|
return baseObject{
|
||||||
o.setMetaData(info)
|
fs: f,
|
||||||
} else {
|
remote: remote,
|
||||||
err := o.readMetaData() // reads info and meta, returning an error
|
id: info.Id,
|
||||||
if err != nil {
|
modifiedDate: modifiedDate,
|
||||||
return nil, err
|
mimeType: info.MimeType,
|
||||||
}
|
bytes: info.Size,
|
||||||
}
|
}
|
||||||
return o, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *Fs) newDocumentObjectWithInfo(remote, extension, mimeType string, info *drive.File) (fs.Object, error) {
|
// newRegularObject creates a fs.Object for a normal drive.File
|
||||||
o, err := f.newObjectWithInfo(remote, info)
|
func (f *Fs) newRegularObject(remote string, info *drive.File) fs.Object {
|
||||||
|
return &Object{
|
||||||
|
baseObject: f.newBaseObject(remote, info),
|
||||||
|
url: fmt.Sprintf("%sfiles/%s?alt=media", f.svc.BasePath, info.Id),
|
||||||
|
md5sum: strings.ToLower(info.Md5Checksum),
|
||||||
|
v2Download: f.opt.V2DownloadMinSize != -1 && info.Size >= int64(f.opt.V2DownloadMinSize),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// newDocumentObject creates a fs.Object for a google docs drive.File
|
||||||
|
func (f *Fs) newDocumentObject(remote string, info *drive.File, extension, exportMimeType string) (fs.Object, error) {
|
||||||
|
mediaType, _, err := mime.ParseMediaType(exportMimeType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
o.(*Object).setGdocsMetaData(info, extension, mimeType)
|
url := fmt.Sprintf("%sfiles/%s/export?mimeType=%s", f.svc.BasePath, info.Id, url.QueryEscape(mediaType))
|
||||||
return o, nil
|
if f.opt.AlternateExport {
|
||||||
|
switch info.MimeType {
|
||||||
|
case "application/vnd.google-apps.drawing":
|
||||||
|
url = fmt.Sprintf("https://docs.google.com/drawings/d/%s/export/%s", info.Id, extension[1:])
|
||||||
|
case "application/vnd.google-apps.document":
|
||||||
|
url = fmt.Sprintf("https://docs.google.com/document/d/%s/export?format=%s", info.Id, extension[1:])
|
||||||
|
case "application/vnd.google-apps.spreadsheet":
|
||||||
|
url = fmt.Sprintf("https://docs.google.com/spreadsheets/d/%s/export?format=%s", info.Id, extension[1:])
|
||||||
|
case "application/vnd.google-apps.presentation":
|
||||||
|
url = fmt.Sprintf("https://docs.google.com/presentation/d/%s/export/%s", info.Id, extension[1:])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
baseObject := f.newBaseObject(remote+extension, info)
|
||||||
|
baseObject.bytes = -1
|
||||||
|
baseObject.mimeType = exportMimeType
|
||||||
|
return &documentObject{
|
||||||
|
baseObject: baseObject,
|
||||||
|
url: url,
|
||||||
|
documentMimeType: info.MimeType,
|
||||||
|
extLen: len(extension),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// newObjectWithInfo creates a fs.Object for any drive.File
|
||||||
|
//
|
||||||
|
// When the drive.File cannot be represented as a fs.Object it will return (nil, nil).
|
||||||
|
func (f *Fs) newObjectWithInfo(remote string, info *drive.File) (fs.Object, error) {
|
||||||
|
// If item has MD5 sum or a length it is a file stored on drive
|
||||||
|
if info.Md5Checksum != "" || info.Size > 0 {
|
||||||
|
return f.newRegularObject(remote, info), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
extension, exportName, exportMimeType, isDocument := f.findExportFormat(info)
|
||||||
|
return f.newObjectWithExportInfo(remote, info, extension, exportName, exportMimeType, isDocument)
|
||||||
|
}
|
||||||
|
|
||||||
|
// newObjectWithExportInfo creates a fs.Object for any drive.File and the result of findExportFormat
|
||||||
|
//
|
||||||
|
// When the drive.File cannot be represented as a fs.Object it will return (nil, nil).
|
||||||
|
func (f *Fs) newObjectWithExportInfo(
|
||||||
|
remote string, info *drive.File,
|
||||||
|
extension, exportName, exportMimeType string, isDocument bool) (fs.Object, error) {
|
||||||
|
switch {
|
||||||
|
case info.Md5Checksum != "" || info.Size > 0:
|
||||||
|
// If item has MD5 sum or a length it is a file stored on drive
|
||||||
|
return f.newRegularObject(remote, info), nil
|
||||||
|
case f.opt.SkipGdocs:
|
||||||
|
fs.Debugf(remote, "Skipping google document type %q", info.MimeType)
|
||||||
|
return nil, nil
|
||||||
|
default:
|
||||||
|
// If item MimeType is in the ExportFormats then it is a google doc
|
||||||
|
if !isDocument {
|
||||||
|
fs.Debugf(remote, "Ignoring unknown document type %q", info.MimeType)
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
if extension == "" {
|
||||||
|
fs.Debugf(remote, "No export formats found for %q", info.MimeType)
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return f.newDocumentObject(remote, info, extension, exportMimeType)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewObject finds the Object at remote. If it can't be found
|
// NewObject finds the Object at remote. If it can't be found
|
||||||
// it returns the error fs.ErrorObjectNotFound.
|
// it returns the error fs.ErrorObjectNotFound.
|
||||||
func (f *Fs) NewObject(remote string) (fs.Object, error) {
|
func (f *Fs) NewObject(remote string) (fs.Object, error) {
|
||||||
return f.newObjectWithInfo(remote, nil)
|
info, extension, exportName, exportMimeType, isDocument, err := f.getRemoteInfoWithExport(remote)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
remote = remote[:len(remote)-len(extension)]
|
||||||
|
obj, err := f.newObjectWithExportInfo(remote, info, extension, exportName, exportMimeType, isDocument)
|
||||||
|
switch {
|
||||||
|
case err != nil:
|
||||||
|
return nil, err
|
||||||
|
case obj == nil:
|
||||||
|
return nil, fs.ErrorObjectNotFound
|
||||||
|
default:
|
||||||
|
return obj, nil
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
// FindLeaf finds a directory of name leaf in the folder with ID pathID
|
||||||
func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err error) {
|
func (f *Fs) FindLeaf(pathID, leaf string) (pathIDOut string, found bool, err error) {
|
||||||
// Find the leaf in pathID
|
// Find the leaf in pathID
|
||||||
found, err = f.list([]string{pathID}, leaf, true, false, false, func(item *drive.File) bool {
|
found, err = f.list([]string{pathID}, leaf, true, false, false, func(item *drive.File) bool {
|
||||||
if item.Name == leaf {
|
|
||||||
pathIDOut = item.Id
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if !f.opt.SkipGdocs {
|
if !f.opt.SkipGdocs {
|
||||||
_, exportName, _, _ := f.findExportFormat(item)
|
_, exportName, _, isDocument := f.findExportFormat(item)
|
||||||
if exportName == leaf {
|
if exportName == leaf {
|
||||||
pathIDOut = item.Id
|
pathIDOut = item.Id
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
if isDocument {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if item.Name == leaf {
|
||||||
|
pathIDOut = item.Id
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
|
@ -1217,6 +1310,9 @@ func (f *Fs) ListR(dir string, callback fs.ListRCallback) (err error) {
|
||||||
return list.Flush()
|
return list.Flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// itemToDirEntry converts a drive.File to a fs.DirEntry.
|
||||||
|
// When the drive.File cannot be represented as a fs.DirEntry
|
||||||
|
// (nil, nil) is returned.
|
||||||
func (f *Fs) itemToDirEntry(remote string, item *drive.File) (fs.DirEntry, error) {
|
func (f *Fs) itemToDirEntry(remote string, item *drive.File) (fs.DirEntry, error) {
|
||||||
switch {
|
switch {
|
||||||
case item.MimeType == driveFolderType:
|
case item.MimeType == driveFolderType:
|
||||||
|
@ -1227,50 +1323,19 @@ func (f *Fs) itemToDirEntry(remote string, item *drive.File) (fs.DirEntry, error
|
||||||
return d, nil
|
return d, nil
|
||||||
case f.opt.AuthOwnerOnly && !isAuthOwned(item):
|
case f.opt.AuthOwnerOnly && !isAuthOwned(item):
|
||||||
// ignore object
|
// ignore object
|
||||||
case item.Md5Checksum != "" || item.Size > 0:
|
|
||||||
// If item has MD5 sum or a length it is a file stored on drive
|
|
||||||
o, err := f.newObjectWithInfo(remote, item)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return o, nil
|
|
||||||
case f.opt.SkipGdocs:
|
|
||||||
fs.Debugf(remote, "Skipping google document type %q", item.MimeType)
|
|
||||||
default:
|
default:
|
||||||
// If item MimeType is in the ExportFormats then it is a google doc
|
return f.newObjectWithInfo(remote, item)
|
||||||
extension, _, exportMimeType, isDocument := f.findExportFormat(item)
|
|
||||||
if !isDocument {
|
|
||||||
fs.Debugf(remote, "Ignoring unknown document type %q", item.MimeType)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if extension == "" {
|
|
||||||
fs.Debugf(remote, "No export formats found for %q", item.MimeType)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
o, err := f.newDocumentObjectWithInfo(remote, extension, exportMimeType, item)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return o, nil
|
|
||||||
}
|
}
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Creates a drive.File info from the parameters passed in and a half
|
// Creates a drive.File info from the parameters passed in.
|
||||||
// finished Object which must have setMetaData called on it
|
|
||||||
//
|
//
|
||||||
// Used to create new objects
|
// Used to create new objects
|
||||||
func (f *Fs) createFileInfo(remote string, modTime time.Time, size int64) (*Object, *drive.File, error) {
|
func (f *Fs) createFileInfo(remote string, modTime time.Time) (*drive.File, error) {
|
||||||
// Temporary Object under construction
|
|
||||||
o := &Object{
|
|
||||||
fs: f,
|
|
||||||
remote: remote,
|
|
||||||
bytes: size,
|
|
||||||
}
|
|
||||||
|
|
||||||
leaf, directoryID, err := f.dirCache.FindRootAndPath(remote, true)
|
leaf, directoryID, err := f.dirCache.FindRootAndPath(remote, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Define the metadata for the file we are going to create.
|
// Define the metadata for the file we are going to create.
|
||||||
|
@ -1281,7 +1346,7 @@ func (f *Fs) createFileInfo(remote string, modTime time.Time, size int64) (*Obje
|
||||||
MimeType: fs.MimeTypeFromName(remote),
|
MimeType: fs.MimeTypeFromName(remote),
|
||||||
ModifiedTime: modTime.Format(timeFormatOut),
|
ModifiedTime: modTime.Format(timeFormatOut),
|
||||||
}
|
}
|
||||||
return o, createInfo, nil
|
return createInfo, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Put the object
|
// Put the object
|
||||||
|
@ -1290,7 +1355,7 @@ func (f *Fs) createFileInfo(remote string, modTime time.Time, size int64) (*Obje
|
||||||
//
|
//
|
||||||
// The new object may have been created if an error is returned
|
// The new object may have been created if an error is returned
|
||||||
func (f *Fs) Put(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) (fs.Object, error) {
|
func (f *Fs) Put(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) (fs.Object, error) {
|
||||||
exisitingObj, err := f.newObjectWithInfo(src.Remote(), nil)
|
exisitingObj, err := f.NewObject(src.Remote())
|
||||||
switch err {
|
switch err {
|
||||||
case nil:
|
case nil:
|
||||||
return exisitingObj, exisitingObj.Update(in, src, options...)
|
return exisitingObj, exisitingObj.Update(in, src, options...)
|
||||||
|
@ -1319,7 +1384,6 @@ func (f *Fs) PutUnchecked(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOpt
|
||||||
srcExt := path.Ext(remote)
|
srcExt := path.Ext(remote)
|
||||||
exportExt := ""
|
exportExt := ""
|
||||||
importMimeType := ""
|
importMimeType := ""
|
||||||
exportMimeType := ""
|
|
||||||
|
|
||||||
if f.importMimeTypes != nil && !f.opt.SkipGdocs {
|
if f.importMimeTypes != nil && !f.opt.SkipGdocs {
|
||||||
importMimeType = f.findImportFormat(srcMimeType)
|
importMimeType = f.findImportFormat(srcMimeType)
|
||||||
|
@ -1327,7 +1391,7 @@ func (f *Fs) PutUnchecked(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOpt
|
||||||
if isInternalMimeType(importMimeType) {
|
if isInternalMimeType(importMimeType) {
|
||||||
remote = remote[:len(remote)-len(srcExt)]
|
remote = remote[:len(remote)-len(srcExt)]
|
||||||
|
|
||||||
exportExt, exportMimeType, _ = f.findExportFormatByMimeType(importMimeType)
|
exportExt, _, _ = f.findExportFormatByMimeType(importMimeType)
|
||||||
if exportExt == "" {
|
if exportExt == "" {
|
||||||
return nil, errors.Errorf("No export format found for %q", importMimeType)
|
return nil, errors.Errorf("No export format found for %q", importMimeType)
|
||||||
}
|
}
|
||||||
|
@ -1337,7 +1401,7 @@ func (f *Fs) PutUnchecked(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOpt
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
o, createInfo, err := f.createFileInfo(remote, modTime, size)
|
createInfo, err := f.createFileInfo(remote, modTime)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -1359,20 +1423,16 @@ func (f *Fs) PutUnchecked(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOpt
|
||||||
return shouldRetry(err)
|
return shouldRetry(err)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return o, err
|
return nil, err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Upload the file in chunks
|
// Upload the file in chunks
|
||||||
info, err = f.Upload(in, size, srcMimeType, "", createInfo, remote)
|
info, err = f.Upload(in, size, srcMimeType, "", createInfo, remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return o, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if isInternalMimeType(importMimeType) {
|
return f.newObjectWithInfo(remote, info)
|
||||||
return f.newDocumentObjectWithInfo(remote, exportExt, exportMimeType, info)
|
|
||||||
}
|
|
||||||
o.setMetaData(info)
|
|
||||||
return o, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// MergeDirs merges the contents of all the directories passed
|
// MergeDirs merges the contents of all the directories passed
|
||||||
|
@ -1510,23 +1570,36 @@ func (f *Fs) Precision() time.Duration {
|
||||||
//
|
//
|
||||||
// If it isn't possible then return fs.ErrorCantCopy
|
// If it isn't possible then return fs.ErrorCantCopy
|
||||||
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
var srcObj *baseObject
|
||||||
if !ok {
|
srcRemote := src.Remote()
|
||||||
|
ext := ""
|
||||||
|
switch src := src.(type) {
|
||||||
|
case *Object:
|
||||||
|
srcObj = &src.baseObject
|
||||||
|
case *documentObject:
|
||||||
|
srcObj = &src.baseObject
|
||||||
|
srcRemote, ext = srcRemote[:len(srcRemote)-src.extLen], srcRemote[len(srcRemote)-src.extLen:]
|
||||||
|
default:
|
||||||
fs.Debugf(src, "Can't copy - not same remote type")
|
fs.Debugf(src, "Can't copy - not same remote type")
|
||||||
return nil, fs.ErrorCantCopy
|
return nil, fs.ErrorCantCopy
|
||||||
}
|
}
|
||||||
if srcObj.documentMimeType != "" {
|
|
||||||
return nil, errors.New("can't copy a Google document")
|
if ext != "" {
|
||||||
|
if !strings.HasSuffix(remote, ext) {
|
||||||
|
fs.Debugf(src, "Can't copy - not same document type")
|
||||||
|
return nil, fs.ErrorCantCopy
|
||||||
|
}
|
||||||
|
remote = remote[:len(remote)-len(ext)]
|
||||||
}
|
}
|
||||||
|
|
||||||
o, createInfo, err := f.createFileInfo(remote, srcObj.ModTime(), srcObj.bytes)
|
createInfo, err := f.createFileInfo(remote, src.ModTime())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var info *drive.File
|
var info *drive.File
|
||||||
err = o.fs.pacer.Call(func() (bool, error) {
|
err = f.pacer.Call(func() (bool, error) {
|
||||||
info, err = o.fs.svc.Files.Copy(srcObj.id, createInfo).
|
info, err = f.svc.Files.Copy(srcObj.id, createInfo).
|
||||||
Fields(googleapi.Field(partialFields)).
|
Fields(googleapi.Field(partialFields)).
|
||||||
SupportsTeamDrives(f.isTeamDrive).
|
SupportsTeamDrives(f.isTeamDrive).
|
||||||
KeepRevisionForever(f.opt.KeepRevisionForever).
|
KeepRevisionForever(f.opt.KeepRevisionForever).
|
||||||
|
@ -1536,9 +1609,7 @@ func (f *Fs) Copy(src fs.Object, remote string) (fs.Object, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
return f.newObjectWithInfo(remote, info)
|
||||||
o.setMetaData(info)
|
|
||||||
return o, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Purge deletes all the files and the container
|
// Purge deletes all the files and the container
|
||||||
|
@ -1629,21 +1700,35 @@ func (f *Fs) About() (*fs.Usage, error) {
|
||||||
//
|
//
|
||||||
// If it isn't possible then return fs.ErrorCantMove
|
// If it isn't possible then return fs.ErrorCantMove
|
||||||
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
||||||
srcObj, ok := src.(*Object)
|
var srcObj *baseObject
|
||||||
if !ok {
|
srcRemote := src.Remote()
|
||||||
|
ext := ""
|
||||||
|
switch src := src.(type) {
|
||||||
|
case *Object:
|
||||||
|
srcObj = &src.baseObject
|
||||||
|
case *documentObject:
|
||||||
|
srcObj = &src.baseObject
|
||||||
|
srcRemote, ext = srcRemote[:len(srcRemote)-src.extLen], srcRemote[len(srcRemote)-src.extLen:]
|
||||||
|
default:
|
||||||
fs.Debugf(src, "Can't move - not same remote type")
|
fs.Debugf(src, "Can't move - not same remote type")
|
||||||
return nil, fs.ErrorCantMove
|
return nil, fs.ErrorCantMove
|
||||||
}
|
}
|
||||||
if srcObj.documentMimeType != "" {
|
|
||||||
return nil, errors.New("can't move a Google document")
|
if ext != "" {
|
||||||
|
if !strings.HasSuffix(remote, ext) {
|
||||||
|
fs.Debugf(src, "Can't move - not same document type")
|
||||||
|
return nil, fs.ErrorCantMove
|
||||||
|
}
|
||||||
|
remote = remote[:len(remote)-len(ext)]
|
||||||
}
|
}
|
||||||
|
|
||||||
_, srcParentID, err := srcObj.fs.dirCache.FindPath(src.Remote(), false)
|
_, srcParentID, err := srcObj.fs.dirCache.FindPath(src.Remote(), false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Temporary Object under construction
|
// Temporary Object under construction
|
||||||
dstObj, dstInfo, err := f.createFileInfo(remote, srcObj.ModTime(), srcObj.bytes)
|
dstInfo, err := f.createFileInfo(remote, src.ModTime())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -1665,8 +1750,7 @@ func (f *Fs) Move(src fs.Object, remote string) (fs.Object, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
dstObj.setMetaData(info)
|
return f.newObjectWithInfo(remote, info)
|
||||||
return dstObj, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// PublicLink adds a "readable by anyone with link" permission on the given file or folder.
|
// PublicLink adds a "readable by anyone with link" permission on the given file or folder.
|
||||||
|
@ -1676,14 +1760,11 @@ func (f *Fs) PublicLink(remote string) (link string, err error) {
|
||||||
fs.Debugf(f, "attempting to share directory '%s'", remote)
|
fs.Debugf(f, "attempting to share directory '%s'", remote)
|
||||||
} else {
|
} else {
|
||||||
fs.Debugf(f, "attempting to share single file '%s'", remote)
|
fs.Debugf(f, "attempting to share single file '%s'", remote)
|
||||||
o := &Object{
|
o, err := f.NewObject(remote)
|
||||||
fs: f,
|
if err != nil {
|
||||||
remote: remote,
|
return "", err
|
||||||
}
|
}
|
||||||
if err = o.readMetaData(); err != nil {
|
id = o.(fs.IDer).ID()
|
||||||
return
|
|
||||||
}
|
|
||||||
id = o.id
|
|
||||||
}
|
}
|
||||||
|
|
||||||
permission := &drive.Permission{
|
permission := &drive.Permission{
|
||||||
|
@ -1967,10 +2048,15 @@ func (f *Fs) Hashes() hash.Set {
|
||||||
// ------------------------------------------------------------
|
// ------------------------------------------------------------
|
||||||
|
|
||||||
// Fs returns the parent Fs
|
// Fs returns the parent Fs
|
||||||
func (o *Object) Fs() fs.Info {
|
func (o *baseObject) Fs() fs.Info {
|
||||||
return o.fs
|
return o.fs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Return a string version
|
||||||
|
func (o *baseObject) String() string {
|
||||||
|
return o.remote
|
||||||
|
}
|
||||||
|
|
||||||
// Return a string version
|
// Return a string version
|
||||||
func (o *Object) String() string {
|
func (o *Object) String() string {
|
||||||
if o == nil {
|
if o == nil {
|
||||||
|
@ -1980,7 +2066,7 @@ func (o *Object) String() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remote returns the remote path
|
// Remote returns the remote path
|
||||||
func (o *Object) Remote() string {
|
func (o *baseObject) Remote() string {
|
||||||
return o.remote
|
return o.remote
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1991,85 +2077,59 @@ func (o *Object) Hash(t hash.Type) (string, error) {
|
||||||
}
|
}
|
||||||
return o.md5sum, nil
|
return o.md5sum, nil
|
||||||
}
|
}
|
||||||
|
func (o *baseObject) Hash(t hash.Type) (string, error) {
|
||||||
|
if t != hash.MD5 {
|
||||||
|
return "", hash.ErrUnsupported
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
// Size returns the size of an object in bytes
|
// Size returns the size of an object in bytes
|
||||||
func (o *Object) Size() int64 {
|
func (o *baseObject) Size() int64 {
|
||||||
return o.bytes
|
return o.bytes
|
||||||
}
|
}
|
||||||
|
|
||||||
// setMetaData sets the fs data from a drive.File
|
// getRemoteInfo returns a drive.File for the remote
|
||||||
func (o *Object) setMetaData(info *drive.File) {
|
func (f *Fs) getRemoteInfo(remote string) (info *drive.File, err error) {
|
||||||
o.id = info.Id
|
info, _, _, _, _, err = f.getRemoteInfoWithExport(remote)
|
||||||
o.url = fmt.Sprintf("%sfiles/%s?alt=media", o.fs.svc.BasePath, info.Id)
|
return
|
||||||
o.md5sum = strings.ToLower(info.Md5Checksum)
|
|
||||||
o.bytes = info.Size
|
|
||||||
if o.bytes != -1 && o.fs.opt.V2DownloadMinSize != -1 && o.bytes >= int64(o.fs.opt.V2DownloadMinSize) {
|
|
||||||
o.v2Download = true
|
|
||||||
}
|
|
||||||
if o.fs.opt.UseCreatedDate {
|
|
||||||
o.modifiedDate = info.CreatedTime
|
|
||||||
} else {
|
|
||||||
o.modifiedDate = info.ModifiedTime
|
|
||||||
}
|
|
||||||
o.mimeType = info.MimeType
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// setGdocsMetaData only sets the gdocs related fields
|
// getRemoteInfoWithExport returns a drive.File and the export settings for the remote
|
||||||
func (o *Object) setGdocsMetaData(info *drive.File, extension, exportMimeType string) {
|
func (f *Fs) getRemoteInfoWithExport(remote string) (
|
||||||
o.url = fmt.Sprintf("%sfiles/%s/export?mimeType=%s", o.fs.svc.BasePath, info.Id, url.QueryEscape(exportMimeType))
|
info *drive.File, extension, exportName, exportMimeType string, isDocument bool, err error) {
|
||||||
if o.fs.opt.AlternateExport {
|
leaf, directoryID, err := f.dirCache.FindRootAndPath(remote, false)
|
||||||
switch info.MimeType {
|
|
||||||
case "application/vnd.google-apps.drawing":
|
|
||||||
o.url = fmt.Sprintf("https://docs.google.com/drawings/d/%s/export/%s", info.Id, extension[1:])
|
|
||||||
case "application/vnd.google-apps.document":
|
|
||||||
o.url = fmt.Sprintf("https://docs.google.com/document/d/%s/export?format=%s", info.Id, extension[1:])
|
|
||||||
case "application/vnd.google-apps.spreadsheet":
|
|
||||||
o.url = fmt.Sprintf("https://docs.google.com/spreadsheets/d/%s/export?format=%s", info.Id, extension[1:])
|
|
||||||
case "application/vnd.google-apps.presentation":
|
|
||||||
o.url = fmt.Sprintf("https://docs.google.com/presentation/d/%s/export/%s", info.Id, extension[1:])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
o.documentMimeType = o.mimeType
|
|
||||||
o.mimeType = exportMimeType
|
|
||||||
o.bytes = -1
|
|
||||||
}
|
|
||||||
|
|
||||||
// readMetaData gets the info if it hasn't already been fetched
|
|
||||||
func (o *Object) readMetaData() (err error) {
|
|
||||||
if o.id != "" {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
leaf, directoryID, err := o.fs.dirCache.FindRootAndPath(o.remote, false)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err == fs.ErrorDirNotFound {
|
if err == fs.ErrorDirNotFound {
|
||||||
return fs.ErrorObjectNotFound
|
return nil, "", "", "", false, fs.ErrorObjectNotFound
|
||||||
}
|
}
|
||||||
return err
|
return nil, "", "", "", false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
found, err := o.fs.list([]string{directoryID}, leaf, false, true, false, func(item *drive.File) bool {
|
found, err := f.list([]string{directoryID}, leaf, false, true, false, func(item *drive.File) bool {
|
||||||
if item.Name == leaf {
|
if !f.opt.SkipGdocs {
|
||||||
o.setMetaData(item)
|
extension, exportName, exportMimeType, isDocument = f.findExportFormat(item)
|
||||||
return true
|
|
||||||
}
|
|
||||||
if !o.fs.opt.SkipGdocs {
|
|
||||||
extension, exportName, exportMimeType, _ := o.fs.findExportFormat(item)
|
|
||||||
if exportName == leaf {
|
if exportName == leaf {
|
||||||
o.setMetaData(item)
|
info = item
|
||||||
o.setGdocsMetaData(item, extension, exportMimeType)
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
if isDocument {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if item.Name == leaf {
|
||||||
|
info = item
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return nil, "", "", "", false, err
|
||||||
}
|
}
|
||||||
if !found {
|
if !found {
|
||||||
return fs.ErrorObjectNotFound
|
return nil, "", "", "", false, fs.ErrorObjectNotFound
|
||||||
}
|
}
|
||||||
return nil
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// ModTime returns the modification time of the object
|
// ModTime returns the modification time of the object
|
||||||
|
@ -2077,12 +2137,7 @@ func (o *Object) readMetaData() (err error) {
|
||||||
//
|
//
|
||||||
// It attempts to read the objects mtime and if that isn't present the
|
// It attempts to read the objects mtime and if that isn't present the
|
||||||
// LastModified returned in the http headers
|
// LastModified returned in the http headers
|
||||||
func (o *Object) ModTime() time.Time {
|
func (o *baseObject) ModTime() time.Time {
|
||||||
err := o.readMetaData()
|
|
||||||
if err != nil {
|
|
||||||
fs.Debugf(o, "Failed to read metadata: %v", err)
|
|
||||||
return time.Now()
|
|
||||||
}
|
|
||||||
modTime, err := time.Parse(timeFormatIn, o.modifiedDate)
|
modTime, err := time.Parse(timeFormatIn, o.modifiedDate)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Debugf(o, "Failed to read mtime from object: %v", err)
|
fs.Debugf(o, "Failed to read mtime from object: %v", err)
|
||||||
|
@ -2092,18 +2147,15 @@ func (o *Object) ModTime() time.Time {
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetModTime sets the modification time of the drive fs object
|
// SetModTime sets the modification time of the drive fs object
|
||||||
func (o *Object) SetModTime(modTime time.Time) error {
|
func (o *baseObject) SetModTime(modTime time.Time) error {
|
||||||
err := o.readMetaData()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
// New metadata
|
// New metadata
|
||||||
updateInfo := &drive.File{
|
updateInfo := &drive.File{
|
||||||
ModifiedTime: modTime.Format(timeFormatOut),
|
ModifiedTime: modTime.Format(timeFormatOut),
|
||||||
}
|
}
|
||||||
// Set modified date
|
// Set modified date
|
||||||
var info *drive.File
|
var info *drive.File
|
||||||
err = o.fs.pacer.Call(func() (bool, error) {
|
err := o.fs.pacer.Call(func() (bool, error) {
|
||||||
|
var err error
|
||||||
info, err = o.fs.svc.Files.Update(o.id, updateInfo).
|
info, err = o.fs.svc.Files.Update(o.id, updateInfo).
|
||||||
Fields(googleapi.Field(partialFields)).
|
Fields(googleapi.Field(partialFields)).
|
||||||
SupportsTeamDrives(o.fs.isTeamDrive).
|
SupportsTeamDrives(o.fs.isTeamDrive).
|
||||||
|
@ -2114,46 +2166,22 @@ func (o *Object) SetModTime(modTime time.Time) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Update info from read data
|
// Update info from read data
|
||||||
o.setMetaData(info)
|
o.modifiedDate = info.ModifiedTime
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Storable returns a boolean as to whether this object is storable
|
// Storable returns a boolean as to whether this object is storable
|
||||||
func (o *Object) Storable() bool {
|
func (o *baseObject) Storable() bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// httpResponse gets an http.Response object for the object o.url
|
// httpResponse gets an http.Response object for the object
|
||||||
// using the method passed in
|
// using the url and method passed in
|
||||||
func (o *Object) httpResponse(method string, options []fs.OpenOption) (req *http.Request, res *http.Response, err error) {
|
func (o *baseObject) httpResponse(url, method string, options []fs.OpenOption) (req *http.Request, res *http.Response, err error) {
|
||||||
if o.url == "" {
|
if url == "" {
|
||||||
return nil, nil, errors.New("forbidden to download - check sharing permission")
|
return nil, nil, errors.New("forbidden to download - check sharing permission")
|
||||||
}
|
}
|
||||||
if o.documentMimeType != "" {
|
req, err = http.NewRequest(method, url, nil)
|
||||||
for _, o := range options {
|
|
||||||
// https://developers.google.com/drive/v3/web/manage-downloads#partial_download
|
|
||||||
if _, ok := o.(*fs.RangeOption); ok {
|
|
||||||
return nil, nil, errors.New("partial downloads are not supported while exporting Google Documents")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if o.v2Download {
|
|
||||||
f := o.fs
|
|
||||||
var v2File *drive_v2.File
|
|
||||||
err = f.pacer.Call(func() (bool, error) {
|
|
||||||
v2File, err = o.fs.v2Svc.Files.Get(o.id).
|
|
||||||
Fields("downloadUrl").
|
|
||||||
SupportsTeamDrives(f.isTeamDrive).
|
|
||||||
Do()
|
|
||||||
return shouldRetry(err)
|
|
||||||
})
|
|
||||||
if err == nil {
|
|
||||||
fs.Debugf(o, "Using v2 download: %v", v2File.DownloadUrl)
|
|
||||||
o.url = v2File.DownloadUrl
|
|
||||||
o.v2Download = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
req, err = http.NewRequest(method, o.url, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return req, nil, err
|
return req, nil, err
|
||||||
}
|
}
|
||||||
|
@ -2174,17 +2202,18 @@ func (o *Object) httpResponse(method string, options []fs.OpenOption) (req *http
|
||||||
return req, res, nil
|
return req, res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// openFile represents an Object open for reading
|
// openDocumentFile represents an documentObject open for reading.
|
||||||
type openFile struct {
|
// Updates the object size after read successfully.
|
||||||
o *Object // Object we are reading for
|
type openDocumentFile struct {
|
||||||
in io.ReadCloser // reading from here
|
o *documentObject // Object we are reading for
|
||||||
bytes int64 // number of bytes read on this connection
|
in io.ReadCloser // reading from here
|
||||||
eof bool // whether we have read end of file
|
bytes int64 // number of bytes read on this connection
|
||||||
errored bool // whether we have encountered an error during reading
|
eof bool // whether we have read end of file
|
||||||
|
errored bool // whether we have encountered an error during reading
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read bytes from the object - see io.Reader
|
// Read bytes from the object - see io.Reader
|
||||||
func (file *openFile) Read(p []byte) (n int, err error) {
|
func (file *openDocumentFile) Read(p []byte) (n int, err error) {
|
||||||
n, err = file.in.Read(p)
|
n, err = file.in.Read(p)
|
||||||
file.bytes += int64(n)
|
file.bytes += int64(n)
|
||||||
if err != nil && err != io.EOF {
|
if err != nil && err != io.EOF {
|
||||||
|
@ -2197,7 +2226,7 @@ func (file *openFile) Read(p []byte) (n int, err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close the object and update bytes read
|
// Close the object and update bytes read
|
||||||
func (file *openFile) Close() (err error) {
|
func (file *openDocumentFile) Close() (err error) {
|
||||||
// If end of file, update bytes read
|
// If end of file, update bytes read
|
||||||
if file.eof && !file.errored {
|
if file.eof && !file.errored {
|
||||||
fs.Debugf(file.o, "Updating size of doc after download to %v", file.bytes)
|
fs.Debugf(file.o, "Updating size of doc after download to %v", file.bytes)
|
||||||
|
@ -2207,7 +2236,7 @@ func (file *openFile) Close() (err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check it satisfies the interfaces
|
// Check it satisfies the interfaces
|
||||||
var _ io.ReadCloser = &openFile{}
|
var _ io.ReadCloser = (*openDocumentFile)(nil)
|
||||||
|
|
||||||
// Checks to see if err is a googleapi.Error with of type what
|
// Checks to see if err is a googleapi.Error with of type what
|
||||||
func isGoogleError(err error, what string) bool {
|
func isGoogleError(err error, what string) bool {
|
||||||
|
@ -2221,20 +2250,20 @@ func isGoogleError(err error, what string) bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Open an object for read
|
// open a url for reading
|
||||||
func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
func (o *baseObject) open(url string, options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
_, res, err := o.httpResponse("GET", options)
|
_, res, err := o.httpResponse(url, "GET", options)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if isGoogleError(err, "cannotDownloadAbusiveFile") {
|
if isGoogleError(err, "cannotDownloadAbusiveFile") {
|
||||||
if o.fs.opt.AcknowledgeAbuse {
|
if o.fs.opt.AcknowledgeAbuse {
|
||||||
// Retry acknowledging abuse
|
// Retry acknowledging abuse
|
||||||
if strings.ContainsRune(o.url, '?') {
|
if strings.ContainsRune(url, '?') {
|
||||||
o.url += "&"
|
url += "&"
|
||||||
} else {
|
} else {
|
||||||
o.url += "?"
|
url += "?"
|
||||||
}
|
}
|
||||||
o.url += "acknowledgeAbuse=true"
|
url += "acknowledgeAbuse=true"
|
||||||
_, res, err = o.httpResponse("GET", options)
|
_, res, err = o.httpResponse(url, "GET", options)
|
||||||
} else {
|
} else {
|
||||||
err = errors.Wrap(err, "Use the --drive-acknowledge-abuse flag to download this file")
|
err = errors.Wrap(err, "Use the --drive-acknowledge-abuse flag to download this file")
|
||||||
}
|
}
|
||||||
|
@ -2243,77 +2272,132 @@ func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
return nil, errors.Wrap(err, "open file failed")
|
return nil, errors.Wrap(err, "open file failed")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If it is a document, update the size with what we are
|
|
||||||
// reading as it can change from the HEAD in the listing to
|
|
||||||
// this GET. This stops rclone marking the transfer as
|
|
||||||
// corrupted.
|
|
||||||
if o.documentMimeType != "" {
|
|
||||||
return &openFile{o: o, in: res.Body}, nil
|
|
||||||
}
|
|
||||||
return res.Body, nil
|
return res.Body, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Open an object for read
|
||||||
|
func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
|
if o.v2Download {
|
||||||
|
var v2File *drive_v2.File
|
||||||
|
err = o.fs.pacer.Call(func() (bool, error) {
|
||||||
|
v2File, err = o.fs.v2Svc.Files.Get(o.id).
|
||||||
|
Fields("downloadUrl").
|
||||||
|
SupportsTeamDrives(o.fs.isTeamDrive).
|
||||||
|
Do()
|
||||||
|
return shouldRetry(err)
|
||||||
|
})
|
||||||
|
if err == nil {
|
||||||
|
fs.Debugf(o, "Using v2 download: %v", v2File.DownloadUrl)
|
||||||
|
o.url = v2File.DownloadUrl
|
||||||
|
o.v2Download = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return o.baseObject.open(o.url, options...)
|
||||||
|
}
|
||||||
|
func (o *documentObject) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
|
// Update the size with what we are reading as it can change from
|
||||||
|
// the HEAD in the listing to this GET. This stops rclone marking
|
||||||
|
// the transfer as corrupted.
|
||||||
|
for _, o := range options {
|
||||||
|
// https://developers.google.com/drive/v3/web/manage-downloads#partial_download
|
||||||
|
if _, ok := o.(*fs.RangeOption); ok {
|
||||||
|
return nil, errors.New("partial downloads are not supported while exporting Google Documents")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
in, err = o.baseObject.open(o.url, options...)
|
||||||
|
if in != nil {
|
||||||
|
in = &openDocumentFile{o: o, in: in}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *baseObject) update(updateInfo *drive.File, uploadMimeType string, in io.Reader,
|
||||||
|
src fs.ObjectInfo) (info *drive.File, err error) {
|
||||||
|
// Make the API request to upload metadata and file data.
|
||||||
|
size := src.Size()
|
||||||
|
if size == 0 || size < int64(o.fs.opt.UploadCutoff) {
|
||||||
|
// Don't retry, return a retry error instead
|
||||||
|
err = o.fs.pacer.CallNoRetry(func() (bool, error) {
|
||||||
|
info, err = o.fs.svc.Files.Update(o.id, updateInfo).
|
||||||
|
Media(in, googleapi.ContentType(uploadMimeType)).
|
||||||
|
Fields(googleapi.Field(partialFields)).
|
||||||
|
SupportsTeamDrives(o.fs.isTeamDrive).
|
||||||
|
KeepRevisionForever(o.fs.opt.KeepRevisionForever).
|
||||||
|
Do()
|
||||||
|
return shouldRetry(err)
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// Upload the file in chunks
|
||||||
|
return o.fs.Upload(in, size, uploadMimeType, o.id, updateInfo, o.remote)
|
||||||
|
}
|
||||||
|
|
||||||
// Update the already existing object
|
// Update the already existing object
|
||||||
//
|
//
|
||||||
// Copy the reader into the object updating modTime and size
|
// Copy the reader into the object updating modTime and size
|
||||||
//
|
//
|
||||||
// The new object may have been created if an error is returned
|
// The new object may have been created if an error is returned
|
||||||
func (o *Object) Update(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) error {
|
func (o *Object) Update(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) error {
|
||||||
size := src.Size()
|
srcMimeType := fs.MimeType(src)
|
||||||
modTime := src.ModTime()
|
updateInfo := &drive.File{
|
||||||
|
MimeType: srcMimeType,
|
||||||
|
ModifiedTime: src.ModTime().Format(timeFormatOut),
|
||||||
|
}
|
||||||
|
info, err := o.baseObject.update(updateInfo, srcMimeType, in, src)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
newO, err := o.fs.newObjectWithInfo(src.Remote(), info)
|
||||||
|
switch newO := newO.(type) {
|
||||||
|
case *Object:
|
||||||
|
*o = *newO
|
||||||
|
default:
|
||||||
|
return errors.New("object type changed by update")
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func (o *documentObject) Update(in io.Reader, src fs.ObjectInfo, options ...fs.OpenOption) error {
|
||||||
srcMimeType := fs.MimeType(src)
|
srcMimeType := fs.MimeType(src)
|
||||||
importMimeType := ""
|
importMimeType := ""
|
||||||
updateInfo := &drive.File{
|
updateInfo := &drive.File{
|
||||||
MimeType: srcMimeType,
|
MimeType: srcMimeType,
|
||||||
ModifiedTime: modTime.Format(timeFormatOut),
|
ModifiedTime: src.ModTime().Format(timeFormatOut),
|
||||||
}
|
}
|
||||||
|
|
||||||
if o.fs.importMimeTypes != nil && !o.fs.opt.SkipGdocs {
|
if o.fs.importMimeTypes == nil || o.fs.opt.SkipGdocs {
|
||||||
importMimeType = o.fs.findImportFormat(updateInfo.MimeType)
|
return errors.Errorf("can't update google document type without --drive-import-formats")
|
||||||
if importMimeType != "" {
|
}
|
||||||
// FIXME: check importMimeType against original object MIME type
|
importMimeType = o.fs.findImportFormat(updateInfo.MimeType)
|
||||||
// if importMimeType != o.mimeType {
|
if importMimeType == "" {
|
||||||
// return errors.Errorf("can't change google document type (o: %q, src: %q, import: %q)", o.mimeType, srcMimeType, importMimeType)
|
return errors.Errorf("no import format found for %q", srcMimeType)
|
||||||
// }
|
}
|
||||||
updateInfo.MimeType = importMimeType
|
if importMimeType != o.documentMimeType {
|
||||||
}
|
return errors.Errorf("can't change google document type (o: %q, src: %q, import: %q)", o.documentMimeType, srcMimeType, importMimeType)
|
||||||
|
}
|
||||||
|
updateInfo.MimeType = importMimeType
|
||||||
|
|
||||||
|
info, err := o.baseObject.update(updateInfo, srcMimeType, in, src)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make the API request to upload metadata and file data.
|
remote := src.Remote()
|
||||||
var err error
|
remote = remote[:len(remote)-o.extLen]
|
||||||
var info *drive.File
|
|
||||||
if size == 0 || size < int64(o.fs.opt.UploadCutoff) {
|
newO, err := o.fs.newObjectWithInfo(remote, info)
|
||||||
// Don't retry, return a retry error instead
|
switch newO := newO.(type) {
|
||||||
err = o.fs.pacer.CallNoRetry(func() (bool, error) {
|
case *documentObject:
|
||||||
info, err = o.fs.svc.Files.Update(o.id, updateInfo).
|
*o = *newO
|
||||||
Media(in, googleapi.ContentType(srcMimeType)).
|
default:
|
||||||
Fields(googleapi.Field(partialFields)).
|
return errors.New("object type changed by update")
|
||||||
SupportsTeamDrives(o.fs.isTeamDrive).
|
|
||||||
KeepRevisionForever(o.fs.opt.KeepRevisionForever).
|
|
||||||
Do()
|
|
||||||
return shouldRetry(err)
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Upload the file in chunks
|
|
||||||
info, err = o.fs.Upload(in, size, srcMimeType, o.id, updateInfo, o.remote)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
o.setMetaData(info)
|
|
||||||
if importMimeType != "" {
|
|
||||||
extension, exportMimeType, _ := o.fs.findExportFormatByMimeType(importMimeType)
|
|
||||||
o.setGdocsMetaData(info, extension, exportMimeType)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove an object
|
// Remove an object
|
||||||
func (o *Object) Remove() error {
|
func (o *baseObject) Remove() error {
|
||||||
var err error
|
var err error
|
||||||
err = o.fs.pacer.Call(func() (bool, error) {
|
err = o.fs.pacer.Call(func() (bool, error) {
|
||||||
if o.fs.opt.UseTrash {
|
if o.fs.opt.UseTrash {
|
||||||
|
@ -2336,17 +2420,12 @@ func (o *Object) Remove() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// MimeType of an Object if known, "" otherwise
|
// MimeType of an Object if known, "" otherwise
|
||||||
func (o *Object) MimeType() string {
|
func (o *baseObject) MimeType() string {
|
||||||
err := o.readMetaData()
|
|
||||||
if err != nil {
|
|
||||||
fs.Debugf(o, "Failed to read metadata: %v", err)
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return o.mimeType
|
return o.mimeType
|
||||||
}
|
}
|
||||||
|
|
||||||
// ID returns the ID of the Object if known, or "" if not
|
// ID returns the ID of the Object if known, or "" if not
|
||||||
func (o *Object) ID() string {
|
func (o *baseObject) ID() string {
|
||||||
return o.id
|
return o.id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2369,4 +2448,7 @@ var (
|
||||||
_ fs.Object = (*Object)(nil)
|
_ fs.Object = (*Object)(nil)
|
||||||
_ fs.MimeTyper = (*Object)(nil)
|
_ fs.MimeTyper = (*Object)(nil)
|
||||||
_ fs.IDer = (*Object)(nil)
|
_ fs.IDer = (*Object)(nil)
|
||||||
|
_ fs.Object = (*documentObject)(nil)
|
||||||
|
_ fs.MimeTyper = (*documentObject)(nil)
|
||||||
|
_ fs.IDer = (*documentObject)(nil)
|
||||||
)
|
)
|
||||||
|
|
Loading…
Reference in a new issue