dao.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316
  1. package bfs
  2. import (
  3. "bytes"
  4. "context"
  5. "crypto/hmac"
  6. "crypto/sha1"
  7. "encoding/base64"
  8. "errors"
  9. "fmt"
  10. "hash"
  11. "io"
  12. "io/ioutil"
  13. "net"
  14. "net/http"
  15. nurl "net/url"
  16. "strconv"
  17. "strings"
  18. "time"
  19. "go-common/app/interface/main/creative/conf"
  20. "go-common/library/ecode"
  21. "go-common/library/log"
  22. )
  23. const (
  24. _bucket = "archive"
  25. _url = "http://bfs.bilibili.co/bfs/archive/"
  26. _method = "PUT"
  27. _key = "8d4e593ba7555502"
  28. _secret = "0bdbd4c7caeeddf587c3c4daec0475"
  29. )
  30. var (
  31. errUpload = errors.New("Upload failed")
  32. errDownload = errors.New("Download out image link failed")
  33. )
  34. // Dao is bfs dao.
  35. type Dao struct {
  36. c *conf.Config
  37. client *http.Client
  38. captureCli *http.Client
  39. }
  40. // New new a bfs dao.
  41. func New(c *conf.Config) (d *Dao) {
  42. d = &Dao{
  43. c: c,
  44. client: &http.Client{
  45. Timeout: time.Duration(c.BFS.Timeout),
  46. },
  47. captureCli: &http.Client{
  48. Timeout: time.Duration(time.Second),
  49. CheckRedirect: func(req *http.Request, via []*http.Request) error {
  50. return ecode.RequestErr
  51. },
  52. },
  53. }
  54. return d
  55. }
  56. // Upload upload bfs.
  57. func (d *Dao) Upload(c context.Context, fileType string, bs []byte) (location string, err error) {
  58. req, err := http.NewRequest(d.c.BFS.Method, d.c.BFS.URL, bytes.NewBuffer(bs))
  59. if err != nil {
  60. log.Error("http.NewRequest error (%v) | fileType(%s)", err, fileType)
  61. return
  62. }
  63. expire := time.Now().Unix()
  64. authorization := authorize(d.c.BFS.Key, d.c.BFS.Secret, d.c.BFS.Method, d.c.BFS.Bucket, expire)
  65. req.Header.Set("Host", d.c.BFS.URL)
  66. req.Header.Add("Date", fmt.Sprint(expire))
  67. req.Header.Add("Authorization", authorization)
  68. req.Header.Add("Content-Type", fileType)
  69. // timeout
  70. ctx, cancel := context.WithTimeout(c, time.Duration(d.c.BFS.Timeout))
  71. req = req.WithContext(ctx)
  72. defer cancel()
  73. resp, err := d.client.Do(req)
  74. if err != nil {
  75. log.Error("d.Client.Do error(%v) | url(%s)", err, d.c.BFS.URL)
  76. err = ecode.BfsUploadServiceUnavailable
  77. return
  78. }
  79. if resp.StatusCode != http.StatusOK {
  80. log.Error("Upload http.StatusCode nq http.StatusOK (%d) | url(%s)", resp.StatusCode, d.c.BFS.URL)
  81. err = errUpload
  82. return
  83. }
  84. header := resp.Header
  85. code := header.Get("Code")
  86. if code != strconv.Itoa(http.StatusOK) {
  87. log.Error("strconv.Itoa err, code(%s) | url(%s)", code, d.c.BFS.URL)
  88. err = errUpload
  89. return
  90. }
  91. location = header.Get("Location")
  92. return
  93. }
  94. // UploadArc upload bfs to archive bucket.
  95. func (d *Dao) UploadArc(c context.Context, fileType string, body io.Reader) (location string, err error) {
  96. req, err := http.NewRequest(_method, _url, body)
  97. if err != nil {
  98. log.Error("http.NewRequest error (%v) | fileType(%s)", err, fileType)
  99. return
  100. }
  101. expire := time.Now().Unix()
  102. authorization := authorize(_key, _secret, _method, _bucket, expire)
  103. req.Header.Set("Host", _url)
  104. req.Header.Add("Date", fmt.Sprint(expire))
  105. req.Header.Add("Authorization", authorization)
  106. req.Header.Add("Content-Type", fileType)
  107. // timeout
  108. c, cancel := context.WithTimeout(c, time.Duration(d.c.BFS.Timeout))
  109. req = req.WithContext(c)
  110. defer cancel()
  111. resp, err := d.client.Do(req)
  112. if err != nil {
  113. log.Error("d.Client.Do error(%v) | url(%s)", err, _url)
  114. err = ecode.BfsUploadServiceUnavailable
  115. return
  116. }
  117. if resp.StatusCode != http.StatusOK {
  118. log.Error("Upload http.StatusCode nq http.StatusOK (%d) | url(%s)", resp.StatusCode, _url)
  119. err = errUpload
  120. return
  121. }
  122. header := resp.Header
  123. code := header.Get("Code")
  124. if code != strconv.Itoa(http.StatusOK) {
  125. log.Error("strconv.Itoa err, code(%s) | url(%s)", code, _url)
  126. err = errUpload
  127. return
  128. }
  129. location = header.Get("Location")
  130. return
  131. }
  132. // authorize returns authorization for upload file to bfs
  133. func authorize(key, secret, method, bucket string, expire int64) (authorization string) {
  134. var (
  135. content string
  136. mac hash.Hash
  137. signature string
  138. )
  139. content = fmt.Sprintf("%s\n%s\n\n%d\n", method, bucket, expire)
  140. mac = hmac.New(sha1.New, []byte(secret))
  141. mac.Write([]byte(content))
  142. signature = base64.StdEncoding.EncodeToString(mac.Sum(nil))
  143. authorization = fmt.Sprintf("%s:%s:%d", key, signature, expire)
  144. return
  145. }
  146. // UploadByFile upload local img file.
  147. func (d *Dao) UploadByFile(c context.Context, imgpath string) (location string, err error) {
  148. data, err := ioutil.ReadFile(imgpath)
  149. if err != nil {
  150. log.Error("UploadByFile ioutil.ReadFile error (%v) | imgpath(%s)", err, imgpath)
  151. return
  152. }
  153. fileType := http.DetectContentType(data)
  154. if fileType != "image/jpeg" && fileType != "image/png" {
  155. log.Error("file type not allow file type(%s)", fileType)
  156. err = ecode.CreativeArticleImageTypeErr
  157. }
  158. body := new(bytes.Buffer)
  159. _, err = body.Write(data)
  160. if err != nil {
  161. log.Error("body.Write error (%v)", err)
  162. return
  163. }
  164. req, err := http.NewRequest(_method, _url, body)
  165. if err != nil {
  166. log.Error("http.NewRequest error (%v) | fileType(%s)", err, fileType)
  167. return
  168. }
  169. expire := time.Now().Unix()
  170. authorization := authorize(_key, _secret, _method, _bucket, expire)
  171. req.Header.Set("Host", _url)
  172. req.Header.Add("Date", fmt.Sprint(expire))
  173. req.Header.Add("Authorization", authorization)
  174. req.Header.Add("Content-Type", fileType)
  175. // timeout
  176. c, cancel := context.WithTimeout(c, time.Duration(d.c.BFS.Timeout))
  177. req = req.WithContext(c)
  178. defer cancel()
  179. resp, err := d.client.Do(req)
  180. if err != nil {
  181. log.Error("d.Client.Do error(%v) | url(%s)", err, _url)
  182. err = ecode.BfsUploadServiceUnavailable
  183. return
  184. }
  185. if resp.StatusCode != http.StatusOK {
  186. log.Error("Upload http.StatusCode nq http.StatusOK (%d) | url(%s)", resp.StatusCode, _url)
  187. err = errUpload
  188. return
  189. }
  190. header := resp.Header
  191. code := header.Get("Code")
  192. if code != strconv.Itoa(http.StatusOK) {
  193. log.Error("strconv.Itoa err, code(%s) | url(%s)", code, _url)
  194. err = errUpload
  195. return
  196. }
  197. location = header.Get("Location")
  198. return
  199. }
  200. //Capture performs a HTTP Get request for the image url and upload bfs.
  201. func (d *Dao) Capture(c context.Context, url string) (loc string, size int, err error) {
  202. if err = checkURL(url); err != nil {
  203. return
  204. }
  205. bs, ct, err := d.download(c, url)
  206. if err != nil {
  207. return
  208. }
  209. size = len(bs)
  210. if size == 0 {
  211. log.Error("capture image size(%d)|url(%s)", size, url)
  212. return
  213. }
  214. if ct != "image/jpeg" && ct != "image/jpg" && ct != "image/png" && ct != "image/gif" {
  215. log.Error("capture not allow image file type(%s)", ct)
  216. err = ecode.CreativeArticleImageTypeErr
  217. return
  218. }
  219. loc, err = d.Upload(c, ct, bs)
  220. return loc, size, err
  221. }
  222. func (d *Dao) download(c context.Context, url string) (bs []byte, ct string, err error) {
  223. req, err := http.NewRequest("GET", url, nil)
  224. if err != nil {
  225. log.Error("capture http.NewRequest error(%v)|url (%s)", err, url)
  226. return
  227. }
  228. // timeout
  229. ctx, cancel := context.WithTimeout(c, 800*time.Millisecond)
  230. req = req.WithContext(ctx)
  231. defer cancel()
  232. resp, err := d.captureCli.Do(req)
  233. if err != nil {
  234. log.Error("capture d.client.Do error(%v)|url(%s)", err, url)
  235. return
  236. }
  237. defer resp.Body.Close()
  238. if resp.StatusCode != http.StatusOK {
  239. log.Error("capture http.StatusCode nq http.StatusOK(%d)|url(%s)", resp.StatusCode, url)
  240. err = errDownload
  241. return
  242. }
  243. if bs, err = ioutil.ReadAll(resp.Body); err != nil {
  244. log.Error("capture ioutil.ReadAll error(%v)", err)
  245. err = errDownload
  246. return
  247. }
  248. ct = http.DetectContentType(bs)
  249. return
  250. }
  251. func checkURL(url string) (err error) {
  252. // http || https
  253. if !strings.HasPrefix(url, "http://") && !strings.HasPrefix(url, "https://") {
  254. log.Error("capture url invalid(%s)", url)
  255. err = ecode.RequestErr
  256. return
  257. }
  258. u, err := nurl.Parse(url)
  259. if err != nil {
  260. log.Error("capture url.Parse error(%v)", err)
  261. err = ecode.RequestErr
  262. return
  263. }
  264. // make sure ip is public. avoid ssrf
  265. ips, err := net.LookupIP(u.Host) // take from 1st argument
  266. if err != nil {
  267. log.Error("capture url(%s) LookupIP failed", url)
  268. err = ecode.RequestErr
  269. return
  270. }
  271. if len(ips) == 0 {
  272. log.Error("capture url(%s) LookupIP length 0", url)
  273. err = ecode.RequestErr
  274. return
  275. }
  276. for _, v := range ips {
  277. if !isPublicIP(v) {
  278. log.Error("capture url(%s) is not public ip(%v)", url, v)
  279. err = ecode.RequestErr
  280. return
  281. }
  282. }
  283. return
  284. }
  285. func isPublicIP(IP net.IP) bool {
  286. if IP.IsLoopback() || IP.IsLinkLocalMulticast() || IP.IsLinkLocalUnicast() {
  287. return false
  288. }
  289. if ip4 := IP.To4(); ip4 != nil {
  290. switch true {
  291. case ip4[0] == 10:
  292. return false
  293. case ip4[0] == 172 && ip4[1] >= 16 && ip4[1] <= 31:
  294. return false
  295. case ip4[0] == 192 && ip4[1] == 168:
  296. return false
  297. default:
  298. return true
  299. }
  300. }
  301. return false
  302. }