unarchiveFile.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. package model
  2. import (
  3. "archive/tar"
  4. "archive/zip"
  5. "bytes"
  6. "golang.org/x/text/encoding/simplifiedchinese"
  7. "golang.org/x/text/transform"
  8. "io/ioutil"
  9. "compress/flate"
  10. "errors"
  11. "github.com/gen2brain/go-unarr"
  12. "github.com/go-rod/rod/lib/utils"
  13. "github.com/mholt/archiver/v3"
  14. "io"
  15. "path"
  16. "path/filepath"
  17. "strings"
  18. "unicode/utf8"
  19. )
  20. // UnArchiveFile 7z 以外的都能搞定中文编码的问题,但是 7z 有梗,需要单独的库去解析,且编码是解决不了的,以后他们搞定了再测试
  21. // 所以效果就是,7z 外的压缩包文件解压ok,字幕可以正常从名称解析出是简体还是繁体,但是7z就没办法了,一定乱码
  22. func UnArchiveFile(fileFullPath, desRootPath string) error {
  23. switch filepath.Ext(strings.ToLower(fileFullPath)) {
  24. case ".zip":
  25. z := archiver.Zip{
  26. CompressionLevel: flate.DefaultCompression,
  27. MkdirAll: true,
  28. SelectiveCompression: true,
  29. ContinueOnError: false,
  30. OverwriteExisting: false,
  31. ImplicitTopLevelFolder: false,
  32. }
  33. err := z.Walk(fileFullPath, func(f archiver.File) error {
  34. if f.IsDir() == true {
  35. return nil
  36. }
  37. zfh, ok := f.Header.(zip.FileHeader)
  38. if ok {
  39. err := processOneFile(f, zfh.NonUTF8, desRootPath)
  40. if err != nil {
  41. return err
  42. }
  43. }
  44. return nil
  45. })
  46. if err != nil {
  47. return err
  48. }
  49. case ".tar":
  50. z := archiver.Tar{
  51. MkdirAll: true,
  52. ContinueOnError: false,
  53. OverwriteExisting: false,
  54. ImplicitTopLevelFolder: false,
  55. }
  56. err := z.Walk(fileFullPath, func(f archiver.File) error {
  57. if f.IsDir() == true {
  58. return nil
  59. }
  60. zfh, ok := f.Header.(tar.Header)
  61. if ok {
  62. err := processOneFile(f, utf8.Valid([]byte(zfh.Name)), desRootPath)
  63. if err != nil {
  64. return err
  65. }
  66. }
  67. return nil
  68. })
  69. if err != nil {
  70. return err
  71. }
  72. case ".rar":
  73. z := archiver.Rar{
  74. MkdirAll: true,
  75. ContinueOnError: false,
  76. OverwriteExisting: false,
  77. ImplicitTopLevelFolder: false,
  78. }
  79. err := z.Walk(fileFullPath, func(f archiver.File) error {
  80. if f.IsDir() == true {
  81. return nil
  82. }
  83. zfh, ok := f.Header.(tar.Header)
  84. if ok {
  85. err := processOneFile(f, utf8.Valid([]byte(zfh.Name)), desRootPath)
  86. if err != nil {
  87. return err
  88. }
  89. }
  90. return nil
  91. })
  92. if err != nil {
  93. return err
  94. }
  95. case ".7z":
  96. return unArr7z(fileFullPath, desRootPath)
  97. default:
  98. return errors.New("not support un archive file ext")
  99. }
  100. return nil
  101. }
  102. func processOneFile(f archiver.File, notUTF8 bool, desRootPath string) error {
  103. decodeName := f.Name()
  104. if notUTF8 == true {
  105. //ouBytes, err := ChangeFileCoding2UTF8([]byte(f.Name()))
  106. //if err != nil {
  107. // return err
  108. //}
  109. i := bytes.NewReader([]byte(f.Name()))
  110. decoder := transform.NewReader(i, simplifiedchinese.GB18030.NewDecoder())
  111. content, _ := ioutil.ReadAll(decoder)
  112. decodeName = string(content)
  113. //decodeName = string(ouBytes)
  114. }
  115. var chunk []byte
  116. buf := make([]byte, 1024)
  117. for {
  118. n, err := f.Read(buf)
  119. if err != nil && err != io.EOF {
  120. return err
  121. }
  122. //说明读取结束
  123. if n == 0 {
  124. break
  125. }
  126. //读取到最终的缓冲区中
  127. chunk = append(chunk, buf[:n]...)
  128. }
  129. err := utils.OutputFile(path.Join(desRootPath, decodeName), chunk)
  130. if err != nil {
  131. return err
  132. }
  133. return nil
  134. }
  135. func unArr7z(fileFullPath, desRootPath string) error {
  136. a, err := unarr.NewArchive(fileFullPath)
  137. if err != nil {
  138. return err
  139. }
  140. defer a.Close()
  141. for {
  142. err := a.Entry()
  143. if err != nil {
  144. if err == io.EOF {
  145. break
  146. } else {
  147. return err
  148. }
  149. }
  150. data, err := a.ReadAll()
  151. if err != nil {
  152. return err
  153. }
  154. decodeName := a.Name()
  155. decodeName = filepath.Base(decodeName)
  156. err = utils.OutputFile(path.Join(desRootPath, decodeName), data)
  157. if err != nil {
  158. return err
  159. }
  160. }
  161. return nil
  162. }