You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

package.go 10 kB

7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. package http
  2. import (
  3. "archive/tar"
  4. "archive/zip"
  5. "fmt"
  6. "io"
  7. "mime/multipart"
  8. "net/http"
  9. "net/url"
  10. "path/filepath"
  11. "github.com/gin-gonic/gin"
  12. "gitlink.org.cn/cloudream/common/consts/errorcode"
  13. "gitlink.org.cn/cloudream/common/pkgs/logger"
  14. "gitlink.org.cn/cloudream/jcs-pub/client/internal/downloader"
  15. "gitlink.org.cn/cloudream/jcs-pub/client/internal/http/types"
  16. cliapi "gitlink.org.cn/cloudream/jcs-pub/client/sdk/api/v1"
  17. clitypes "gitlink.org.cn/cloudream/jcs-pub/client/types"
  18. "gitlink.org.cn/cloudream/jcs-pub/common/ecode"
  19. "gitlink.org.cn/cloudream/jcs-pub/common/pkgs/iterator"
  20. )
  21. // PackageService 包服务,负责处理包相关的HTTP请求。
  22. type PackageService struct {
  23. *Server
  24. }
  25. // Package 返回PackageService的实例。
  26. func (s *Server) Package() *PackageService {
  27. return &PackageService{
  28. Server: s,
  29. }
  30. }
  31. func (s *PackageService) Get(ctx *gin.Context) {
  32. log := logger.WithField("HTTP", "Package.Get")
  33. var req cliapi.PackageGet
  34. if err := ctx.ShouldBindQuery(&req); err != nil {
  35. log.Warnf("binding body: %s", err.Error())
  36. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  37. return
  38. }
  39. pkg, err := s.svc.PackageSvc().Get(req.PackageID)
  40. if err != nil {
  41. log.Warnf("getting package: %s", err.Error())
  42. ctx.JSON(http.StatusOK, types.FailedError(err))
  43. return
  44. }
  45. ctx.JSON(http.StatusOK, types.OK(cliapi.PackageGetResp{Package: pkg}))
  46. }
  47. func (s *PackageService) GetByFullName(ctx *gin.Context) {
  48. log := logger.WithField("HTTP", "Package.GetByFullName")
  49. var req cliapi.PackageGetByFullName
  50. if err := ctx.ShouldBindQuery(&req); err != nil {
  51. log.Warnf("binding query: %s", err.Error())
  52. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  53. return
  54. }
  55. pkg, err := s.svc.PackageSvc().GetByFullName(req.BucketName, req.PackageName)
  56. if err != nil {
  57. log.Warnf("getting package by name: %s", err.Error())
  58. ctx.JSON(http.StatusOK, types.FailedError(err))
  59. return
  60. }
  61. ctx.JSON(http.StatusOK, types.OK(cliapi.PackageGetByFullNameResp{Package: pkg}))
  62. }
  63. // Create 处理创建新包的HTTP请求。
  64. func (s *PackageService) Create(ctx *gin.Context) {
  65. log := logger.WithField("HTTP", "Package.Create")
  66. var req cliapi.PackageCreate
  67. if err := ctx.ShouldBindJSON(&req); err != nil {
  68. log.Warnf("binding body: %s", err.Error())
  69. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  70. return
  71. }
  72. pkg, err := s.svc.PackageSvc().Create(req.BucketID, req.Name)
  73. if err != nil {
  74. log.Warnf("creating package: %s", err.Error())
  75. ctx.JSON(http.StatusOK, types.FailedError(err))
  76. return
  77. }
  78. ctx.JSON(http.StatusOK, types.OK(cliapi.PackageCreateResp{
  79. Package: pkg,
  80. }))
  81. }
  82. type PackageCreateUpload struct {
  83. Info cliapi.PackageCreateUploadInfo `form:"info" binding:"required"`
  84. Files []*multipart.FileHeader `form:"files"`
  85. }
  86. func (s *PackageService) CreateLoad(ctx *gin.Context) {
  87. log := logger.WithField("HTTP", "Package.CreateUpload")
  88. var req PackageCreateUpload
  89. if err := ctx.ShouldBind(&req); err != nil {
  90. log.Warnf("binding body: %s", err.Error())
  91. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  92. return
  93. }
  94. if len(req.Info.CopyTo) != len(req.Info.CopyToPath) {
  95. log.Warnf("CopyTo and CopyToPath count not match")
  96. ctx.JSON(http.StatusOK, types.Failed(ecode.BadArgument, "CopyTo and CopyToPath count not match"))
  97. return
  98. }
  99. copyToPath := make([]clitypes.JPath, 0, len(req.Info.CopyToPath))
  100. for _, p := range req.Info.CopyToPath {
  101. copyToPath = append(copyToPath, clitypes.PathFromJcsPathString(p))
  102. }
  103. up, err := s.svc.Uploader.BeginCreateUpload(req.Info.BucketID, req.Info.Name, req.Info.CopyTo, copyToPath)
  104. if err != nil {
  105. log.Warnf("begin package create upload: %s", err.Error())
  106. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, "%v", err))
  107. return
  108. }
  109. defer up.Abort()
  110. var pathes []string
  111. for _, file := range req.Files {
  112. f, err := file.Open()
  113. if err != nil {
  114. log.Warnf("open file: %s", err.Error())
  115. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, fmt.Sprintf("open file %v: %v", file.Filename, err)))
  116. return
  117. }
  118. path, err := url.PathUnescape(file.Filename)
  119. if err != nil {
  120. log.Warnf("unescape filename: %s", err.Error())
  121. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, fmt.Sprintf("unescape filename %v: %v", file.Filename, err)))
  122. return
  123. }
  124. path = filepath.ToSlash(path)
  125. err = up.Upload(clitypes.PathFromJcsPathString(path), f)
  126. if err != nil {
  127. log.Warnf("uploading file: %s", err.Error())
  128. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, fmt.Sprintf("uploading file %v: %v", file.Filename, err)))
  129. return
  130. }
  131. pathes = append(pathes, path)
  132. }
  133. ret, err := up.Commit()
  134. if err != nil {
  135. log.Warnf("commit create upload: %s", err.Error())
  136. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, fmt.Sprintf("commit create upload: %v", err)))
  137. return
  138. }
  139. objs := make([]clitypes.Object, len(pathes))
  140. for i := range pathes {
  141. objs[i] = ret.Objects[pathes[i]]
  142. }
  143. ctx.JSON(http.StatusOK, types.OK(cliapi.PackageCreateUploadResp{Package: ret.Package, Objects: objs}))
  144. }
  145. func (s *PackageService) Download(ctx *gin.Context) {
  146. log := logger.WithField("HTTP", "Package.Download")
  147. var req cliapi.PackageDownload
  148. if err := ctx.ShouldBindQuery(&req); err != nil {
  149. log.Warnf("binding query: %s", err.Error())
  150. ctx.JSON(http.StatusBadRequest, types.Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  151. return
  152. }
  153. pkg, iter, err := s.svc.Downloader.DownloadPackage(req.PackageID, req.Prefix)
  154. if err != nil {
  155. log.Warnf("downloading package: %s", err.Error())
  156. ctx.JSON(http.StatusOK, types.Failed(errorcode.OperationFailed, err.Error()))
  157. return
  158. }
  159. defer iter.Close()
  160. if req.Zip {
  161. s.downloadZip(ctx, req, pkg, iter)
  162. } else {
  163. s.downloadTar(ctx, req, pkg, iter)
  164. }
  165. }
  166. func (s *PackageService) downloadZip(ctx *gin.Context, req cliapi.PackageDownload, pkg clitypes.Package, iter downloader.DownloadIterator) {
  167. log := logger.WithField("HTTP", "Package.Download")
  168. ctx.Header("Content-Disposition", "attachment; filename="+url.PathEscape(pkg.Name)+".zip")
  169. ctx.Header("Content-Type", "application/zip")
  170. ctx.Header("Content-Transfer-Encoding", "binary")
  171. zipFile := zip.NewWriter(ctx.Writer)
  172. defer zipFile.Close()
  173. for {
  174. item, err := iter.MoveNext()
  175. if err == iterator.ErrNoMoreItem {
  176. return
  177. }
  178. if err != nil {
  179. log.Warnf("iterating next object: %v", err)
  180. return
  181. }
  182. filePath := item.Object.Path
  183. if req.Prefix != "" && req.NewPrefix != nil {
  184. filePath = *req.NewPrefix + filePath[len(req.Prefix):]
  185. }
  186. zf, err := zipFile.Create(filePath)
  187. if err != nil {
  188. log.Warnf("creating zip file: %v", err)
  189. item.File.Close()
  190. return
  191. }
  192. _, err = io.Copy(zf, item.File)
  193. if err != nil {
  194. log.Warnf("copying file to zip: %v", err)
  195. item.File.Close()
  196. return
  197. }
  198. item.File.Close()
  199. }
  200. }
  201. func (s *PackageService) downloadTar(ctx *gin.Context, req cliapi.PackageDownload, pkg clitypes.Package, iter downloader.DownloadIterator) {
  202. log := logger.WithField("HTTP", "Package.Download")
  203. ctx.Header("Content-Disposition", "attachment; filename="+url.PathEscape(pkg.Name)+".tar")
  204. ctx.Header("Content-Type", "application/x-tar")
  205. ctx.Header("Content-Transfer-Encoding", "binary")
  206. tarFile := tar.NewWriter(ctx.Writer)
  207. defer tarFile.Close()
  208. for {
  209. item, err := iter.MoveNext()
  210. if err == iterator.ErrNoMoreItem {
  211. return
  212. }
  213. if err != nil {
  214. log.Warnf("iterating next object: %v", err)
  215. return
  216. }
  217. filePath := item.Object.Path
  218. if req.Prefix != "" && req.NewPrefix != nil {
  219. filePath = *req.NewPrefix + filePath[len(req.Prefix):]
  220. }
  221. err = tarFile.WriteHeader(&tar.Header{
  222. Typeflag: tar.TypeReg,
  223. Name: filePath,
  224. Size: item.Object.Size,
  225. ModTime: item.Object.CreateTime,
  226. })
  227. if err != nil {
  228. log.Warnf("creating tar header: %v", err)
  229. item.File.Close()
  230. return
  231. }
  232. _, err = io.Copy(tarFile, item.File)
  233. if err != nil {
  234. log.Warnf("copying file to tar: %v", err)
  235. item.File.Close()
  236. return
  237. }
  238. item.File.Close()
  239. }
  240. }
  241. func (s *PackageService) Delete(ctx *gin.Context) {
  242. log := logger.WithField("HTTP", "Package.Delete")
  243. var req cliapi.PackageDelete
  244. if err := ctx.ShouldBindJSON(&req); err != nil {
  245. log.Warnf("binding body: %s", err.Error())
  246. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  247. return
  248. }
  249. err := s.svc.PackageSvc().DeletePackage(req.PackageID)
  250. if err != nil {
  251. log.Warnf("deleting package: %s", err.Error())
  252. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, "delete package failed"))
  253. return
  254. }
  255. ctx.JSON(http.StatusOK, types.OK(nil))
  256. }
  257. func (s *PackageService) Clone(ctx *gin.Context) {
  258. log := logger.WithField("HTTP", "Package.Clone")
  259. var req cliapi.PackageClone
  260. if err := ctx.ShouldBindJSON(&req); err != nil {
  261. log.Warnf("binding body: %s", err.Error())
  262. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  263. return
  264. }
  265. pkg, err := s.svc.PackageSvc().Clone(req.PackageID, req.BucketID, req.Name)
  266. if err != nil {
  267. log.Warnf("cloning package: %s", err.Error())
  268. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, "clone package failed"))
  269. return
  270. }
  271. ctx.JSON(http.StatusOK, types.OK(cliapi.PackageCloneResp{
  272. Package: pkg,
  273. }))
  274. }
  275. func (s *PackageService) ListBucketPackages(ctx *gin.Context) {
  276. log := logger.WithField("HTTP", "Package.ListBucketPackages")
  277. var req cliapi.PackageListBucketPackages
  278. if err := ctx.ShouldBindQuery(&req); err != nil {
  279. log.Warnf("binding query: %s", err.Error())
  280. ctx.JSON(http.StatusBadRequest, types.Failed(ecode.BadArgument, "missing argument or invalid argument"))
  281. return
  282. }
  283. pkgs, err := s.svc.PackageSvc().GetBucketPackages(req.BucketID)
  284. if err != nil {
  285. log.Warnf("getting bucket packages: %s", err.Error())
  286. ctx.JSON(http.StatusOK, types.Failed(ecode.OperationFailed, "get bucket packages failed"))
  287. return
  288. }
  289. ctx.JSON(http.StatusOK, types.OK(cliapi.PackageListBucketPackagesResp{
  290. Packages: pkgs,
  291. }))
  292. }

本项目旨在将云际存储公共基础设施化,使个人及企业可低门槛使用高效的云际存储服务(安装开箱即用云际存储客户端即可,无需关注其他组件的部署),同时支持用户灵活便捷定制云际存储的功能细节。