* feat(doubao): implement List() * feat(doubao): implement Link() * feat(doubao): implement MakeDir() * refactor(doubao): add type Object to store key * feat(doubao): implement Move() * feat(doubao): implement Rename() * feat(doubao): implement Remove()
This commit is contained in:
parent
1335f80362
commit
5668e4a4ea
@ -22,6 +22,7 @@ import (
|
|||||||
_ "github.com/alist-org/alist/v3/drivers/chaoxing"
|
_ "github.com/alist-org/alist/v3/drivers/chaoxing"
|
||||||
_ "github.com/alist-org/alist/v3/drivers/cloudreve"
|
_ "github.com/alist-org/alist/v3/drivers/cloudreve"
|
||||||
_ "github.com/alist-org/alist/v3/drivers/crypt"
|
_ "github.com/alist-org/alist/v3/drivers/crypt"
|
||||||
|
_ "github.com/alist-org/alist/v3/drivers/doubao"
|
||||||
_ "github.com/alist-org/alist/v3/drivers/dropbox"
|
_ "github.com/alist-org/alist/v3/drivers/dropbox"
|
||||||
_ "github.com/alist-org/alist/v3/drivers/febbox"
|
_ "github.com/alist-org/alist/v3/drivers/febbox"
|
||||||
_ "github.com/alist-org/alist/v3/drivers/ftp"
|
_ "github.com/alist-org/alist/v3/drivers/ftp"
|
||||||
|
174
drivers/doubao/driver.go
Normal file
174
drivers/doubao/driver.go
Normal file
@ -0,0 +1,174 @@
|
|||||||
|
package doubao
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"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/go-resty/resty/v2"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Doubao struct {
|
||||||
|
model.Storage
|
||||||
|
Addition
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Config() driver.Config {
|
||||||
|
return config
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) GetAddition() driver.Additional {
|
||||||
|
return &d.Addition
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Init(ctx context.Context) error {
|
||||||
|
// TODO login / refresh token
|
||||||
|
//op.MustSaveDriverStorage(d)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Drop(ctx context.Context) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) {
|
||||||
|
var files []model.Obj
|
||||||
|
var r NodeInfoResp
|
||||||
|
_, err := d.request("/samantha/aispace/node_info", "POST", func(req *resty.Request) {
|
||||||
|
req.SetBody(base.Json{
|
||||||
|
"node_id": dir.GetID(),
|
||||||
|
"need_full_path": false,
|
||||||
|
})
|
||||||
|
}, &r)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, child := range r.Data.Children {
|
||||||
|
files = append(files, &Object{
|
||||||
|
Object: model.Object{
|
||||||
|
ID: child.ID,
|
||||||
|
Path: child.ParentID,
|
||||||
|
Name: child.Name,
|
||||||
|
Size: int64(child.Size),
|
||||||
|
Modified: time.Unix(int64(child.UpdateTime), 0),
|
||||||
|
Ctime: time.Unix(int64(child.CreateTime), 0),
|
||||||
|
IsFolder: child.NodeType == 1,
|
||||||
|
},
|
||||||
|
Key: child.Key,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return files, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) {
|
||||||
|
if u, ok := file.(*Object); ok {
|
||||||
|
var r GetFileUrlResp
|
||||||
|
_, err := d.request("/alice/message/get_file_url", "POST", func(req *resty.Request) {
|
||||||
|
req.SetBody(base.Json{
|
||||||
|
"uris": []string{u.Key},
|
||||||
|
"type": "file",
|
||||||
|
})
|
||||||
|
}, &r)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &model.Link{
|
||||||
|
URL: r.Data.FileUrls[0].MainURL,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
return nil, errors.New("can't convert obj to URL")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) error {
|
||||||
|
var r UploadNodeResp
|
||||||
|
_, err := d.request("/samantha/aispace/upload_node", "POST", func(req *resty.Request) {
|
||||||
|
req.SetBody(base.Json{
|
||||||
|
"node_list": []base.Json{
|
||||||
|
{
|
||||||
|
"local_id": uuid.New().String(),
|
||||||
|
"name": dirName,
|
||||||
|
"parent_id": parentDir.GetID(),
|
||||||
|
"node_type": 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}, &r)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Move(ctx context.Context, srcObj, dstDir model.Obj) error {
|
||||||
|
var r UploadNodeResp
|
||||||
|
_, err := d.request("/samantha/aispace/move_node", "POST", func(req *resty.Request) {
|
||||||
|
req.SetBody(base.Json{
|
||||||
|
"node_list": []base.Json{
|
||||||
|
{"id": srcObj.GetID()},
|
||||||
|
},
|
||||||
|
"current_parent_id": srcObj.GetPath(),
|
||||||
|
"target_parent_id": dstDir.GetID(),
|
||||||
|
})
|
||||||
|
}, &r)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Rename(ctx context.Context, srcObj model.Obj, newName string) error {
|
||||||
|
var r BaseResp
|
||||||
|
_, err := d.request("/samantha/aispace/rename_node", "POST", func(req *resty.Request) {
|
||||||
|
req.SetBody(base.Json{
|
||||||
|
"node_id": srcObj.GetID(),
|
||||||
|
"node_name": newName,
|
||||||
|
})
|
||||||
|
}, &r)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Copy(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) {
|
||||||
|
// TODO copy obj, optional
|
||||||
|
return nil, errs.NotImplement
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Remove(ctx context.Context, obj model.Obj) error {
|
||||||
|
var r BaseResp
|
||||||
|
_, err := d.request("/samantha/aispace/delete_node", "POST", func(req *resty.Request) {
|
||||||
|
req.SetBody(base.Json{"node_list": []base.Json{{"id": obj.GetID()}}})
|
||||||
|
}, &r)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Put(ctx context.Context, dstDir model.Obj, file model.FileStreamer, up driver.UpdateProgress) (model.Obj, error) {
|
||||||
|
// TODO upload file, optional
|
||||||
|
return nil, errs.NotImplement
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) GetArchiveMeta(ctx context.Context, obj model.Obj, args model.ArchiveArgs) (model.ArchiveMeta, error) {
|
||||||
|
// TODO get archive file meta-info, return errs.NotImplement to use an internal archive tool, optional
|
||||||
|
return nil, errs.NotImplement
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) ListArchive(ctx context.Context, obj model.Obj, args model.ArchiveInnerArgs) ([]model.Obj, error) {
|
||||||
|
// TODO list args.InnerPath in the archive obj, return errs.NotImplement to use an internal archive tool, optional
|
||||||
|
return nil, errs.NotImplement
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) Extract(ctx context.Context, obj model.Obj, args model.ArchiveInnerArgs) (*model.Link, error) {
|
||||||
|
// TODO return link of file args.InnerPath in the archive obj, return errs.NotImplement to use an internal archive tool, optional
|
||||||
|
return nil, errs.NotImplement
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Doubao) ArchiveDecompress(ctx context.Context, srcObj, dstDir model.Obj, args model.ArchiveDecompressArgs) ([]model.Obj, error) {
|
||||||
|
// TODO extract args.InnerPath path in the archive srcObj to the dstDir location, optional
|
||||||
|
// a folder with the same name as the archive file needs to be created to store the extracted results if args.PutIntoNewDir
|
||||||
|
// return errs.NotImplement to use an internal archive tool
|
||||||
|
return nil, errs.NotImplement
|
||||||
|
}
|
||||||
|
|
||||||
|
//func (d *Doubao) Other(ctx context.Context, args model.OtherArgs) (interface{}, error) {
|
||||||
|
// return nil, errs.NotSupport
|
||||||
|
//}
|
||||||
|
|
||||||
|
var _ driver.Driver = (*Doubao)(nil)
|
34
drivers/doubao/meta.go
Normal file
34
drivers/doubao/meta.go
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
package doubao
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/alist-org/alist/v3/internal/driver"
|
||||||
|
"github.com/alist-org/alist/v3/internal/op"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Addition struct {
|
||||||
|
// Usually one of two
|
||||||
|
// driver.RootPath
|
||||||
|
driver.RootID
|
||||||
|
// define other
|
||||||
|
Cookie string `json:"cookie" type:"text"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var config = driver.Config{
|
||||||
|
Name: "Doubao",
|
||||||
|
LocalSort: true,
|
||||||
|
OnlyLocal: false,
|
||||||
|
OnlyProxy: false,
|
||||||
|
NoCache: false,
|
||||||
|
NoUpload: true,
|
||||||
|
NeedMs: false,
|
||||||
|
DefaultRoot: "0",
|
||||||
|
CheckStatus: false,
|
||||||
|
Alert: "",
|
||||||
|
NoOverwriteUpload: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
op.RegisterDriver(func() driver.Driver {
|
||||||
|
return &Doubao{}
|
||||||
|
})
|
||||||
|
}
|
64
drivers/doubao/types.go
Normal file
64
drivers/doubao/types.go
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
package doubao
|
||||||
|
|
||||||
|
import "github.com/alist-org/alist/v3/internal/model"
|
||||||
|
|
||||||
|
type BaseResp struct {
|
||||||
|
Code int `json:"code"`
|
||||||
|
Msg string `json:"msg"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type NodeInfoResp struct {
|
||||||
|
BaseResp
|
||||||
|
Data struct {
|
||||||
|
NodeInfo NodeInfo `json:"node_info"`
|
||||||
|
Children []NodeInfo `json:"children"`
|
||||||
|
NextCursor string `json:"next_cursor"`
|
||||||
|
HasMore bool `json:"has_more"`
|
||||||
|
} `json:"data"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type NodeInfo struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Key string `json:"key"`
|
||||||
|
NodeType int `json:"node_type"` // 0: 文件, 1: 文件夹
|
||||||
|
Size int `json:"size"`
|
||||||
|
Source int `json:"source"`
|
||||||
|
NameReviewStatus int `json:"name_review_status"`
|
||||||
|
ContentReviewStatus int `json:"content_review_status"`
|
||||||
|
RiskReviewStatus int `json:"risk_review_status"`
|
||||||
|
ConversationID string `json:"conversation_id"`
|
||||||
|
ParentID string `json:"parent_id"`
|
||||||
|
CreateTime int `json:"create_time"`
|
||||||
|
UpdateTime int `json:"update_time"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GetFileUrlResp struct {
|
||||||
|
BaseResp
|
||||||
|
Data struct {
|
||||||
|
FileUrls []struct {
|
||||||
|
URI string `json:"uri"`
|
||||||
|
MainURL string `json:"main_url"`
|
||||||
|
BackURL string `json:"back_url"`
|
||||||
|
} `json:"file_urls"`
|
||||||
|
} `json:"data"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UploadNodeResp struct {
|
||||||
|
BaseResp
|
||||||
|
Data struct {
|
||||||
|
NodeList []struct {
|
||||||
|
LocalID string `json:"local_id"`
|
||||||
|
ID string `json:"id"`
|
||||||
|
ParentID string `json:"parent_id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Key string `json:"key"`
|
||||||
|
NodeType int `json:"node_type"` // 0: 文件, 1: 文件夹
|
||||||
|
} `json:"node_list"`
|
||||||
|
} `json:"data"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Object struct {
|
||||||
|
model.Object
|
||||||
|
Key string
|
||||||
|
}
|
38
drivers/doubao/util.go
Normal file
38
drivers/doubao/util.go
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
package doubao
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/alist-org/alist/v3/drivers/base"
|
||||||
|
"github.com/alist-org/alist/v3/pkg/utils"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
// do others that not defined in Driver interface
|
||||||
|
func (d *Doubao) request(path string, method string, callback base.ReqCallback, resp interface{}) ([]byte, error) {
|
||||||
|
url := "https://www.doubao.com" + path
|
||||||
|
req := base.RestyClient.R()
|
||||||
|
req.SetHeader("Cookie", d.Cookie)
|
||||||
|
if callback != nil {
|
||||||
|
callback(req)
|
||||||
|
}
|
||||||
|
var r BaseResp
|
||||||
|
req.SetResult(&r)
|
||||||
|
res, err := req.Execute(method, url)
|
||||||
|
log.Debugln(res.String())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// 业务状态码检查(优先于HTTP状态码)
|
||||||
|
if r.Code != 0 {
|
||||||
|
return res.Body(), errors.New(r.Msg)
|
||||||
|
}
|
||||||
|
if resp != nil {
|
||||||
|
err = utils.Json.Unmarshal(res.Body(), resp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res.Body(), nil
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user