mirror of https://github.com/Xhofe/alist
				
				
				
			
		
			
				
	
	
		
			285 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			285 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			Go
		
	
	
| package _123
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"hash/crc32"
 | |
| 	"math"
 | |
| 	"math/rand"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/alist-org/alist/v3/drivers/base"
 | |
| 	"github.com/alist-org/alist/v3/pkg/utils"
 | |
| 	"github.com/go-resty/resty/v2"
 | |
| 	jsoniter "github.com/json-iterator/go"
 | |
| 	log "github.com/sirupsen/logrus"
 | |
| )
 | |
| 
 | |
| // do others that not defined in Driver interface
 | |
| 
 | |
| const (
 | |
| 	Api              = "https://www.123pan.com/api"
 | |
| 	AApi             = "https://www.123pan.com/a/api"
 | |
| 	BApi             = "https://www.123pan.com/b/api"
 | |
| 	LoginApi         = "https://login.123pan.com/api"
 | |
| 	MainApi          = BApi
 | |
| 	SignIn           = LoginApi + "/user/sign_in"
 | |
| 	Logout           = MainApi + "/user/logout"
 | |
| 	UserInfo         = MainApi + "/user/info"
 | |
| 	FileList         = MainApi + "/file/list/new"
 | |
| 	DownloadInfo     = MainApi + "/file/download_info"
 | |
| 	Mkdir            = MainApi + "/file/upload_request"
 | |
| 	Move             = MainApi + "/file/mod_pid"
 | |
| 	Rename           = MainApi + "/file/rename"
 | |
| 	Trash            = MainApi + "/file/trash"
 | |
| 	UploadRequest    = MainApi + "/file/upload_request"
 | |
| 	UploadComplete   = MainApi + "/file/upload_complete"
 | |
| 	S3PreSignedUrls  = MainApi + "/file/s3_repare_upload_parts_batch"
 | |
| 	S3Auth           = MainApi + "/file/s3_upload_object/auth"
 | |
| 	UploadCompleteV2 = MainApi + "/file/upload_complete/v2"
 | |
| 	S3Complete       = MainApi + "/file/s3_complete_multipart_upload"
 | |
| 	//AuthKeySalt      = "8-8D$sL8gPjom7bk#cY"
 | |
| )
 | |
| 
 | |
| func signPath(path string, os string, version string) (k string, v string) {
 | |
| 	table := []byte{'a', 'd', 'e', 'f', 'g', 'h', 'l', 'm', 'y', 'i', 'j', 'n', 'o', 'p', 'k', 'q', 'r', 's', 't', 'u', 'b', 'c', 'v', 'w', 's', 'z'}
 | |
| 	random := fmt.Sprintf("%.f", math.Round(1e7*rand.Float64()))
 | |
| 	now := time.Now().In(time.FixedZone("CST", 8*3600))
 | |
| 	timestamp := fmt.Sprint(now.Unix())
 | |
| 	nowStr := []byte(now.Format("200601021504"))
 | |
| 	for i := 0; i < len(nowStr); i++ {
 | |
| 		nowStr[i] = table[nowStr[i]-48]
 | |
| 	}
 | |
| 	timeSign := fmt.Sprint(crc32.ChecksumIEEE(nowStr))
 | |
| 	data := strings.Join([]string{timestamp, random, path, os, version, timeSign}, "|")
 | |
| 	dataSign := fmt.Sprint(crc32.ChecksumIEEE([]byte(data)))
 | |
| 	return timeSign, strings.Join([]string{timestamp, random, dataSign}, "-")
 | |
| }
 | |
| 
 | |
| func GetApi(rawUrl string) string {
 | |
| 	u, _ := url.Parse(rawUrl)
 | |
| 	query := u.Query()
 | |
| 	query.Add(signPath(u.Path, "web", "3"))
 | |
| 	u.RawQuery = query.Encode()
 | |
| 	return u.String()
 | |
| }
 | |
| 
 | |
| //func GetApi(url string) string {
 | |
| //	vm := js.New()
 | |
| //	vm.Set("url", url[22:])
 | |
| //	r, err := vm.RunString(`
 | |
| //	(function(e){
 | |
| //        function A(t, e) {
 | |
| //            e = 1 < arguments.length && void 0 !== e ? e : 10;
 | |
| //            for (var n = function() {
 | |
| //                for (var t = [], e = 0; e < 256; e++) {
 | |
| //                    for (var n = e, r = 0; r < 8; r++)
 | |
| //                        n = 1 & n ? 3988292384 ^ n >>> 1 : n >>> 1;
 | |
| //                    t[e] = n
 | |
| //                }
 | |
| //                return t
 | |
| //            }(), r = function(t) {
 | |
| //                t = t.replace(/\\r\\n/g, "\\n");
 | |
