metadata.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. package metadata
  2. import (
  3. "encoding/base64"
  4. "encoding/json"
  5. "errors"
  6. "log"
  7. "net/http"
  8. "os"
  9. "path/filepath"
  10. "strings"
  11. "sync"
  12. "time"
  13. "github.com/gorilla/websocket"
  14. "imuslab.com/arozos/mod/filesystem"
  15. "imuslab.com/arozos/mod/filesystem/fssort"
  16. hidden "imuslab.com/arozos/mod/filesystem/hidden"
  17. "imuslab.com/arozos/mod/utils"
  18. )
  19. /*
  20. This package is used to extract meta data from files like mp3 and mp4
  21. Also support image caching
  22. */
  23. type RenderHandler struct {
  24. renderingFiles sync.Map
  25. renderingFolder sync.Map
  26. }
  27. // Create a new RenderHandler
  28. func NewRenderHandler() *RenderHandler {
  29. return &RenderHandler{
  30. renderingFiles: sync.Map{},
  31. renderingFolder: sync.Map{},
  32. }
  33. }
  34. // Build cache for all files (non recursive) for the given filepath
  35. func (rh *RenderHandler) BuildCacheForFolder(fsh *filesystem.FileSystemHandler, vpath string, username string) error {
  36. fshAbs := fsh.FileSystemAbstraction
  37. rpath, _ := fshAbs.VirtualPathToRealPath(vpath, username)
  38. //Get a list of all files inside this path
  39. fis, err := fshAbs.ReadDir(filepath.ToSlash(filepath.Clean(rpath)))
  40. if err != nil {
  41. return err
  42. }
  43. for _, fi := range fis {
  44. //Load Cache in generate mode
  45. rh.LoadCache(fsh, filepath.Join(rpath, fi.Name()), true)
  46. }
  47. //Check if the cache folder has file. If not, remove it
  48. cachedFiles, _ := fshAbs.ReadDir(filepath.ToSlash(filepath.Join(filepath.Clean(rpath), "/.metadata/.cache/")))
  49. if len(cachedFiles) == 0 {
  50. fshAbs.RemoveAll(filepath.ToSlash(filepath.Join(filepath.Clean(rpath), "/.metadata/.cache/")) + "/")
  51. }
  52. return nil
  53. }
  54. func (rh *RenderHandler) LoadCacheAsBytes(fsh *filesystem.FileSystemHandler, vpath string, username string, generateOnly bool) ([]byte, error) {
  55. fshAbs := fsh.FileSystemAbstraction
  56. rpath, _ := fshAbs.VirtualPathToRealPath(vpath, username)
  57. b64, err := rh.LoadCache(fsh, rpath, generateOnly)
  58. if err != nil {
  59. return []byte{}, err
  60. }
  61. resultingBytes, _ := base64.StdEncoding.DecodeString(b64)
  62. return resultingBytes, nil
  63. }
  64. // Try to load a cache from file. If not exists, generate it now
  65. func (rh *RenderHandler) LoadCache(fsh *filesystem.FileSystemHandler, rpath string, generateOnly bool) (string, error) {
  66. //Create a cache folder
  67. fshAbs := fsh.FileSystemAbstraction
  68. cacheFolder := filepath.ToSlash(filepath.Join(filepath.Clean(filepath.Dir(rpath)), "/.metadata/.cache/") + "/")
  69. fshAbs.MkdirAll(cacheFolder, 0755)
  70. hidden.HideFile(filepath.Dir(filepath.Clean(cacheFolder)))
  71. hidden.HideFile(cacheFolder)
  72. needsGen, cacheData, err := rh.checkCacheNeeded(fsh, rpath, generateOnly, cacheFolder)
  73. if err != nil {
  74. return "", err
  75. }
  76. if !needsGen {
  77. return cacheData, nil
  78. }
  79. //Generate
  80. return rh.generateCache(fsh, cacheFolder, rpath, generateOnly)
  81. }
  82. // checkCacheNeeded checks if cache generation is needed
  83. func (rh *RenderHandler) checkCacheNeeded(fsh *filesystem.FileSystemHandler, rpath string, generateOnly bool, cacheFolder string) (needsGeneration bool, cacheData string, err error) {
  84. if CacheExists(fsh, rpath) {
  85. if generateOnly {
  86. return false, "", nil
  87. }
  88. ext := ".jpg"
  89. if !fsh.FileSystemAbstraction.FileExists(cacheFolder + filepath.Base(rpath) + ".jpg") {
  90. ext = ".png"
  91. }
  92. folderModeTime, _ := fsh.FileSystemAbstraction.GetModTime(rpath)
  93. cacheImageModeTime, _ := fsh.FileSystemAbstraction.GetModTime(cacheFolder + filepath.Base(rpath) + ext)
  94. if folderModeTime > cacheImageModeTime {
  95. fsh.FileSystemAbstraction.Remove(cacheFolder + filepath.Base(rpath) + ext)
  96. return true, "", nil
  97. } else {
  98. counter := 0
  99. for rh.fileIsBusy(rpath) && counter < 15 {
  100. counter += 1
  101. time.Sleep(1 * time.Second)
  102. }
  103. if rh.fileIsBusy(rpath) {
  104. return false, "", errors.New("process racing for cache file")
  105. }
  106. ctx, err := getImageAsBase64(fsh, cacheFolder+filepath.Base(rpath)+ext)
  107. return false, ctx, err
  108. }
  109. } else if fsh.ReadOnly {
  110. return false, "", errors.New("cannot generate thumbnail on readonly file system")
  111. } else {
  112. if rh.fileIsBusy(rpath) {
  113. return false, "", errors.New("process racing for cache file")
  114. }
  115. }
  116. return true, "", nil
  117. }
  118. func (rh *RenderHandler) generateCache(fsh *filesystem.FileSystemHandler, cacheFolder string, rpath string, generateOnly bool) (string, error) {
  119. //Cache image not exists. Set this file to busy
  120. rh.renderingFiles.Store(rpath, "busy")
  121. //That object not exists. Generate cache image
  122. //Audio formats that might contains id4 thumbnail
  123. id4Formats := []string{".mp3", ".ogg", ".flac"}
  124. if utils.StringInArray(id4Formats, strings.ToLower(filepath.Ext(rpath))) {
  125. img, err := generateThumbnailForAudio(fsh, cacheFolder, rpath, generateOnly)
  126. rh.renderingFiles.Delete(rpath)
  127. return img, err
  128. }
  129. //Generate resized image for images
  130. imageFormats := []string{".png", ".jpeg", ".jpg"}
  131. if utils.StringInArray(imageFormats, strings.ToLower(filepath.Ext(rpath))) {
  132. img, err := generateThumbnailForImage(fsh, cacheFolder, rpath, generateOnly)
  133. rh.renderingFiles.Delete(rpath)
  134. return img, err
  135. }
  136. //Video formats, extract from the 5 sec mark
  137. vidFormats := []string{".mkv", ".mp4", ".webm", ".ogv", ".avi", ".rmvb"}
  138. if utils.StringInArray(vidFormats, strings.ToLower(filepath.Ext(rpath))) {
  139. img, err := generateThumbnailForVideo(fsh, cacheFolder, rpath, generateOnly)
  140. rh.renderingFiles.Delete(rpath)
  141. return img, err
  142. }
  143. //3D Model Formats
  144. modelFormats := []string{".stl", ".obj"}
  145. if utils.StringInArray(modelFormats, strings.ToLower(filepath.Ext(rpath))) {
  146. img, err := generateThumbnailForModel(fsh, cacheFolder, rpath, generateOnly)
  147. rh.renderingFiles.Delete(rpath)
  148. return img, err
  149. }
  150. //Photoshop file
  151. if strings.ToLower(filepath.Ext(rpath)) == ".psd" {
  152. img, err := generateThumbnailForPSD(fsh, cacheFolder, rpath, generateOnly)
  153. rh.renderingFiles.Delete(rpath)
  154. return img, err
  155. }
  156. //SVG file
  157. if strings.ToLower(filepath.Ext(rpath)) == ".svg" {
  158. img, err := generateThumbnailForSVG(fsh, cacheFolder, rpath, generateOnly)
  159. rh.renderingFiles.Delete(rpath)
  160. return img, err
  161. }
  162. //Folder preview renderer
  163. if fsh.FileSystemAbstraction.IsDir(rpath) && len(filepath.Base(rpath)) > 0 && filepath.Base(rpath)[:1] != "." {
  164. img, err := generateThumbnailForFolder(fsh, cacheFolder, rpath, generateOnly)
  165. rh.renderingFiles.Delete(rpath)
  166. return img, err
  167. }
  168. //Other filters
  169. rh.renderingFiles.Delete(rpath)
  170. return "", errors.New("no supported format")
  171. }
  172. func (rh *RenderHandler) fileIsBusy(path string) bool {
  173. if rh == nil {
  174. log.Println("RenderHandler is null!")
  175. return true
  176. }
  177. _, ok := rh.renderingFiles.Load(path)
  178. if !ok {
  179. //File path is not being process by another process
  180. return false
  181. } else {
  182. return true
  183. }
  184. }
  185. func getImageAsBase64(fsh *filesystem.FileSystemHandler, rpath string) (string, error) {
  186. fshAbs := fsh.FileSystemAbstraction
  187. content, err := fshAbs.ReadFile(rpath)
  188. if err != nil {
  189. return "", err
  190. }
  191. encoded := base64.StdEncoding.EncodeToString(content)
  192. return string(encoded), nil
  193. }
  194. // Load a list of folder cache from websocket, pass in "" (empty string) for default sorting method
  195. func (rh *RenderHandler) HandleLoadCache(w http.ResponseWriter, r *http.Request, fsh *filesystem.FileSystemHandler, rpath string, sortmode string) {
  196. //Get a list of files pending to be cached and sent
  197. targetPath := filepath.ToSlash(filepath.Clean(rpath))
  198. //Check if this path already exists another websocket ongoing connection.
  199. //If yes, disconnect the oldone
  200. oldc, ok := rh.renderingFolder.Load(targetPath)
  201. if ok {
  202. //Close and remove the old connection
  203. oldc.(*websocket.Conn).Close()
  204. }
  205. fis, err := fsh.FileSystemAbstraction.ReadDir(targetPath)
  206. if err != nil {
  207. w.WriteHeader(http.StatusInternalServerError)
  208. w.Write([]byte("500 - Internal Server Error"))
  209. return
  210. }
  211. //Upgrade the connection to websocket
  212. var upgrader = websocket.Upgrader{}
  213. upgrader.CheckOrigin = func(r *http.Request) bool { return true }
  214. c, err := upgrader.Upgrade(w, r, nil)
  215. if err != nil {
  216. log.Print("upgrade:", err)
  217. w.WriteHeader(http.StatusInternalServerError)
  218. w.Write([]byte("500 - Internal Server Error"))
  219. return
  220. }
  221. //Set this realpath as websocket connected
  222. rh.renderingFolder.Store(targetPath, c)
  223. //For each file, serve a cached image preview
  224. errorExists := false
  225. filesWithoutCache := []string{}
  226. //Updates implementation 02/10/2021: Load thumbnail of files first before folder and apply user preference sort mode
  227. if sortmode == "" {
  228. sortmode = "default"
  229. }
  230. sortedFis := fssort.SortDirEntryList(fis, sortmode)
  231. pendingFiles := []string{}
  232. pendingFolders := []string{}
  233. for _, fileInfo := range sortedFis {
  234. if !fileInfo.IsDir() {
  235. pendingFiles = append(pendingFiles, filepath.Join(targetPath, fileInfo.Name()))
  236. } else {
  237. pendingFolders = append(pendingFolders, filepath.Join(targetPath, fileInfo.Name()))
  238. }
  239. }
  240. pendingFiles = append(pendingFiles, pendingFolders...)
  241. files := pendingFiles
  242. //Updated implementation 24/12/2020: Load image with cache first before rendering those without
  243. for _, file := range files {
  244. if !CacheExists(fsh, file) {
  245. //Cache not exists. Render this later
  246. filesWithoutCache = append(filesWithoutCache, file)
  247. } else {
  248. //Cache exists. Send it out first
  249. cachedImage, err := rh.LoadCache(fsh, file, false)
  250. if err != nil {
  251. } else {
  252. jsonString, _ := json.Marshal([]string{filepath.Base(file), cachedImage})
  253. err := c.WriteMessage(1, jsonString)
  254. if err != nil {
  255. //Connection closed
  256. errorExists = true
  257. break
  258. }
  259. }
  260. }
  261. }
  262. retryList := []string{}
  263. //Render the remaining cache files
  264. for _, file := range filesWithoutCache {
  265. //Load the image cache
  266. cachedImage, err := rh.LoadCache(fsh, file, false)
  267. if err != nil {
  268. //Unable to load this file's cache. Push it to retry list
  269. retryList = append(retryList, file)
  270. } else {
  271. jsonString, _ := json.Marshal([]string{filepath.Base(file), cachedImage})
  272. err := c.WriteMessage(1, jsonString)
  273. if err != nil {
  274. //Connection closed
  275. errorExists = true
  276. break
  277. }
  278. }
  279. }
  280. //Process the retry list after some wait time
  281. if len(retryList) > 0 {
  282. time.Sleep(1000 * time.Millisecond)
  283. for _, file := range retryList {
  284. //Load the image cache
  285. cachedImage, err := rh.LoadCache(fsh, file, false)
  286. if err != nil {
  287. } else {
  288. jsonString, _ := json.Marshal([]string{filepath.Base(file), cachedImage})
  289. err := c.WriteMessage(1, jsonString)
  290. if err != nil {
  291. //Connection closed
  292. errorExists = true
  293. break
  294. }
  295. }
  296. }
  297. }
  298. //Clear record from syncmap
  299. if !errorExists {
  300. //This ended normally. Delete the targetPath
  301. rh.renderingFolder.Delete(targetPath)
  302. }
  303. c.Close()
  304. }
  305. // Check if the cache for a file exists
  306. func CacheExists(fsh *filesystem.FileSystemHandler, file string) bool {
  307. cacheFolder := filepath.ToSlash(filepath.Join(filepath.Clean(filepath.Dir(file)), "/.metadata/.cache/") + "/")
  308. return fsh.FileSystemAbstraction.FileExists(cacheFolder+filepath.Base(file)+".jpg") || fsh.FileSystemAbstraction.FileExists(cacheFolder+filepath.Base(file)+".png")
  309. }
  310. // Get cache path for this file, given realpath
  311. func GetCacheFilePath(fsh *filesystem.FileSystemHandler, file string) (string, error) {
  312. if CacheExists(fsh, file) {
  313. fshAbs := fsh.FileSystemAbstraction
  314. cacheFolder := filepath.ToSlash(filepath.Join(filepath.Clean(filepath.Dir(file)), "/.metadata/.cache/") + "/")
  315. if fshAbs.FileExists(cacheFolder + filepath.Base(file) + ".jpg") {
  316. return cacheFolder + filepath.Base(file) + ".jpg", nil
  317. } else if fshAbs.FileExists(cacheFolder + filepath.Base(file) + ".png") {
  318. return cacheFolder + filepath.Base(file) + ".png", nil
  319. } else {
  320. return "", errors.New("Unable to resolve thumbnail cache location")
  321. }
  322. } else {
  323. return "", errors.New("No thumbnail cached for this file")
  324. }
  325. }
  326. // Remove cache if exists, given realpath
  327. func RemoveCache(fsh *filesystem.FileSystemHandler, file string) error {
  328. if CacheExists(fsh, file) {
  329. cachePath, err := GetCacheFilePath(fsh, file)
  330. //log.Println("Removing ", cachePath, err)
  331. if err != nil {
  332. return err
  333. }
  334. //Remove the thumbnail cache
  335. os.Remove(cachePath)
  336. return nil
  337. } else {
  338. //log.Println("Cache not exists: ", file)
  339. return errors.New("Thumbnail cache not exists for this file")
  340. }
  341. }