commit_archive.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. // Copyright 2015 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package git
  5. import (
  6. "archive/zip"
  7. "fmt"
  8. "io"
  9. "os"
  10. "path/filepath"
  11. "strings"
  12. "github.com/G-Node/gogs/pkg/setting"
  13. )
  14. type ArchiveType int
  15. const (
  16. ZIP ArchiveType = iota + 1
  17. TARGZ
  18. GIN
  19. )
  20. func (c *Commit) CreateArchive(target string, archiveType ArchiveType, cloneL string) error {
  21. var format string
  22. switch archiveType {
  23. case ZIP:
  24. format = "zip"
  25. case TARGZ:
  26. format = "tar.gz"
  27. case GIN:
  28. to := filepath.Join(setting.Repository.Upload.TempPath, "archives", filepath.Base(strings.TrimSuffix(c.repo.Path, ".git")))
  29. defer os.RemoveAll(to)
  30. _, err := NewCommand("clone", c.repo.Path, to).RunTimeout(-1)
  31. if err != nil {
  32. return err
  33. }
  34. _, err = NewCommand("remote", "set-url", "origin", cloneL).RunInDirTimeout(-1, to)
  35. if err != nil {
  36. return err
  37. }
  38. fp, err := os.Create(target)
  39. defer fp.Close()
  40. if err != nil {
  41. return err
  42. }
  43. err = mkzip(to, fp)
  44. return err
  45. default:
  46. return fmt.Errorf("unknown format: %v", archiveType)
  47. }
  48. _, err := NewCommand("archive", "--prefix="+filepath.Base(strings.TrimSuffix(c.repo.Path, ".git"))+"/", "--format="+format, "-o", target, c.ID.String()).RunInDir(c.repo.Path)
  49. return err
  50. }
  51. // NOTE: TEMPORARY COPY FROM gin-doi
  52. // mkzip walks the directory tree rooted at src and writes each file found to the given writers.
  53. // The function accepts multiple writers to allow for multiple outputs (e.g., a file or md5 hash).
  54. func mkzip(src string, writers ...io.Writer) error {
  55. // ensure the src actually exists before trying to zip it
  56. if _, err := os.Stat(src); err != nil {
  57. return fmt.Errorf("Unable to zip files: %s", err.Error())
  58. }
  59. mw := io.MultiWriter(writers...)
  60. tw := zip.NewWriter(mw)
  61. defer tw.Close()
  62. // walk path
  63. return filepath.Walk(src, func(file string, fi os.FileInfo, err error) error {
  64. // return on any error
  65. if err != nil {
  66. return err
  67. }
  68. // create a new dir/file header
  69. header, err := zip.FileInfoHeader(fi)
  70. if err != nil {
  71. return err
  72. }
  73. // update the name to correctly reflect the desired destination when unzipping
  74. header.Name = strings.TrimPrefix(strings.Replace(file, src, "", -1), string(filepath.Separator))
  75. // write the header
  76. w, err := tw.CreateHeader(header)
  77. if err != nil {
  78. return err
  79. }
  80. // return on directories since there will be no content to zip
  81. if fi.Mode().IsDir() {
  82. return nil
  83. }
  84. mode := fi.Mode()
  85. fmt.Print(mode)
  86. if fi.Mode()&os.ModeSymlink != 0 {
  87. data, err := os.Readlink(file)
  88. if err != nil {
  89. return err
  90. }
  91. if _, err := io.Copy(w, strings.NewReader(data)); err != nil {
  92. return err
  93. }
  94. return nil
  95. }
  96. // open files for zipping
  97. f, err := os.Open(file)
  98. defer f.Close()
  99. if err != nil {
  100. return err
  101. }
  102. // copy file data into zip writer
  103. if _, err := io.Copy(w, f); err != nil {
  104. return err
  105. }
  106. return nil
  107. })
  108. }