| //                for (var e = "", n = 0; n < t.length; n++) {
 | |
| //                    var r = t.charCodeAt(n);
 | |
| //                    r < 128 ? e += String.fromCharCode(r) : e = 127 < r && r < 2048 ? (e += String.fromCharCode(r >> 6 | 192)) + String.fromCharCode(63 & r | 128) : (e = (e += String.fromCharCode(r >> 12 | 224)) + String.fromCharCode(r >> 6 & 63 | 128)) + String.fromCharCode(63 & r | 128)
 | |
| //                }
 | |
| //                return e
 | |
| //            }(t), a = -1, i = 0; i < r.length; i++)
 | |
| //                a = a >>> 8 ^ n[255 & (a ^ r.charCodeAt(i))];
 | |
| //            return (a = (-1 ^ a) >>> 0).toString(e)
 | |
| //        }
 | |
| //
 | |
| //	   function v(t) {
 | |
| //	       return (v = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function(t) {
 | |
| //	                   return typeof t
 | |
| //	               }
 | |
| //	               : function(t) {
 | |
| //	                   return t && "function" == typeof Symbol && t.constructor === Symbol && t !== Symbol.prototype ? "symbol" : typeof t
 | |
| //	               }
 | |
| //	       )(t)
 | |
| //	   }
 | |
| //
 | |
| //		for (p in a = Math.round(1e7 * Math.random()),
 | |
| //		o = Math.round(((new Date).getTime() + 60 * (new Date).getTimezoneOffset() * 1e3 + 288e5) / 1e3).toString(),
 | |
| //		m = ["a", "d", "e", "f", "g", "h", "l", "m", "y", "i", "j", "n", "o", "p", "k", "q", "r", "s", "t", "u", "b", "c", "v", "w", "s", "z"],
 | |
| //		u = function(t, e, n) {
 | |
| //			var r;
 | |
| //			n = 2 < arguments.length && void 0 !== n ? n : 8;
 | |
| //			return 0 === arguments.length ? null : (r = "object" === v(t) ? t : (10 === "".concat(t).length && (t = 1e3 * Number.parseInt(t)),
 | |
| //			new Date(t)),
 | |
| //			t += 6e4 * new Date(t).getTimezoneOffset(),
 | |
| //			{
 | |
| //				y: (r = new Date(t + 36e5 * n)).getFullYear(),
 | |
| //				m: r.getMonth() + 1 < 10 ? "0".concat(r.getMonth() + 1) : r.getMonth() + 1,
 | |
| //				d: r.getDate() < 10 ? "0".concat(r.getDate()) : r.getDate(),
 | |
| //				h: r.getHours() < 10 ? "0".concat(r.getHours()) : r.getHours(),
 | |
| //				f: r.getMinutes() < 10 ? "0".concat(r.getMinutes()) : r.getMinutes()
 | |
| //			})
 | |
| //		}(o),
 | |
| //		h = u.y,
 | |
| //		g = u.m,
 | |
| //		l = u.d,
 | |
| //		c = u.h,
 | |
| //		u = u.f,
 | |
| //		d = [h, g, l, c, u].join(""),
 | |
| //		f = [],
 | |
| //		d)
 | |
| //			f.push(m[Number(d[p])]);
 | |
| //		return h = A(f.join("")),
 | |
| //		g = A("".concat(o, "|").concat(a, "|").concat(e, "|").concat("web", "|").concat("3", "|").concat(h)),
 | |
| //		"".concat(h, "=").concat(o, "-").concat(a, "-").concat(g);
 | |
| //	})(url)
 | |
| //	   `)
 | |
| //	if err != nil {
 | |
| //		fmt.Println(err)
 | |
| //		return url
 | |
| //	}
 | |
| //	v, _ := r.Export().(string)
 | |
| //	return url + "?" + v
 | |
| //}
 | |
| 
 | |
