亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Golang > 正文

使用Golang簡單實現七牛圖片處理API

2020-04-01 19:09:20
字體:
來源:轉載
供稿:網友

之前一直在用qiniu的存儲服務,生成圖片的縮略圖,模糊圖,視頻的webp,現在需要把存儲移到s3上,那么這些圖片,視頻處理就要自己動手寫了,本文梳理一下大致的思路。

分析需求

先看一下qiniu的接口是如何處理圖片的,例如先截取視頻第一秒的圖片,再把圖片縮略,最后存儲到一個新的key,命令可以這么寫 vframe/jpg/offset/1|imageMogr2/thumbnail/400x|saveas/xxx, 可以看到三個操作之間用 | 符號分割,類似unix 的 pipe 操作。

上面的操作算作一個cmd, 一次API請求可以同時處理多個cmd,cmd之間用分號分割, 處理完畢后,在回調中把處理結果返回,例如

 

復制代碼 代碼如下:

{
    "id": "xxxxx",
    "pipeline": "xxx",
    "code": 0,
    "desc": "The fop was completed successfully",
    "reqid": "xTsAAFnxUbR5J10U",
    "inputBucket": "xxx",
    "inputKey": "xxxxx",
    "items": [
        {
            "cmd": "vframe/jpg/offset/1|imageMogr2/thumbnail/400x|saveas/ZmFtZS1wcml2YXRlOm1vbWVudC9jb3Zlci9zbmFwL3ZpZGVvL2M5YzdjZjQ5LTU3NGQtNGZjMS1iZDFkLTRkYjZkMzlkZWY1Ni8wLzA=",
            "code": 0,
            "desc": "The fop was completed successfully",
            "hash": "FhdN6V8EI4vW4XJGALSfxutvMEIv",
            "key": "xx",
            "returnOld": 0
        },
        {
            "cmd": "vframe/jpg/offset/1|imageMogr2/thumbnail/400x|imageMogr2/blur/45x8|saveas/ZmFtZS1wcml2YXRlOm1vbWVudC9jb3Zlci9zbmFwL3ZpZGVvL2M5YzdjZjQ5LTU3NGQtNGZjMS1iZDFkLTRkYjZkMzlkZWY1Ni8wLzBfYmx1cg==",
            "code": 0,
            "desc": "The fop was completed successfully",
            "hash": "FgNiRzrCsa7TZx1xVSb_4d5TiaK3",
            "key": "xxx",
            "returnOld": 0
        }
    ]
}

 

分解需求

這個程序大致需要這么幾個部分:

一個http接口,接受任務,接受后把任務扔到隊列,返回一個job ID。 worker異步處理任務,worker的個數 和 每個worker 并行的處理的個數 能夠配置,worker有重試機制。
從 job payload 中解析出需要做的任務,解析出每個cmd, 最好能并行執行每一個 cmd, 記錄每一個cmd的結果

每個cmd中有多個 operation, 并且用 pipe 連接,前一個operaion的輸出是后一個operation的輸入

可以把 1 和 2,3 分開來看,1 比較獨立,之前寫過一個worker的模型,參考的是這篇文章 Handling 1 Million Requests per Minute with Go,比較詳細,是用 go channel 作為queue的,我加了一個 beanstalk 作為 queue的 providor。還有一點改進是,文章中只提供了worker數量的設置,我再加了一個參數,設定每個worker可以并行執行的協程數。所以下面主要講講3, 2的解決辦法

Pipe

可以參考這個庫 pipe, 用法如下:

 

復制代碼 代碼如下:

p := pipe.Line(
    pipe.ReadFile("test.png"),
    resize(300, 300),
    blur(0.5),
)

 

output, err := pipe.CombinedOutput(p)
if err != nil {
    fmt.Printf("%v/n", err)
}

buf := bytes.NewBuffer(output)
img, _ := imaging.Decode(buf)

imaging.Save(img, "test_a.png")

 

還是比較方便的,建一個 Cmd struct, 利用正則匹配一下每個 Operation 的參數,放入一個 []Op slice, 最后執行,struct和方法如下:

 

復制代碼 代碼如下:

type Cmd struct {
    cmd    string
    saveas string
    ops    []Op
    err    error
}

 

type Op interface {
    getPipe() pipe.Pipe
}

type ResizeOp struct {
    width, height int
}

func (c ResizeOp) getPipe() pipe.Pipe {
    return resize(c.width, c.height)
}

