go-common/app/admin/main/upload/service/upload.go
2019-04-22 18:49:16 +08:00

286 lines
8.5 KiB
Go

package service
import (
"bytes"
"context"
"crypto/hmac"
"crypto/sha1"
"encoding/base64"
"fmt"
"hash"
"io/ioutil"
"net/http"
"strconv"
"time"
"go-common/app/admin/main/upload/model"
"go-common/library/ecode"
"go-common/library/log"
"github.com/pkg/errors"
)
const (
_downloadURL = "http://%s/bfs/%s/%s"
_uploadURL = "http://%s/bfs/%s/%s"
_deleteURL = "http://%s/bfs/%s/%s"
_privateBucket = "facepri" // bucket to save yellow pic
_privateBucketAppKey = "8923aff2e1124bb2"
_privateBucketAppSecret = "b237e8927823cc2984aee980123cb0"
)
// Add will add a record into bfs_upload_admin table
func (s *Service) Add(c context.Context, ap *model.AddParam) (err error) {
record := new(model.Record)
record.Bucket = ap.Bucket
record.FileName = ap.FileName
record.URL = ap.URL
record.Sex = ap.Sex
record.Politics = ap.Politics
err = s.orm.Create(&record).Error
return
}
// List lists records
func (s *Service) List(c context.Context, lp *model.ListParam) (listResSlice []*model.Record, err error) {
listResSlice = make([]*model.Record, 0)
err = s.orm.Limit(10).Order("id desc").Where("state=?", lp.State).Where("bucket=?", lp.Bucket).Find(&listResSlice).Error
return
}
// MultiList lists records from multi bucket
func (s *Service) MultiList(c context.Context, lp *model.MultiListParam) (result []*model.MultiListResult, err error) {
result = make([]*model.MultiListResult, 0)
if len(lp.Bucket) == 0 {
var buckets []*model.Bucket
if err = s.orm.Table("bucket").Order("id desc").Find(&buckets).Error; err != nil {
log.Error("read bucket error(%v)", err)
return
}
for _, v := range buckets {
lp.Bucket = append(lp.Bucket, v.BucketName)
}
}
for _, bucket := range lp.Bucket {
tmpResult := &model.MultiListResult{}
tmpResult.Bucket = bucket
tmpRecord := make([]*model.Record, 0)
if err = s.orm.Limit(10).Order("id desc").Where("state=?", lp.State).Where("bucket=?", bucket).Find(&tmpRecord).Error; err != nil {
return
}
tmpResult.Imgs = tmpRecord
result = append(result, tmpResult)
}
return
}
// Delete deletes a record and delete file in bfs
func (s *Service) Delete(c context.Context, dp *model.DeleteParam) (err error) {
var (
downloadBytes []byte
contentType string
)
record := new(model.Record)
if err = s.orm.Where("id=?", dp.Rid).Find(&record).Error; err != nil {
err = errors.Wrapf(err, "Query(%d)", dp.Rid)
return
}
dp.Bucket = record.Bucket
dp.FileName = record.FileName
if downloadBytes, contentType, err = s.download(dp); err != nil {
return
}
if err = s.upload(dp, contentType, downloadBytes); err != nil {
return
}
if err = s.delete(dp); err != nil {
return
}
if err = s.orm.Where("id=?", dp.Rid).Update("state", 1).Update("adminid", dp.AdminID).Error; err != nil {
err = errors.Wrapf(err, "Update(%d,%d,%d)", dp.Rid, 1, dp.AdminID)
return
}
return
}
// DeleteRaw delete file in bfs
func (s *Service) DeleteRaw(c context.Context, dp *model.DeleteRawParam) (err error) {
d := &model.DeleteParam{
Bucket: dp.Bucket,
FileName: dp.FileName,
}
return s.delete(d)
}
// download from bfs
func (s *Service) download(dp *model.DeleteParam) (downloadBytes []byte, contentType string, err error) {
var (
downloadReq *http.Request
resp *http.Response
bfsDownloadURL string
)
client := &http.Client{
Timeout: time.Duration(s.c.HTTPClient.Read.Timeout),
}
bfsDownloadURL = fmt.Sprintf(_downloadURL, s.c.BfsDownloadHost, dp.Bucket, dp.FileName)
if downloadReq, err = http.NewRequest(http.MethodGet, bfsDownloadURL, nil); err != nil {
log.Error("client.NewRequest(%s) error(%v)", bfsDownloadURL, err)
return
}
if resp, err = client.Do(downloadReq); err != nil {
log.Error("client.Do(%v) error(%v)", downloadReq, err)
return
}
contentType = resp.Header.Get("Content-Type")
if downloadBytes, err = ioutil.ReadAll(resp.Body); err != nil {
log.Error("ioutil.ReadAll(%v) error(%v)", resp.Body, err)
return
}
return
}
// upload file to facepri bucket
func (s *Service) upload(dp *model.DeleteParam, contentType string, body []byte) (err error) {
var (
uploadReq *http.Request
resp *http.Response
bfsUploadURL string
)
client := &http.Client{
Timeout: time.Duration(s.c.HTTPClient.Read.Timeout),
}
bfsUploadURL = fmt.Sprintf(_uploadURL, s.c.BfsUpdateHost, dp.Bucket, dp.FileName)
if uploadReq, err = http.NewRequest(http.MethodPut, bfsUploadURL, bytes.NewReader(body)); err != nil {
return
}
auth := s.authorize(_privateBucketAppKey, _privateBucketAppSecret, http.MethodPut, _privateBucket, dp.FileName, time.Now().Unix())
uploadReq.Header.Add("Host", "bfs.bilibili.co")
uploadReq.Header.Add("Date", time.Now().String())
uploadReq.Header.Add("Authorization", auth)
uploadReq.Header.Add("Content-Type", contentType)
uploadReq.Header.Add("Date", fmt.Sprint(time.Now().Unix()))
if resp, err = client.Do(uploadReq); err != nil {
log.Error("client.Do(%v) error(%v)", uploadReq, err)
return
}
// judge response code
switch resp.StatusCode {
case http.StatusOK:
case http.StatusBadRequest:
err = ecode.RequestErr
return
case http.StatusUnauthorized:
// 验证不通过
err = ecode.BfsUploadAuthErr
return
case http.StatusRequestEntityTooLarge:
err = ecode.FileTooLarge
return
case http.StatusNotFound:
err = ecode.NothingFound
return
case http.StatusMethodNotAllowed:
err = ecode.MethodNotAllowed
return
case http.StatusServiceUnavailable:
err = ecode.BfsUploadServiceUnavailable
return
case http.StatusInternalServerError:
err = ecode.ServerErr
return
default:
err = ecode.BfsUploadStatusErr
return
}
code, err := strconv.Atoi(resp.Header.Get("code"))
if err != nil || code != 200 {
err = ecode.BfsUploadCodeErr
return
}
return
}
// delete file in old bucket
func (s *Service) delete(dp *model.DeleteParam) (err error) {
var (
deleteReq *http.Request
resp *http.Response
bfsDeleteURL string
)
client := &http.Client{
Timeout: time.Duration(s.c.HTTPClient.Read.Timeout),
}
bfsDeleteURL = fmt.Sprintf(_deleteURL, s.c.BfsDeleteHost, dp.Bucket, dp.FileName)
if deleteReq, err = http.NewRequest("DELETE", bfsDeleteURL, nil); err != nil {
log.Error("client.NewRequest(%s) error(%v)", bfsDeleteURL, err)
return
}
item, ok := s.bucketCache[dp.Bucket]
if !ok {
err = errors.Wrapf(ecode.NothingFound, "bucket not exist: %s", dp.Bucket)
log.Error("bucket not exist: %s", dp.Bucket)
return
}
deleteReq.Header.Add("Host", "bfs.bilibili.co")
deleteReq.Header.Add("Date", fmt.Sprint(time.Now().Unix()))
deleteReq.Header.Add("Authorization", s.authorize(item.KeyID, item.KeySecret, http.MethodDelete, dp.Bucket, dp.FileName, time.Now().Unix()))
if resp, err = client.Do(deleteReq); err != nil {
log.Error("client.Do(%v) error(%v)", deleteReq, err)
return
}
if resp.StatusCode != 200 {
log.Error("bfs delete error code: %d", resp.StatusCode)
return
}
return
}
// authorize return token
func (s *Service) authorize(key, secret, method, bucket, fileName string, expire int64) (authorization string) {
var (
content string
mac hash.Hash
signature string
)
content = fmt.Sprintf("%s\n%s\n%s\n%d\n", method, bucket, fileName, expire)
mac = hmac.New(sha1.New, []byte(secret))
mac.Write([]byte(content))
signature = base64.StdEncoding.EncodeToString(mac.Sum(nil))
authorization = fmt.Sprintf("%s:%s:%d", key, signature, expire)
return
}
// DeleteV2 deletes a record and delete file in bfs
func (s *Service) DeleteV2(c context.Context, dp *model.DeleteV2Param) (err error) {
switch dp.Status {
case model.PassStatus:
if err = s.orm.Table("upload_yellowing").Where("id=?", dp.Rid).Update("state", model.PassStatus).Update("adminid", dp.AdminID).Error; err != nil {
err = errors.Wrapf(err, "Update(%d,%d,%d)", dp.Rid, model.PassStatus, dp.AdminID)
return
}
case model.DeleteStatus:
record := new(model.Record)
if err = s.orm.Table("upload_yellowing").Where("id=?", dp.Rid).Find(&record).Error; err != nil {
err = errors.Wrapf(err, "Query(%d)", dp.Rid)
return
}
dp.Bucket = record.Bucket
dp.FileName = record.FileName
if err = s.delete(&model.DeleteParam{
Bucket: dp.Bucket,
FileName: dp.FileName,
}); err != nil {
return
}
if err = s.orm.Table("upload_yellowing").Where("id=?", dp.Rid).Update("state", model.DeleteStatus).Update("adminid", dp.AdminID).Error; err != nil {
err = errors.Wrapf(err, "Update(%d,%d,%d)", dp.Rid, model.DeleteStatus, dp.AdminID)
return
}
default:
err = errors.Wrapf(err, "illegal Status(%d,%d,%d)", dp.Rid, model.DeleteStatus, dp.AdminID)
return
}
return
}