| func (d *Pan123) login() error {
 | |
| 	var body base.Json
 | |
| 	if utils.IsEmailFormat(d.Username) {
 | |
| 		body = base.Json{
 | |
| 			"mail":     d.Username,
 | |
| 			"password": d.Password,
 | |
| 			"type":     2,
 | |
| 		}
 | |
| 	} else {
 | |
| 		body = base.Json{
 | |
| 			"passport": d.Username,
 | |
| 			"password": d.Password,
 | |
| 			"remember": true,
 | |
| 		}
 | |
| 	}
 | |
| 	res, err := base.RestyClient.R().
 | |
| 		SetHeaders(map[string]string{
 | |
| 			"origin":      "https://www.123pan.com",
 | |
| 			"referer":     "https://www.123pan.com/",
 | |
| 			"user-agent":  "Dart/2.19(dart:io)-alist",
 | |
| 			"platform":    "web",
 | |
| 			"app-version": "3",
 | |
| 			//"user-agent":  base.UserAgent,
 | |
| 		}).
 | |
| 		SetBody(body).Post(SignIn)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if utils.Json.Get(res.Body(), "code").ToInt() != 200 {
 | |
| 		err = fmt.Errorf(utils.Json.Get(res.Body(), "message").ToString())
 | |
| 	} else {
 | |
| 		d.AccessToken = utils.Json.Get(res.Body(), "data", "token").ToString()
 | |
| 	}
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| //func authKey(reqUrl string) (*string, error) {
 | |
| //	reqURL, err := url.Parse(reqUrl)
 | |
| //	if err != nil {
 | |
| //		return nil, err
 | |
| //	}
 | |
| //
 | |
| //	nowUnix := time.Now().Unix()
 | |
| //	random := rand.Intn(0x989680)
 | |
| //
 | |
| //	p4 := fmt.Sprintf("%d|%d|%s|%s|%s|%s", nowUnix, random, reqURL.Path, "web", "3", AuthKeySalt)
 | |
| //	authKey := fmt.Sprintf("%d-%d-%x", nowUnix, random, md5.Sum([]byte(p4)))
 | |
| //	return &authKey, nil
 | |
| //}
 | |
| 
 | |
| func (d *Pan123) Request(url string, method string, callback base.ReqCallback, resp interface{}) ([]byte, error) {
 | |
| 	isRetry := false
 | |
| do:
 | |
| 	req := base.RestyClient.R()
 | |
| 	req.SetHeaders(map[string]string{
 | |
| 		"origin":        "https://www.123pan.com",
 | |
| 		"referer":       "https://www.123pan.com/",
 | |
| 		"authorization": "Bearer " + d.AccessToken,
 | |
| 		"user-agent":    "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) alist-client",
 | |
| 		"platform":      "web",
 | |
| 		"app-version":   "3",
 | |
| 		//"user-agent":    base.UserAgent,
 | |
| 	})
 | |
| 	if callback != nil {
 | |
| 		callback(req)
 | |
| 	}
 | |
| 	if resp != nil {
 | |
| 		req.SetResult(resp)
 | |
| 	}
 | |
| 	//authKey, err := authKey(url)
 | |
| 	//if err != nil {
 | |
| 	//	return nil, err
 | |
| 	//}
 | |
| 	//req.SetQueryParam("auth-key", *authKey)
 | |
| 	res, err := req.Execute(method, GetApi(url))
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	body := res.Body()
 | |
| 	code := utils.Json.Get(body, "code").ToInt()
 | |
| 	if code != 0 {
 | |
| 		if !isRetry && code == 401 {
 | |
| 			err := d.login()
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			isRetry = true
 | |
| 			goto do
 | |
| 		}
 | |
| 		return nil, errors.New(jsoniter.Get(body, "message").ToString())
 | |
| 	}
 | |
| 	return body, nil
 | |
| }
 | |
| 
 | |
| func (d *Pan123) getFiles(ctx context.Context, parentId string, name string) ([]File, error) {
 | |
| 	page := 1
 | |
| 	total := 0
 | |
| 	res := make([]File, 0)
 | |
| 	// 2024-02-06 fix concurrency by 123pan
 | |
| 	for {
 | |
| 		if err := d.APIRateLimit(ctx, FileList); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		var resp Files
 | |
| 		query := map[string]string{
 | |
| 			"driveId":              "0",
 | |
| 			"limit":                "100",
 | |
| 			"next":                 "0",
 | |
| 			"orderBy":              "file_id",
 | |
| 			"orderDirection":       "desc",
 | |
| 			"parentFileId":         parentId,
 | |
| 			"trashed":              "false",
 | |
| 			"SearchData":           "",
 | |
| 			"Page":                 strconv.Itoa(page),
 | |
| 			"OnlyLookAbnormalFile": "0",
 | |
| 			"event":                "homeListFile",
 | |
| 			"operateType":          "4",
 | |
| 			"inDirectSpace":        "false",
 | |
| 		}
 | |
| 		_res, err := d.Request(FileList, http.MethodGet, func(req *resty.Request) {
 | |
| 			req.SetQueryParams(query)
 | |
| 		}, &resp)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		log.Debug(string(_res))
 | |
| 		page++
 | |
| 		res = append(res, resp.Data.InfoList...)
 | |
| 		total = resp.Data.Total
 | |
| 		if len(resp.Data.InfoList) == 0 || resp.Data.Next == "-1" {
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 	if len(res) != total {
 | |
| 		log.Warnf("incorrect file count from remote at %s: expected %d, got %d", name, total, len(res))
 | |
| 	}
 | |
| 	return res, nil
 | |
| }
 |