mirror of https://github.com/Xhofe/alist
170 lines
4.6 KiB
Go
170 lines
4.6 KiB
Go
package google_drive
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"github.com/alist-org/alist/v3/drivers/base"
|
|
"github.com/alist-org/alist/v3/internal/driver"
|
|
"github.com/alist-org/alist/v3/internal/errs"
|
|
"github.com/alist-org/alist/v3/internal/model"
|
|
"github.com/alist-org/alist/v3/pkg/utils"
|
|
"github.com/go-resty/resty/v2"
|
|
)
|
|
|
|
type GoogleDrive struct {
|
|
model.Storage
|
|
Addition
|
|
AccessToken string
|
|
ServiceAccountFile int
|
|
ServiceAccountFileList []string
|
|
}
|
|
|
|
func (d *GoogleDrive) Config() driver.Config {
|
|
return config
|
|
}
|
|
|
|
func (d *GoogleDrive) GetAddition() driver.Additional {
|
|
return &d.Addition
|
|
}
|
|
|
|
func (d *GoogleDrive) Init(ctx context.Context) error {
|
|
if d.ChunkSize == 0 {
|
|
d.ChunkSize = 5
|
|
}
|
|
return d.refreshToken()
|
|
}
|
|
|
|
func (d *GoogleDrive) Drop(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (d *GoogleDrive) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) {
|
|
files, err := d.getFiles(dir.GetID())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return utils.SliceConvert(files, func(src File) (model.Obj, error) {
|
|
return fileToObj(src), nil
|
|
})
|
|
}
|
|
|
|
func (d *GoogleDrive) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) {
|
|
url := fmt.Sprintf("https://www.googleapis.com/drive/v3/files/%s?includeItemsFromAllDrives=true&supportsAllDrives=true", file.GetID())
|
|
_, err := d.request(url, http.MethodGet, nil, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
link := model.Link{
|
|
URL: url + "&alt=media&acknowledgeAbuse=true",
|
|
Header: http.Header{
|
|
"Authorization": []string{"Bearer " + d.AccessToken},
|
|
},
|
|
}
|
|
return &link, nil
|
|
}
|
|
|
|
func (d *GoogleDrive) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) error {
|
|
data := base.Json{
|
|
"name": dirName,
|
|
"parents": []string{parentDir.GetID()},
|
|
"mimeType": "application/vnd.google-apps.folder",
|
|
}
|
|
_, err := d.request("https://www.googleapis.com/drive/v3/files", http.MethodPost, func(req *resty.Request) {
|
|
req.SetBody(data)
|
|
}, nil)
|
|
return err
|
|
}
|
|
|
|
func (d *GoogleDrive) Move(ctx context.Context, srcObj, dstDir model.Obj) error {
|
|
query := map[string]string{
|
|
"addParents": dstDir.GetID(),
|
|
"removeParents": "root",
|
|
}
|
|
url := "https://www.googleapis.com/drive/v3/files/" + srcObj.GetID()
|
|
_, err := d.request(url, http.MethodPatch, func(req *resty.Request) {
|
|
req.SetQueryParams(query)
|
|
}, nil)
|
|
return err
|
|
}
|
|
|
|
func (d *GoogleDrive) Rename(ctx context.Context, srcObj model.Obj, newName string) error {
|
|
data := base.Json{
|
|
"name": newName,
|
|
}
|
|
url := "https://www.googleapis.com/drive/v3/files/" + srcObj.GetID()
|
|
_, err := d.request(url, http.MethodPatch, func(req *resty.Request) {
|
|
req.SetBody(data)
|
|
}, nil)
|
|
return err
|
|
}
|
|
|
|
func (d *GoogleDrive) Copy(ctx context.Context, srcObj, dstDir model.Obj) error {
|
|
return errs.NotSupport
|
|
}
|
|
|
|
func (d *GoogleDrive) Remove(ctx context.Context, obj model.Obj) error {
|
|
url := "https://www.googleapis.com/drive/v3/files/" + obj.GetID()
|
|
_, err := d.request(url, http.MethodDelete, nil, nil)
|
|
return err
|
|
}
|
|
|
|
func (d *GoogleDrive) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) error {
|
|
obj := stream.GetExist()
|
|
var (
|
|
e Error
|
|
url string
|
|
data base.Json
|
|
res *resty.Response
|
|
err error
|
|
)
|
|
if obj != nil {
|
|
url = fmt.Sprintf("https://www.googleapis.com/upload/drive/v3/files/%s?uploadType=resumable&supportsAllDrives=true", obj.GetID())
|
|
data = base.Json{}
|
|
} else {
|
|
data = base.Json{
|
|
"name": stream.GetName(),
|
|
"parents": []string{dstDir.GetID()},
|
|
}
|
|
url = "https://www.googleapis.com/upload/drive/v3/files?uploadType=resumable&supportsAllDrives=true"
|
|
}
|
|
req := base.NoRedirectClient.R().
|
|
SetHeaders(map[string]string{
|
|
"Authorization": "Bearer " + d.AccessToken,
|
|
"X-Upload-Content-Type": stream.GetMimetype(),
|
|
"X-Upload-Content-Length": strconv.FormatInt(stream.GetSize(), 10),
|
|
}).
|
|
SetError(&e).SetBody(data).SetContext(ctx)
|
|
if obj != nil {
|
|
res, err = req.Patch(url)
|
|
} else {
|
|
res, err = req.Post(url)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if e.Error.Code != 0 {
|
|
if e.Error.Code == 401 {
|
|
err = d.refreshToken()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return d.Put(ctx, dstDir, stream, up)
|
|
}
|
|
return fmt.Errorf("%s: %v", e.Error.Message, e.Error.Errors)
|
|
}
|
|
putUrl := res.Header().Get("location")
|
|
if stream.GetSize() < d.ChunkSize*1024*1024 {
|
|
_, err = d.request(putUrl, http.MethodPut, func(req *resty.Request) {
|
|
req.SetHeader("Content-Length", strconv.FormatInt(stream.GetSize(), 10)).SetBody(stream)
|
|
}, nil)
|
|
} else {
|
|
err = d.chunkUpload(ctx, stream, putUrl)
|
|
}
|
|
return err
|
|
}
|
|
|
|
var _ driver.Driver = (*GoogleDrive)(nil)
|