//使用方法
cmdStr := `file/test.png|thumbnail/x300|blur/20x8`
cmd := Cmd{cmdStr, "test_b.png", nil, nil}

cmd.parse()
cmd.doOps()
sync.WaitGroup

 

單個cmd處理解決后,就是多個cmd的并行問題,沒啥好想的,直接用 sync.WaitGroup 就可以完美解決。一步一步來,我們先看看這個struct的使用方法:

 

復制代碼 代碼如下:

func main() {
    cmds := []string{}
    for i := 0; i < 10000; i++ {
        cmds = append(cmds, fmt.Sprintf("cmd-%d", i))
    }

 

    results := handleCmds(cmds)

    fmt.Println(len(results)) // 10000
}

func doCmd(cmd string) string {
    return fmt.Sprintf("cmd=%s", cmd)
}

func handleCmds(cmds []string) (results []string) {
    fmt.Println(len(cmds)) //10000
    var count uint64

    group := sync.WaitGroup{}
    lock := sync.Mutex{}
    for _, item := range cmds {
        // 計數加一
        group.Add(1)
        go func(cmd string) {
            result := doCmd(cmd)
            atomic.AddUint64(&count, 1)

            lock.Lock()
            results = append(results, result)
            lock.Unlock()
           
            // 計數減一
            group.Done()
        }(item)
    }

    // 阻塞
    group.Wait()

    fmt.Printf("count=%d /n", count) // 10000
    return
}

 

group本質大概是一個計數器,計數 > 0時, group.Wait() 會阻塞,直到 計數 == 0. 這里還有一點要注意,就是 results = append(results, result) 的操作是線程不安全的,清楚這里 results 是共享的,需要加鎖來保證同步,否則最后 len(results) 不為 10000。

我們建一個BenchCmd, 來存放 cmds. 如下:

 

復制代碼 代碼如下:

type BenchCmd struct {
    cmds      []Cmd
    waitGroup sync.WaitGroup
    errs      []error
    lock      sync.Mutex
}

 

func (b *BenchCmd) doCmds() {
    for _, item := range b.cmds {
        b.waitGroup.Add(1)

        go func(cmd Cmd) {
            cmd.parse()
            err := cmd.doOps()

            b.lock.Lock()
            b.errs = append(b.errs, err)
            b.lock.Unlock()

            b.waitGroup.Done()
        }(item)
    }

    b.waitGroup.Wait()
}

 

最后的調用就像這樣:

 

復制代碼 代碼如下:

var cmds []Cmd
cmd_a := Cmd{`file/test.png|thumbnail/x300|blur/20x8`, "test_a.png", nil, nil}
cmd_b := Cmd{`file/test.png|thumbnail/500x1000|blur/20x108`, "test_b.png", nil, nil}
cmd_c := Cmd{`file/test.png|thumbnail/300x300`, "test_c.png", nil, nil}

 

cmds = append(cmds, cmd_a)
cmds = append(cmds, cmd_b)
cmds = append(cmds, cmd_c)

bench := BenchCmd{
    cmds:      cmds,
    waitGroup: sync.WaitGroup{},
    lock:      sync.Mutex{},
}

bench.doCmds()

fmt.Println(bench.errs)

 

