generated from bing/readnotes
538 lines
12 KiB
Go
538 lines
12 KiB
Go
|
package main
|
|||
|
|
|||
|
import (
|
|||
|
"bytes"
|
|||
|
"flag"
|
|||
|
"fmt"
|
|||
|
"io"
|
|||
|
"net/http"
|
|||
|
"os"
|
|||
|
"path"
|
|||
|
"sort"
|
|||
|
"strconv"
|
|||
|
"strings"
|
|||
|
"time"
|
|||
|
|
|||
|
"github.com/aliyun/aliyun-oss-go-sdk/oss"
|
|||
|
)
|
|||
|
|
|||
|
const (
|
|||
|
metaKeyUpdatedAt = "UpdatedAt"
|
|||
|
metaKeyModifiedAt = "Last-Modified"
|
|||
|
metaKeyFileSize = "Content-Length"
|
|||
|
)
|
|||
|
|
|||
|
type AliOSS struct {
|
|||
|
bucket *oss.Bucket
|
|||
|
client *oss.Client
|
|||
|
endpoint string
|
|||
|
accessKey string
|
|||
|
secret string
|
|||
|
bucketName string
|
|||
|
}
|
|||
|
|
|||
|
func NewAliOSS(endpoint, accessKey, secret string, bucket string) *AliOSS {
|
|||
|
return &AliOSS{
|
|||
|
endpoint: endpoint,
|
|||
|
accessKey: accessKey,
|
|||
|
secret: secret,
|
|||
|
bucketName: bucket,
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) Init(opts ...oss.ClientOption) error {
|
|||
|
client, err := oss.New(a.endpoint, a.accessKey, a.secret, opts...)
|
|||
|
if err != nil {
|
|||
|
return err
|
|||
|
}
|
|||
|
a.client = client
|
|||
|
a.bucket, err = client.Bucket(a.bucketName)
|
|||
|
return err
|
|||
|
}
|
|||
|
|
|||
|
// Copy 拷贝文件
|
|||
|
func (a *AliOSS) Copy(src, dst string, opts ...oss.Option) error {
|
|||
|
|
|||
|
_, err := a.bucket.CopyObject(src, dst, opts...)
|
|||
|
return err
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) Delete(f []string, opts ...oss.Option) ([]string, error) {
|
|||
|
res, err := a.bucket.DeleteObjects(f, opts...)
|
|||
|
return res.DeletedObjects, err
|
|||
|
}
|
|||
|
|
|||
|
// Download 下载文件
|
|||
|
func (a *AliOSS) Download(obj, savePath string, opts ...oss.Option) error {
|
|||
|
err := a.bucket.GetObjectToFile(obj, savePath, opts...)
|
|||
|
return err
|
|||
|
}
|
|||
|
|
|||
|
// IsExist 判断文件是否存在
|
|||
|
func (a *AliOSS) IsExist(obj string) (bool, error) {
|
|||
|
return a.bucket.IsObjectExist(obj)
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) UploadObject(dest string, src io.Reader, opts ...oss.Option) error {
|
|||
|
return a.bucket.PutObject(dest, src, opts...)
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) UploadByte(dest string, buf []byte, opts ...oss.Option) error {
|
|||
|
return a.UploadObject(dest, bytes.NewReader(buf), opts...)
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) UploadStream(dest string, src *os.File, opts ...oss.Option) error {
|
|||
|
// 读取本地文件。
|
|||
|
return a.UploadObject(dest, src, opts...)
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) UploadFile(dest string, filePath string, opts ...oss.Option) error {
|
|||
|
err := a.bucket.PutObjectFromFile(dest, filePath, opts...)
|
|||
|
return err
|
|||
|
}
|
|||
|
|
|||
|
// set meta
|
|||
|
func (a *AliOSS) SetObjectMeta(file string, key, val string) error {
|
|||
|
meta := oss.Meta(key, val)
|
|||
|
err := a.bucket.SetObjectMeta(file, meta)
|
|||
|
return err
|
|||
|
}
|
|||
|
|
|||
|
// set metas
|
|||
|
func (a *AliOSS) SetObjectMetaMap(file string, metaMap map[string]string) error {
|
|||
|
metas := []oss.Option{}
|
|||
|
for k, v := range metaMap {
|
|||
|
metas = append(metas, oss.Meta(k, v))
|
|||
|
}
|
|||
|
if len(metas) == 0 {
|
|||
|
return nil
|
|||
|
}
|
|||
|
return a.bucket.SetObjectMeta(file, metas...)
|
|||
|
}
|
|||
|
|
|||
|
// get metas
|
|||
|
func (a *AliOSS) GetObjectMeta(file string) (map[string]string, error) {
|
|||
|
res := make(map[string]string)
|
|||
|
metas, err := a.bucket.GetObjectMeta(file)
|
|||
|
for k, v := range metas {
|
|||
|
val := ""
|
|||
|
if len(v) > 0 {
|
|||
|
val = v[0]
|
|||
|
}
|
|||
|
res[k] = val
|
|||
|
}
|
|||
|
return res, err
|
|||
|
}
|
|||
|
|
|||
|
// set updatedAt
|
|||
|
func (a *AliOSS) SetUpdatedAt(file string, updatedAt *time.Time) error {
|
|||
|
if updatedAt == nil {
|
|||
|
now := time.Now()
|
|||
|
updatedAt = &now
|
|||
|
}
|
|||
|
return a.SetObjectMeta(file, metaKeyUpdatedAt, strconv.FormatInt(updatedAt.Unix(), 10))
|
|||
|
}
|
|||
|
|
|||
|
// get updatedAt
|
|||
|
func (a *AliOSS) GetUpdatedAt(file string) (int64, error) {
|
|||
|
metas, err := a.GetObjectMeta(file)
|
|||
|
if err != nil {
|
|||
|
return 0, err
|
|||
|
}
|
|||
|
if u, ok := metas[metaKeyUpdatedAt]; ok {
|
|||
|
i, err := strconv.ParseInt(u, 10, 64)
|
|||
|
return i, err
|
|||
|
}
|
|||
|
return 0, nil
|
|||
|
}
|
|||
|
|
|||
|
// get updatedAt
|
|||
|
func (a *AliOSS) GetModifiedAt(file string) (int64, error) {
|
|||
|
metas, err := a.GetObjectMeta(file)
|
|||
|
if err != nil {
|
|||
|
return 0, err
|
|||
|
}
|
|||
|
if u, ok := metas[metaKeyModifiedAt]; ok {
|
|||
|
//Thu, 07 Apr 2022 03:13:49 GMT
|
|||
|
t, err := time.Parse(time.RFC1123, u)
|
|||
|
return t.Unix(), err
|
|||
|
}
|
|||
|
return 0, nil
|
|||
|
}
|
|||
|
|
|||
|
// upload stream with set updated at
|
|||
|
func (a *AliOSS) UploadObjectWithUpdatedAt(dest string, src io.Reader, opts ...oss.Option) error {
|
|||
|
opt := oss.Meta(metaKeyUpdatedAt, strconv.FormatInt(time.Now().Unix(), 10))
|
|||
|
opts = append(opts, opt)
|
|||
|
return a.UploadObject(dest, src, opts...)
|
|||
|
}
|
|||
|
|
|||
|
// upload file with set updated at
|
|||
|
func (a *AliOSS) UploadFileWithUpdatedAt(dest string, src string, opts ...oss.Option) error {
|
|||
|
err := a.UploadFile(dest, src, opts...)
|
|||
|
if err != nil {
|
|||
|
return err
|
|||
|
}
|
|||
|
return a.SetUpdatedAt(dest, nil)
|
|||
|
}
|
|||
|
|
|||
|
// upload bytes with set updated at
|
|||
|
func (a *AliOSS) UploadBytesWithUpdatedAt(dest string, src []byte, opts ...oss.Option) error {
|
|||
|
opt := oss.Meta(metaKeyUpdatedAt, strconv.FormatInt(time.Now().Unix(), 10))
|
|||
|
opts = append(opts, opt)
|
|||
|
return a.UploadByte(dest, src, opts...)
|
|||
|
}
|
|||
|
|
|||
|
func (a *AliOSS) Name() string {
|
|||
|
return a.bucketName
|
|||
|
}
|
|||
|
|
|||
|
// check file version, if not need download ,return true
|
|||
|
func (a *AliOSS) CheckFileVersion(obj string, info os.FileInfo) (bool, error) {
|
|||
|
metas, err := a.GetObjectMeta(obj)
|
|||
|
if err != nil {
|
|||
|
return true, err
|
|||
|
}
|
|||
|
// var modifiedAt time.Time
|
|||
|
var modifiedAt int64
|
|||
|
if u, ok := metas[metaKeyModifiedAt]; ok {
|
|||
|
//Thu, 07 Apr 2022 03:13:49 GMT
|
|||
|
t, err := time.Parse(time.RFC1123, u)
|
|||
|
if err == nil {
|
|||
|
modifiedAt = t.Unix()
|
|||
|
}
|
|||
|
// modifiedAt, _ = strconv.ParseInt(u, 10, 64)
|
|||
|
}
|
|||
|
//TODO is need to check file size??
|
|||
|
// var length int64 = 0
|
|||
|
// if u, ok := metas[metaKeyFileSize]; ok {
|
|||
|
// n, err := strconv.ParseInt(u, 10, 64)
|
|||
|
// if err == nil {
|
|||
|
// length = n
|
|||
|
// }
|
|||
|
// }
|
|||
|
//如果本地文件时间更靠后或者本地文件更大,返回true
|
|||
|
// return info.ModTime().Unix() >= modifiedAt || length <= info.Size(), nil
|
|||
|
//如果本地文件时间更靠后,返回true, 防止意外退出,未成功上传
|
|||
|
return info.ModTime().Unix() >= modifiedAt, nil
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
// const host = "http://testspeed.xk.design"
|
|||
|
|
|||
|
const MAX = 300
|
|||
|
|
|||
|
type file struct{
|
|||
|
Name string
|
|||
|
Width int
|
|||
|
Height int
|
|||
|
Src string
|
|||
|
}
|
|||
|
|
|||
|
var test1 = []file{
|
|||
|
{
|
|||
|
Name: "test4k.jpg",
|
|||
|
Width: 0,
|
|||
|
Height: 0,
|
|||
|
Src: "test4k.jpg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "test2k.jpg",
|
|||
|
Width: 2590,
|
|||
|
Height: 1619,
|
|||
|
Src: "test4k.jpg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "test1k.jpg",
|
|||
|
Width: 1831,
|
|||
|
Height: 1144,
|
|||
|
Src: "test4k.jpg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "test600.jpg",
|
|||
|
Width: 647,
|
|||
|
Height: 404,
|
|||
|
Src: "test4k.jpg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "test1of10.jpg",
|
|||
|
Width: 384,
|
|||
|
Height: 240,
|
|||
|
Src: "test4k.jpg",
|
|||
|
},
|
|||
|
}
|
|||
|
|
|||
|
var test2 = []file{
|
|||
|
{
|
|||
|
Name: "8ktestsrc.jpeg",
|
|||
|
Width: 7680,
|
|||
|
Height: 4320,
|
|||
|
Src: "8ktestsrc.jpeg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "8ktest3k.jpeg",
|
|||
|
Width: 2730,
|
|||
|
Height: 1536,
|
|||
|
Src: "8ktestsrc.jpeg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "8ktest1080.jpeg",
|
|||
|
Width: 1930,
|
|||
|
Height: 1086,
|
|||
|
Src: "8ktestsrc.jpeg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "8ktest682.jpeg",
|
|||
|
Width: 682,
|
|||
|
Height: 384,
|
|||
|
Src: "8ktestsrc.jpeg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "8ktest400.jpeg",
|
|||
|
Width: 400,
|
|||
|
Height: 225,
|
|||
|
Src: "8ktestsrc.jpeg",
|
|||
|
},
|
|||
|
{
|
|||
|
Name: "8ktest300.jpeg",
|
|||
|
Width: 300,
|
|||
|
Height: 300,
|
|||
|
Src: "8ktestsrc.jpeg",
|
|||
|
},
|
|||
|
}
|
|||
|
|
|||
|
func NewRequest(url string)(*http.Request, error){
|
|||
|
req, err := http.NewRequest(http.MethodGet, url, nil)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
req.Header.Add("Accept","text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9")
|
|||
|
req.Header.Add("Accept-Encoding","gzip, deflate")
|
|||
|
req.Header.Add("User-Agent","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36 Edg/107.0.1418.56")
|
|||
|
req.Header.Add("Accept-Language","zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6")
|
|||
|
req.Header.Add("Cache-Control","no-cache")
|
|||
|
req.Header.Add("Connection"," keep-alive")
|
|||
|
req.Header.Add("Cookie"," Hm_lvt_2e2c80c1bd8a51afc2d7de641330a397=1669089775,1669615598,1669704782,1669875251; Hm_lpvt_2e2c80c1bd8a51afc2d7de641330a397=1669964998")
|
|||
|
req.Header.Add("Host"," test-pic.xk.design")
|
|||
|
// req.Header.Add("If-Modified-Since"," Fri, 02 Dec 2022 06:31:28 GMT")
|
|||
|
// req.Header.Add("If-None-Match","\"4B8579BA5E07DD57405973606E777476\"")
|
|||
|
// req.Header.Add("Upgrade-Insecure-Requests","1")
|
|||
|
return req, nil
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
func test(f file, count int, host string){
|
|||
|
fmt.Printf("begin normal testing for %s\n", f.Name)
|
|||
|
startedAt := time.Now().UnixMilli()
|
|||
|
errCount := 0
|
|||
|
costMS := []int64{}
|
|||
|
for i := 0; i < count; i ++ {
|
|||
|
req, err := NewRequest(host + "/" + f.Name)
|
|||
|
if err != nil {
|
|||
|
errCount ++
|
|||
|
continue
|
|||
|
}
|
|||
|
client := &http.Client{}
|
|||
|
begin := time.Now().UnixMilli()
|
|||
|
resp, err := client.Do(req)
|
|||
|
if err != nil {
|
|||
|
errCount++
|
|||
|
continue
|
|||
|
}
|
|||
|
defer resp.Body.Close()
|
|||
|
data, err := io.ReadAll(resp.Body)
|
|||
|
os.WriteFile("/dev/null", data, os.ModePerm)
|
|||
|
if err != nil {
|
|||
|
errCount++
|
|||
|
continue
|
|||
|
}
|
|||
|
end := time.Now().UnixMilli()
|
|||
|
costMS = append(costMS, end-begin)
|
|||
|
}
|
|||
|
finishedAt := time.Now().UnixMilli()
|
|||
|
totalCost := finishedAt - startedAt
|
|||
|
fmt.Printf("in %d tests for file %s@%dx%d, cost %d ms, errors: %d", MAX, f.Src, f.Width, f.Height, totalCost, errCount)
|
|||
|
statistics(costMS)
|
|||
|
}
|
|||
|
|
|||
|
func testResize(f file, count int, host string)(int64, int64, int64){
|
|||
|
fmt.Printf("begin resize testing for %s\n", f.Name)
|
|||
|
startedAt := time.Now().UnixMilli()
|
|||
|
errCount := 0
|
|||
|
costMS := []int64{}
|
|||
|
for i := 0; i < count; i ++ {
|
|||
|
query := "?x-oss-process=image/resize,m_lfit,w_"+strconv.Itoa(f.Width) + ",h_" + strconv.Itoa(f.Height)
|
|||
|
if f.Height == 0 || f.Width == 0 {
|
|||
|
query = ""
|
|||
|
}
|
|||
|
req, err := NewRequest(host + "/" + f.Src + query)
|
|||
|
if err != nil {
|
|||
|
errCount++
|
|||
|
continue
|
|||
|
}
|
|||
|
client := http.Client{}
|
|||
|
begin := time.Now().UnixMilli()
|
|||
|
resp, err := client.Do(req)
|
|||
|
if err != nil {
|
|||
|
errCount++
|
|||
|
continue
|
|||
|
}
|
|||
|
defer resp.Body.Close()
|
|||
|
data, err := io.ReadAll(resp.Body)
|
|||
|
os.WriteFile("/dev/null", data, os.ModePerm)
|
|||
|
if err != nil {
|
|||
|
errCount++
|
|||
|
continue
|
|||
|
}
|
|||
|
end := time.Now().UnixMilli()
|
|||
|
costMS = append(costMS, end-begin)
|
|||
|
// if resp != nil {
|
|||
|
// fmt.Println(resp , resp.StatusCode, resp.Status)
|
|||
|
// }
|
|||
|
}
|
|||
|
finishedAt := time.Now().UnixMilli()
|
|||
|
totalCost := finishedAt - startedAt
|
|||
|
fmt.Printf("in %d tests for file %s@%dx%d, cost %d ms, errors: %d", MAX, f.Src, f.Width, f.Height, totalCost, errCount)
|
|||
|
return statistics(costMS)
|
|||
|
}
|
|||
|
|
|||
|
func filelist(dir string)([]string, error){
|
|||
|
fis, err := os.ReadDir(dir)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
fs := []string{}
|
|||
|
for _, f := range fis {
|
|||
|
if !f.IsDir() {
|
|||
|
fs = append(fs, f.Name())
|
|||
|
}
|
|||
|
}
|
|||
|
return fs, nil
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
func case3(dir string, ak string, sk string, count int, host string)error{
|
|||
|
files, err := filelist(dir)
|
|||
|
if err != nil {
|
|||
|
return err
|
|||
|
}
|
|||
|
alioss := NewAliOSS("oss-cn-hangzhou.aliyuncs.com", ak,sk, "oxslmimg")
|
|||
|
alioss.Init()
|
|||
|
avgmap := make(map[string][]string)
|
|||
|
p95map := make(map[string][]string)
|
|||
|
for _, f := range files {
|
|||
|
err = alioss.UploadFile(f, path.Join(dir,f))
|
|||
|
if err != nil {
|
|||
|
fmt.Println("upload failed:", err, f, path.Join(dir,f))
|
|||
|
continue
|
|||
|
}
|
|||
|
avgmap[f] = []string{}
|
|||
|
p95map[f] = []string{}
|
|||
|
for _, t := range test1 {
|
|||
|
t.Src = f
|
|||
|
t.Name = f
|
|||
|
avg, p95, _ := testResize(t, count, host)
|
|||
|
avgmap[f] = append(avgmap[f], strconv.FormatInt(avg, 10))
|
|||
|
p95map[f] = append(p95map[f], strconv.FormatInt(p95, 10))
|
|||
|
}
|
|||
|
}
|
|||
|
fmt.Println("")
|
|||
|
for f, v := range avgmap {
|
|||
|
fmt.Printf("%s\t%s\n", f, strings.Join(v, "\t"))
|
|||
|
}
|
|||
|
for f, v := range p95map {
|
|||
|
fmt.Printf("%s\t%s\n", f, strings.Join(v, "\t"))
|
|||
|
}
|
|||
|
return nil
|
|||
|
}
|
|||
|
|
|||
|
func avg(data []int64)int64{
|
|||
|
if len(data) == 0 {
|
|||
|
return 0
|
|||
|
}
|
|||
|
total := int64(0)
|
|||
|
for _, i := range data {
|
|||
|
total = total + i
|
|||
|
}
|
|||
|
return total/int64(len(data))
|
|||
|
}
|
|||
|
|
|||
|
type SortInt64 []int64
|
|||
|
|
|||
|
func (a SortInt64) Len() int { return len(a) }
|
|||
|
func (a SortInt64) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|||
|
func (a SortInt64) Less(i, j int) bool { return a[i] < a[j] }
|
|||
|
|
|||
|
func percent50(data []int64)int64{
|
|||
|
if len(data) == 0 {
|
|||
|
return 0
|
|||
|
}
|
|||
|
i := int(float64(len(data)) * 0.5 )
|
|||
|
if i == len(data) {
|
|||
|
return 0
|
|||
|
}
|
|||
|
return data[i]
|
|||
|
}
|
|||
|
|
|||
|
func percent90(data []int64)int64{
|
|||
|
if len(data) == 0 {
|
|||
|
return 0
|
|||
|
}
|
|||
|
i := int(float64(len(data)) * 0.9 )
|
|||
|
if i == len(data) {
|
|||
|
return 0
|
|||
|
}
|
|||
|
return data[i]
|
|||
|
}
|
|||
|
|
|||
|
func percent95(data []int64)int64{
|
|||
|
if len(data) == 0 {
|
|||
|
return 0
|
|||
|
}
|
|||
|
i := int(float64(len(data)) * 0.95 )
|
|||
|
if i == len(data) {
|
|||
|
return 0
|
|||
|
}
|
|||
|
return data[i]
|
|||
|
}
|
|||
|
|
|||
|
func max(data []int64)int64{
|
|||
|
return data[len(data)-1]
|
|||
|
}
|
|||
|
|
|||
|
func statistics(data []int64)(int64, int64, int64){
|
|||
|
a := avg(data)
|
|||
|
sort.Sort(SortInt64(data))
|
|||
|
m := percent50(data)
|
|||
|
p9 := percent90(data)
|
|||
|
p95 := percent95(data)
|
|||
|
ma := max(data)
|
|||
|
fmt.Printf("\t avg: %d middle: %dms percent90: %dms percent95: %dms max: %dms\n", a, m, p9, p95, ma)
|
|||
|
return a, p95, ma
|
|||
|
}
|
|||
|
|
|||
|
func main(){
|
|||
|
var tcase, count int
|
|||
|
var host, ak, sk, dir string
|
|||
|
flag.IntVar(&tcase, "case", 1, "which test case to use")
|
|||
|
flag.IntVar(&count, "count", 100, "total test count")
|
|||
|
flag.StringVar(&host, "host", "http://testspeed.xk.design", "access host")
|
|||
|
flag.StringVar(&ak, "ak", "", "access key")
|
|||
|
flag.StringVar(&sk, "sk", "", "access secret")
|
|||
|
flag.StringVar(&dir , "dir", "", "img path")
|
|||
|
flag.Parse()
|
|||
|
switch tcase {
|
|||
|
case 1:
|
|||
|
for _, f := range test1 {
|
|||
|
test(f, count, host)
|
|||
|
testResize(f, count, host)
|
|||
|
}
|
|||
|
case 2:
|
|||
|
for _, f := range test2 {
|
|||
|
test(f, count, host)
|
|||
|
testResize(f, count, host)
|
|||
|
}
|
|||
|
case 3:
|
|||
|
case3(dir, ak, sk, count, host)
|
|||
|
}
|
|||
|
}
|