| | package cloudreve |
| |
|
| | import ( |
| | "bytes" |
| | "context" |
| | "encoding/base64" |
| | "errors" |
| | "fmt" |
| | "io" |
| | "net/http" |
| | "strconv" |
| | "strings" |
| |
|
| | "github.com/alist-org/alist/v3/drivers/base" |
| | "github.com/alist-org/alist/v3/internal/conf" |
| | "github.com/alist-org/alist/v3/internal/driver" |
| | "github.com/alist-org/alist/v3/internal/model" |
| | "github.com/alist-org/alist/v3/internal/setting" |
| | "github.com/alist-org/alist/v3/pkg/cookie" |
| | "github.com/alist-org/alist/v3/pkg/utils" |
| | "github.com/go-resty/resty/v2" |
| | json "github.com/json-iterator/go" |
| | jsoniter "github.com/json-iterator/go" |
| | ) |
| |
|
| | |
| |
|
| | const loginPath = "/user/session" |
| |
|
| | func (d *Cloudreve) request(method string, path string, callback base.ReqCallback, out interface{}) error { |
| | u := d.Address + "/api/v3" + path |
| | ua := d.CustomUA |
| | if ua == "" { |
| | ua = base.UserAgent |
| | } |
| | req := base.RestyClient.R() |
| | req.SetHeaders(map[string]string{ |
| | "Cookie": "cloudreve-session=" + d.Cookie, |
| | "Accept": "application/json, text/plain, */*", |
| | "User-Agent": ua, |
| | }) |
| |
|
| | var r Resp |
| | req.SetResult(&r) |
| |
|
| | if callback != nil { |
| | callback(req) |
| | } |
| |
|
| | resp, err := req.Execute(method, u) |
| | if err != nil { |
| | return err |
| | } |
| | if !resp.IsSuccess() { |
| | return errors.New(resp.String()) |
| | } |
| |
|
| | if r.Code != 0 { |
| |
|
| | |
| | if r.Code == http.StatusUnauthorized && path != loginPath { |
| | if d.Username != "" && d.Password != "" { |
| | err = d.login() |
| | if err != nil { |
| | return err |
| | } |
| | return d.request(method, path, callback, out) |
| | } |
| | } |
| |
|
| | return errors.New(r.Msg) |
| | } |
| | sess := cookie.GetCookie(resp.Cookies(), "cloudreve-session") |
| | if sess != nil { |
| | d.Cookie = sess.Value |
| | } |
| | if out != nil && r.Data != nil { |
| | var marshal []byte |
| | marshal, err = json.Marshal(r.Data) |
| | if err != nil { |
| | return err |
| | } |
| | err = json.Unmarshal(marshal, out) |
| | if err != nil { |
| | return err |
| | } |
| | } |
| |
|
| | return nil |
| | } |
| |
|
| | func (d *Cloudreve) login() error { |
| | var siteConfig Config |
| | err := d.request(http.MethodGet, "/site/config", nil, &siteConfig) |
| | if err != nil { |
| | return err |
| | } |
| | for i := 0; i < 5; i++ { |
| | err = d.doLogin(siteConfig.LoginCaptcha) |
| | if err == nil { |
| | break |
| | } |
| | if err != nil && err.Error() != "CAPTCHA not match." { |
| | break |
| | } |
| | } |
| | return err |
| | } |
| |
|
| | func (d *Cloudreve) doLogin(needCaptcha bool) error { |
| | var captchaCode string |
| | var err error |
| | if needCaptcha { |
| | var captcha string |
| | err = d.request(http.MethodGet, "/site/captcha", nil, &captcha) |
| | if err != nil { |
| | return err |
| | } |
| | if len(captcha) == 0 { |
| | return errors.New("can not get captcha") |
| | } |
| | i := strings.Index(captcha, ",") |
| | dec := base64.NewDecoder(base64.StdEncoding, strings.NewReader(captcha[i+1:])) |
| | vRes, err := base.RestyClient.R().SetMultipartField( |
| | "image", "validateCode.png", "image/png", dec). |
| | Post(setting.GetStr(conf.OcrApi)) |
| | if err != nil { |
| | return err |
| | } |
| | if jsoniter.Get(vRes.Body(), "status").ToInt() != 200 { |
| | return errors.New("ocr error:" + jsoniter.Get(vRes.Body(), "msg").ToString()) |
| | } |
| | captchaCode = jsoniter.Get(vRes.Body(), "result").ToString() |
| | } |
| | var resp Resp |
| | err = d.request(http.MethodPost, loginPath, func(req *resty.Request) { |
| | req.SetBody(base.Json{ |
| | "username": d.Addition.Username, |
| | "Password": d.Addition.Password, |
| | "captchaCode": captchaCode, |
| | }) |
| | }, &resp) |
| | return err |
| | } |
| |
|
| | func convertSrc(obj model.Obj) map[string]interface{} { |
| | m := make(map[string]interface{}) |
| | var dirs []string |
| | var items []string |
| | if obj.IsDir() { |
| | dirs = append(dirs, obj.GetID()) |
| | } else { |
| | items = append(items, obj.GetID()) |
| | } |
| | m["dirs"] = dirs |
| | m["items"] = items |
| | return m |
| | } |
| |
|
| | func (d *Cloudreve) GetThumb(file Object) (model.Thumbnail, error) { |
| | if !d.Addition.EnableThumbAndFolderSize { |
| | return model.Thumbnail{}, nil |
| | } |
| | ua := d.CustomUA |
| | if ua == "" { |
| | ua = base.UserAgent |
| | } |
| | req := base.NoRedirectClient.R() |
| | req.SetHeaders(map[string]string{ |
| | "Cookie": "cloudreve-session=" + d.Cookie, |
| | "Accept": "image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8", |
| | "User-Agent": ua, |
| | }) |
| | resp, err := req.Execute(http.MethodGet, d.Address+"/api/v3/file/thumb/"+file.Id) |
| | if err != nil { |
| | return model.Thumbnail{}, err |
| | } |
| | return model.Thumbnail{ |
| | Thumbnail: resp.Header().Get("Location"), |
| | }, nil |
| | } |
| |
|
| | func (d *Cloudreve) upRemote(ctx context.Context, stream model.FileStreamer, u UploadInfo, up driver.UpdateProgress) error { |
| | uploadUrl := u.UploadURLs[0] |
| | credential := u.Credential |
| | var finish int64 = 0 |
| | var chunk int = 0 |
| | DEFAULT := int64(u.ChunkSize) |
| | for finish < stream.GetSize() { |
| | if utils.IsCanceled(ctx) { |
| | return ctx.Err() |
| | } |
| | utils.Log.Debugf("[Cloudreve-Remote] upload: %d", finish) |
| | var byteSize = DEFAULT |
| | left := stream.GetSize() - finish |
| | if left < DEFAULT { |
| | byteSize = left |
| | } |
| | byteData := make([]byte, byteSize) |
| | n, err := io.ReadFull(stream, byteData) |
| | utils.Log.Debug(err, n) |
| | if err != nil { |
| | return err |
| | } |
| | req, err := http.NewRequest("POST", uploadUrl+"?chunk="+strconv.Itoa(chunk), bytes.NewBuffer(byteData)) |
| | if err != nil { |
| | return err |
| | } |
| | req = req.WithContext(ctx) |
| | req.Header.Set("Content-Length", strconv.Itoa(int(byteSize))) |
| | req.Header.Set("Authorization", fmt.Sprint(credential)) |
| | finish += byteSize |
| | res, err := base.HttpClient.Do(req) |
| | if err != nil { |
| | return err |
| | } |
| | res.Body.Close() |
| | up(float64(finish) * 100 / float64(stream.GetSize())) |
| | chunk++ |
| | } |
| | return nil |
| | } |
| |
|
| | func (d *Cloudreve) upOneDrive(ctx context.Context, stream model.FileStreamer, u UploadInfo, up driver.UpdateProgress) error { |
| | uploadUrl := u.UploadURLs[0] |
| | var finish int64 = 0 |
| | DEFAULT := int64(u.ChunkSize) |
| | for finish < stream.GetSize() { |
| | if utils.IsCanceled(ctx) { |
| | return ctx.Err() |
| | } |
| | utils.Log.Debugf("[Cloudreve-OneDrive] upload: %d", finish) |
| | var byteSize = DEFAULT |
| | left := stream.GetSize() - finish |
| | if left < DEFAULT { |
| | byteSize = left |
| | } |
| | byteData := make([]byte, byteSize) |
| | n, err := io.ReadFull(stream, byteData) |
| | utils.Log.Debug(err, n) |
| | if err != nil { |
| | return err |
| | } |
| | req, err := http.NewRequest("PUT", uploadUrl, bytes.NewBuffer(byteData)) |
| | if err != nil { |
| | return err |
| | } |
| | req = req.WithContext(ctx) |
| | req.Header.Set("Content-Length", strconv.Itoa(int(byteSize))) |
| | req.Header.Set("Content-Range", fmt.Sprintf("bytes %d-%d/%d", finish, finish+byteSize-1, stream.GetSize())) |
| | finish += byteSize |
| | res, err := base.HttpClient.Do(req) |
| | if err != nil { |
| | return err |
| | } |
| | |
| | if res.StatusCode != 201 && res.StatusCode != 202 && res.StatusCode != 200 { |
| | data, _ := io.ReadAll(res.Body) |
| | res.Body.Close() |
| | return errors.New(string(data)) |
| | } |
| | res.Body.Close() |
| | up(float64(finish) * 100 / float64(stream.GetSize())) |
| | } |
| | |
| | err := d.request(http.MethodPost, "/callback/onedrive/finish/"+u.SessionID, func(req *resty.Request) { |
| | req.SetBody("{}") |
| | }, nil) |
| | if err != nil { |
| | return err |
| | } |
| | return nil |
| | } |
| |
|