util.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. package model
  2. import (
  3. "fmt"
  4. "github.com/allanpk716/ChineseSubFinder/common"
  5. "github.com/go-resty/resty/v2"
  6. "io"
  7. "io/ioutil"
  8. "net/http"
  9. "os"
  10. "path"
  11. "path/filepath"
  12. "regexp"
  13. "strconv"
  14. "strings"
  15. )
  16. // NewHttpClient 新建一个 resty 的对象
  17. func NewHttpClient(_reqParam ...common.ReqParam) *resty.Client {
  18. //const defUserAgent = "Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10_6_8; en-us) AppleWebKit/534.50 (KHTML, like Gecko) Version/5.1 Safari/534.50"
  19. const defUserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36 Edg/91.0.864.41"
  20. var reqParam common.ReqParam
  21. var HttpProxy, UserAgent, Referer string
  22. if len(_reqParam) > 0 {
  23. reqParam = _reqParam[0]
  24. }
  25. if len(reqParam.HttpProxy) > 0 {
  26. HttpProxy = reqParam.HttpProxy
  27. }
  28. if len(reqParam.UserAgent) > 0 {
  29. UserAgent = reqParam.UserAgent
  30. } else {
  31. UserAgent = defUserAgent
  32. }
  33. if len(reqParam.Referer) > 0 {
  34. Referer = reqParam.Referer
  35. }
  36. httpClient := resty.New()
  37. httpClient.SetTimeout(common.HTMLTimeOut)
  38. if HttpProxy != "" {
  39. httpClient.SetProxy(HttpProxy)
  40. }
  41. httpClient.SetHeaders(map[string]string{
  42. "Content-Type": "application/json",
  43. "User-Agent": UserAgent,
  44. })
  45. if len(Referer) > 0 {
  46. httpClient.SetHeader("Referer", Referer)
  47. }
  48. return httpClient
  49. }
  50. // DownFile 从指定的 url 下载文件
  51. func DownFile(urlStr string, _reqParam ...common.ReqParam) ([]byte, string, error) {
  52. var reqParam common.ReqParam
  53. if len(_reqParam) > 0 {
  54. reqParam = _reqParam[0]
  55. }
  56. httpClient := NewHttpClient(reqParam)
  57. resp, err := httpClient.R().Get(urlStr)
  58. if err != nil {
  59. return nil, "", err
  60. }
  61. filename := GetFileName(resp.RawResponse)
  62. return resp.Body(), filename, nil
  63. }
  64. // GetFileName 获取下载文件的文件名
  65. func GetFileName(resp *http.Response) string {
  66. contentDisposition := resp.Header.Get("Content-Disposition")
  67. if len(contentDisposition) == 0 {
  68. return ""
  69. }
  70. re := regexp.MustCompile(`filename=["]*([^"]+)["]*`)
  71. matched := re.FindStringSubmatch(contentDisposition)
  72. if matched == nil || len(matched) == 0 || len(matched[0]) == 0 {
  73. //fmt.Println("######")
  74. return ""
  75. }
  76. return matched[1]
  77. }
  78. // AddBaseUrl 判断传入的 url 是否需要拼接 baseUrl
  79. func AddBaseUrl(baseUrl, url string) string {
  80. if strings.Contains(url, "://") {
  81. return url
  82. }
  83. return fmt.Sprintf("%s%s", baseUrl, url)
  84. }
  85. func GetDebugFolder() (string, error) {
  86. if defDebugFolder == "" {
  87. nowProcessRoot, _ := os.Getwd()
  88. nowProcessRoot = path.Join(nowProcessRoot, common.DebugFolder)
  89. err := os.MkdirAll(nowProcessRoot, os.ModePerm)
  90. if err != nil {
  91. return "", err
  92. }
  93. defDebugFolder = nowProcessRoot
  94. return nowProcessRoot, err
  95. }
  96. return defDebugFolder, nil
  97. }
  98. // GetRootTmpFolder 获取缓存的根目录,每一个视频的缓存将在其中额外新建子集文件夹
  99. func GetRootTmpFolder() (string, error) {
  100. if defTmpFolder == "" {
  101. nowProcessRoot, _ := os.Getwd()
  102. nowProcessRoot = path.Join(nowProcessRoot, common.TmpFolder)
  103. err := os.MkdirAll(nowProcessRoot, os.ModePerm)
  104. if err != nil {
  105. return "", err
  106. }
  107. defTmpFolder = nowProcessRoot
  108. return nowProcessRoot, err
  109. }
  110. return defTmpFolder, nil
  111. }
  112. // ClearRootTmpFolder 清理缓存的根目录,将里面的子文件夹一并清理
  113. func ClearRootTmpFolder() error {
  114. nowTmpFolder, err := GetRootTmpFolder()
  115. if err != nil {
  116. return err
  117. }
  118. pathSep := string(os.PathSeparator)
  119. files, err := ioutil.ReadDir(nowTmpFolder)
  120. if err != nil {
  121. return err
  122. }
  123. for _, curFile := range files {
  124. fullPath := nowTmpFolder + pathSep + curFile.Name()
  125. if curFile.IsDir() {
  126. err = os.RemoveAll(fullPath)
  127. if err != nil {
  128. return err
  129. }
  130. } else {
  131. // 这里就是文件了
  132. err = os.Remove(fullPath)
  133. if err != nil {
  134. return err
  135. }
  136. }
  137. }
  138. return nil
  139. }
  140. // GetTmpFolder 获取缓存的文件夹,没有则新建
  141. func GetTmpFolder(folderName string) (string, error) {
  142. rootPath, err := GetRootTmpFolder()
  143. if err != nil {
  144. return "", err
  145. }
  146. tmpFolderFullPath :=path.Join(rootPath, folderName)
  147. err = os.MkdirAll(tmpFolderFullPath, os.ModePerm)
  148. if err != nil {
  149. return "", err
  150. }
  151. return tmpFolderFullPath, nil
  152. }
  153. // ClearTmpFolder 清理指定的缓存文件夹
  154. func ClearTmpFolder(folderName string) error {
  155. nowTmpFolder, err := GetTmpFolder(folderName)
  156. if err != nil {
  157. return err
  158. }
  159. pathSep := string(os.PathSeparator)
  160. files, err := ioutil.ReadDir(nowTmpFolder)
  161. if err != nil {
  162. return err
  163. }
  164. for _, curFile := range files {
  165. fullPath := nowTmpFolder + pathSep + curFile.Name()
  166. if curFile.IsDir() {
  167. err = os.RemoveAll(fullPath)
  168. if err != nil {
  169. return err
  170. }
  171. } else {
  172. // 这里就是文件了
  173. err = os.Remove(fullPath)
  174. if err != nil {
  175. return err
  176. }
  177. }
  178. }
  179. return nil
  180. }
  181. func CopyFile(dstName, srcName string) (written int64, err error) {
  182. src, err := os.Open(srcName)
  183. if err != nil {
  184. return
  185. }
  186. defer src.Close()
  187. dst, err := os.OpenFile(dstName, os.O_WRONLY|os.O_CREATE, 0644)
  188. if err != nil {
  189. return
  190. }
  191. defer dst.Close()
  192. return io.Copy(dst, src)
  193. }
  194. func IsDir(path string) bool {
  195. s, err := os.Stat(path)
  196. if err != nil {
  197. return false
  198. }
  199. return s.IsDir()
  200. }
  201. // VideoNameSearchKeywordMaker 拼接视频搜索的 title 和 年份
  202. func VideoNameSearchKeywordMaker(title string, year string) string {
  203. iYear, err := strconv.Atoi(year)
  204. if err != nil {
  205. // 允许的错误
  206. GetLogger().Errorln("VideoNameSearchKeywordMaker", "year to int", err)
  207. iYear = 0
  208. }
  209. searchKeyword := title
  210. if iYear >= 2020 {
  211. searchKeyword = searchKeyword + " " + year
  212. }
  213. return searchKeyword
  214. }
  215. // SearchMatchedVideoFile 搜索符合后缀名的视频文件
  216. func SearchMatchedVideoFile(dir string) ([]string, error) {
  217. var fileFullPathList = make([]string, 0)
  218. pathSep := string(os.PathSeparator)
  219. files, err := ioutil.ReadDir(dir)
  220. if err != nil {
  221. return nil, err
  222. }
  223. for _, curFile := range files {
  224. fullPath := dir + pathSep + curFile.Name()
  225. if curFile.IsDir() {
  226. // 内层的错误就无视了
  227. oneList, _ := SearchMatchedVideoFile(fullPath)
  228. if oneList != nil {
  229. fileFullPathList = append(fileFullPathList, oneList...)
  230. }
  231. } else {
  232. // 这里就是文件了
  233. if IsWantedVideoExtDef(curFile.Name()) == true {
  234. fileFullPathList = append(fileFullPathList, fullPath)
  235. }
  236. }
  237. }
  238. return fileFullPathList, nil
  239. }
  240. // SearchMatchedSubFile 搜索符合后缀名的视频文件
  241. func SearchMatchedSubFile(dir string) ([]string, error) {
  242. // 这里有个梗,会出现 __MACOSX 这类文件夹,那么里面会有一样的文件,需要用文件大小排除一下,至少大于 1 kb 吧
  243. var fileFullPathList = make([]string, 0)
  244. pathSep := string(os.PathSeparator)
  245. files, err := ioutil.ReadDir(dir)
  246. if err != nil {
  247. return nil, err
  248. }
  249. for _, curFile := range files {
  250. fullPath := dir + pathSep + curFile.Name()
  251. if curFile.IsDir() {
  252. // 内层的错误就无视了
  253. oneList, _ := SearchMatchedSubFile(fullPath)
  254. if oneList != nil {
  255. fileFullPathList = append(fileFullPathList, oneList...)
  256. }
  257. } else {
  258. // 这里就是文件了
  259. if curFile.Size() < 1000 {
  260. continue
  261. }
  262. if IsSubExtWanted(filepath.Ext(curFile.Name())) == true {
  263. fileFullPathList = append(fileFullPathList, fullPath)
  264. }
  265. }
  266. }
  267. return fileFullPathList, nil
  268. }
  269. // IsWantedVideoExtDef 后缀名是否符合规则
  270. func IsWantedVideoExtDef(fileName string) bool {
  271. // TODO 强制使用固定的视频后缀名匹配列表,后续有需求再考虑额实现外部可配置的列表
  272. if len(wantedExtList) < 1 {
  273. defExtList = append(defExtList, common.VideoExtMp4)
  274. defExtList = append(defExtList, common.VideoExtMkv)
  275. defExtList = append(defExtList, common.VideoExtRmvb)
  276. defExtList = append(defExtList, common.VideoExtIso)
  277. wantedExtList = append(defExtList, common.VideoExtMp4)
  278. wantedExtList = append(defExtList, common.VideoExtMkv)
  279. wantedExtList = append(defExtList, common.VideoExtRmvb)
  280. wantedExtList = append(defExtList, common.VideoExtIso)
  281. }
  282. fileName = strings.ToLower(filepath.Ext(fileName))
  283. for _, s := range wantedExtList {
  284. if s == fileName {
  285. return true
  286. }
  287. }
  288. return false
  289. }
  290. func GetEpisodeKeyName(season, eps int) string {
  291. return "S" + strconv.Itoa(season) + "E" +strconv.Itoa(eps)
  292. }
  293. var (
  294. defDebugFolder = ""
  295. defTmpFolder = ""
  296. wantedExtList = make([]string, 0) // 人工确认的需要监控的视频后缀名
  297. defExtList = make([]string, 0) // 内置支持的视频后缀名列表
  298. )