這只是一個初級的實驗,思考還不夠全面,并且只是模仿API,qiniu應該不是這么做的,耦合更低,可能各個Cmd都有各自處理的集群,那pipe這個庫就暫時沒法解決了,目前的局限在于 每個Cmd必須都在一個進程中。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产日本欧美一区| 九九精品视频在线| 久久91精品国产91久久跳| 一本色道久久综合狠狠躁篇怎么玩| 91精品国产综合久久久久久久久| 亚洲成人网在线观看| 色综合久久久久久中文网| 国产精品吹潮在线观看| 日韩av在线网页| 久久五月天色综合| 日韩av在线网站| 久久久日本电影| 国产成人欧美在线观看| 日韩精品极品在线观看播放免费视频| 亚洲欧美变态国产另类| 亚洲国产精品电影在线观看| 国产精品一区久久| 美女精品视频一区| 98视频在线噜噜噜国产| 国产精品久久久久久婷婷天堂| 一本色道久久88精品综合| 国产精品精品视频一区二区三区| 亚洲成人三级在线| 久久久久中文字幕2018| 欧美成人免费观看| 欧美一区二粉嫩精品国产一线天| 中国日韩欧美久久久久久久久| 欧美激情一区二区三区高清视频| 日本免费一区二区三区视频观看| 亚洲午夜久久久久久久| 97久久超碰福利国产精品…| 日韩一区二区欧美| 色综合伊人色综合网| 成人黄色av网站| 欧美裸体xxxx极品少妇软件| 91精品久久久久久| 国产精品免费福利| 97视频在线观看成人| 国产精品久久久久免费a∨| 久久亚洲精品一区二区| 欧美日韩在线视频观看| 日韩一级裸体免费视频| 91高清在线免费观看| 成人精品视频久久久久| 国产亚洲a∨片在线观看| 91精品国产乱码久久久久久蜜臀| 裸体女人亚洲精品一区| 亚洲图片欧美午夜| 亚洲欧美激情精品一区二区| 日韩电影免费在线观看| 久久777国产线看观看精品| 2019最新中文字幕| 国产精品99蜜臀久久不卡二区| 亚洲成人网在线观看| 欧美激情xxxxx| 2023亚洲男人天堂| 亚洲美女动态图120秒| 日本一区二区不卡| 国产精品福利无圣光在线一区| 九九热视频这里只有精品| 日韩精品视频在线观看网址| 欧美黑人巨大xxx极品| 欧美成人精品三级在线观看| 亚洲欧美国内爽妇网| 91地址最新发布| 国产日韩欧美日韩大片| 91情侣偷在线精品国产| 91日本在线观看| 日韩欧美精品网址| 欧美午夜宅男影院在线观看| 欧美日韩一区二区三区在线免费观看| 国产精品亚洲美女av网站| 国产成人久久精品| 亚洲最大成人网色| 97久久久免费福利网址| 亚洲精品在线观看www| 亚洲天堂网在线观看| 久久久免费精品| 正在播放欧美一区| 久久精品小视频| 国产精品爱啪在线线免费观看| 欧美激情一二三| 亚洲精品国产免费| 欧美孕妇与黑人孕交| 中文字幕亚洲欧美日韩高清| 欧美日韩亚洲网| 欧美日韩中国免费专区在线看| 亚洲精品自拍偷拍| 日韩中文字幕在线视频播放| 黑人极品videos精品欧美裸| 国产精品青青在线观看爽香蕉| 色先锋久久影院av| 成人网在线免费看| 国产精品久久久久免费a∨| 国产精品福利久久久| 日本久久精品视频| 亚洲成人黄色在线观看| 中日韩美女免费视频网址在线观看| 亚洲精选在线观看| 欧美成人免费小视频| 久久久亚洲网站| 美女黄色丝袜一区| 国产精品美女在线| 亚洲国产古装精品网站| 欧美成年人视频网站| 国产在线观看精品| 国产精国产精品| 亚洲情综合五月天| 夜夜嗨av一区二区三区四区| 亚洲人成电影网站色www| 久久久国产视频| 日韩美女在线观看一区| 久久男人av资源网站| 亚洲天堂成人在线视频| 国产精品xxxxx| 欧美巨猛xxxx猛交黑人97人| 欧美成年人视频| 国产91九色视频| 亚洲伦理中文字幕| 亚洲午夜精品视频| 亚洲欧美国产一本综合首页| 亚洲视频在线免费观看| 欧美成人剧情片在线观看| 国产欧美久久久久久| 欧美日韩一区二区免费在线观看| 伊人久久男人天堂| 欧美精品在线观看| 国产视频自拍一区| 欧美日韩国产在线看| 欧美激情亚洲视频| 国产精品一香蕉国产线看观看| 日韩精品有码在线观看| 国产欧美亚洲视频| 国产国产精品人在线视| 国产精品老牛影院在线观看| 国产999精品视频| 91久久精品国产91久久性色| 色老头一区二区三区| 欧美最猛黑人xxxx黑人猛叫黄| 国产精品第一区| 欧美电影免费观看高清完整| 亚洲精品久久久久久久久| 精品美女久久久久久免费| 91精品国产沙发| 神马久久桃色视频| 岛国av一区二区| 欧美精品在线播放| 97免费中文视频在线观看| 北条麻妃99精品青青久久| 欧美成人免费在线观看| 国产激情综合五月久久| 在线观看精品国产视频| 91精品视频一区| 91精品视频在线| 欧美香蕉大胸在线视频观看| 国产精品偷伦免费视频观看的| 亚洲999一在线观看www| 国产亚洲一区精品| 日韩免费在线电影| 久久琪琪电影院| 国产精品久久久久91| 国产91精品视频在线观看| 色狠狠久久aa北条麻妃| 国产亚洲精品美女久久久|