| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405 | package metadataimport (	"encoding/base64"	"encoding/json"	"errors"	"log"	"net/http"	"os"	"path/filepath"	"strings"	"sync"	"time"	"github.com/gorilla/websocket"	"imuslab.com/arozos/mod/filesystem"	"imuslab.com/arozos/mod/filesystem/fssort"	hidden "imuslab.com/arozos/mod/filesystem/hidden")/*	This package is used to extract meta data from files like mp3 and mp4	Also support image caching*/type RenderHandler struct {	renderingFiles  sync.Map	renderingFolder sync.Map}//Create a new RenderHandlerfunc NewRenderHandler() *RenderHandler {	return &RenderHandler{		renderingFiles:  sync.Map{},		renderingFolder: sync.Map{},	}}//Build cache for all files (non recursive) for the given filepathfunc (rh *RenderHandler) BuildCacheForFolder(fsh *filesystem.FileSystemHandler, vpath string, username string) error {	fshAbs := fsh.FileSystemAbstraction	rpath, _ := fshAbs.RealPathToVirtualPath(vpath, username)	//Get a list of all files inside this path	files, err := fshAbs.Glob(filepath.ToSlash(filepath.Clean(rpath)) + "/*")	if err != nil {		return err	}	for _, file := range files {		//Load Cache in generate mode		rh.LoadCache(fsh, file, true)	}	//Check if the cache folder has file. If not, remove it	cachedFiles, _ := fshAbs.Glob(filepath.ToSlash(filepath.Join(filepath.Clean(rpath), "/.metadata/.cache/*")))	if len(cachedFiles) == 0 {		fshAbs.RemoveAll(filepath.ToSlash(filepath.Join(filepath.Clean(rpath), "/.metadata/.cache/")) + "/")	}	return nil}func (rh *RenderHandler) LoadCacheAsBytes(fsh *filesystem.FileSystemHandler, vpath string, username string, generateOnly bool) ([]byte, error) {	fshAbs := fsh.FileSystemAbstraction	rpath, _ := fshAbs.VirtualPathToRealPath(vpath, username)	b64, err := rh.LoadCache(fsh, rpath, generateOnly)	if err != nil {		return []byte{}, err	}	resultingBytes, _ := base64.StdEncoding.DecodeString(b64)	return resultingBytes, nil}//Try to load a cache from file. If not exists, generate it nowfunc (rh *RenderHandler) LoadCache(fsh *filesystem.FileSystemHandler, rpath string, generateOnly bool) (string, error) {	//Create a cache folder	fshAbs := fsh.FileSystemAbstraction	cacheFolder := filepath.ToSlash(filepath.Join(filepath.Clean(filepath.Dir(rpath)), "/.metadata/.cache/") + "/")	fshAbs.MkdirAll(cacheFolder, 0755)	hidden.HideFile(filepath.Dir(filepath.Clean(cacheFolder)))	hidden.HideFile(cacheFolder)	//Check if cache already exists. If yes, return the image from the cache folder	if CacheExists(fsh, rpath) {		if generateOnly {			//Only generate, do not return image			return "", nil		}		//Allow thumbnail to be either jpg or png file		ext := ".jpg"		if !fshAbs.FileExists(cacheFolder + filepath.Base(rpath) + ".jpg") {			ext = ".png"		}		//Updates 02/10/2021: Check if the source file is newer than the cache. Update the cache if true		folderModeTime, _ := fshAbs.GetModTime(rpath)		cacheImageModeTime, _ := fshAbs.GetModTime(cacheFolder + filepath.Base(rpath) + ext)		if folderModeTime > cacheImageModeTime {			//File is newer than cache. Delete the cache			fshAbs.Remove(cacheFolder + filepath.Base(rpath) + ext)		} else {			//Check if the file is being writting by another process. If yes, wait for it			counter := 0			for rh.fileIsBusy(rpath) && counter < 15 {				counter += 1				time.Sleep(1 * time.Second)			}			//Time out and the file is still busy			if rh.fileIsBusy(rpath) {				log.Println("Process racing for cache file. Skipping", filepath.Base(rpath))				return "", errors.New("Process racing for cache file. Skipping")			}			//Read and return the image			ctx, err := getImageAsBase64(fsh, cacheFolder+filepath.Base(rpath)+ext)			return ctx, err		}	} else {		//This file not exists yet. Check if it is being hold by another process already		if rh.fileIsBusy(rpath) {			log.Println("Process racing for cache file. Skipping", filepath.Base(rpath))			return "", errors.New("Process racing for cache file. Skipping")		}	}	//Cache image not exists. Set this file to busy	rh.renderingFiles.Store(rpath, "busy")	//That object not exists. Generate cache image	//Audio formats that might contains id4 thumbnail	id4Formats := []string{".mp3", ".ogg", ".flac"}	if inArray(id4Formats, strings.ToLower(filepath.Ext(rpath))) {		img, err := generateThumbnailForAudio(fsh, cacheFolder, rpath, generateOnly)		rh.renderingFiles.Delete(rpath)		return img, err	}	//Generate resized image for images	imageFormats := []string{".png", ".jpeg", ".jpg"}	if inArray(imageFormats, strings.ToLower(filepath.Ext(rpath))) {		img, err := generateThumbnailForImage(fsh, cacheFolder, rpath, generateOnly)		rh.renderingFiles.Delete(rpath)		return img, err	}	//Video formats, extract from the 5 sec mark	vidFormats := []string{".mkv", ".mp4", ".webm", ".ogv", ".avi", ".rmvb"}	if inArray(vidFormats, strings.ToLower(filepath.Ext(rpath))) {		img, err := generateThumbnailForVideo(fsh, cacheFolder, rpath, generateOnly)		rh.renderingFiles.Delete(rpath)		return img, err	}	//3D Model Formats	modelFormats := []string{".stl", ".obj"}	if inArray(modelFormats, strings.ToLower(filepath.Ext(rpath))) {		img, err := generateThumbnailForModel(fsh, cacheFolder, rpath, generateOnly)		rh.renderingFiles.Delete(rpath)		return img, err	}	//Photoshop file	if strings.ToLower(filepath.Ext(rpath)) == ".psd" {		img, err := generateThumbnailForPSD(fsh, cacheFolder, rpath, generateOnly)		rh.renderingFiles.Delete(rpath)		return img, err	}	//Folder preview renderer	if fshAbs.IsDir(rpath) && len(filepath.Base(rpath)) > 0 && filepath.Base(rpath)[:1] != "." {		img, err := generateThumbnailForFolder(fsh, cacheFolder, rpath, generateOnly)		rh.renderingFiles.Delete(rpath)		return img, err	}	//Other filters	rh.renderingFiles.Delete(rpath)	return "", errors.New("No supported format")}func (rh *RenderHandler) fileIsBusy(path string) bool {	if rh == nil {		log.Println("RenderHandler is null!")		return true	}	_, ok := rh.renderingFiles.Load(path)	if !ok {		//File path is not being process by another process		return false	} else {		return true	}}func getImageAsBase64(fsh *filesystem.FileSystemHandler, rpath string) (string, error) {	fshAbs := fsh.FileSystemAbstraction	content, err := fshAbs.ReadFile(rpath)	if err != nil {		return "", err	}	encoded := base64.StdEncoding.EncodeToString(content)	return string(encoded), nil}//Load a list of folder cache from websocket, pass in "" (empty string) for default sorting methodfunc (rh *RenderHandler) HandleLoadCache(w http.ResponseWriter, r *http.Request, fsh *filesystem.FileSystemHandler, rpath string, sortmode string) {	//Get a list of files pending to be cached and sent	targetPath := filepath.ToSlash(filepath.Clean(rpath))	//Check if this path already exists another websocket ongoing connection.	//If yes, disconnect the oldone	oldc, ok := rh.renderingFolder.Load(targetPath)	if ok {		//Close and remove the old connection		oldc.(*websocket.Conn).Close()	}	files, err := specialGlob(fsh, targetPath+"/*")	if err != nil {		w.WriteHeader(http.StatusInternalServerError)		w.Write([]byte("500 - Internal Server Error"))		return	}	//Upgrade the connection to websocket	var upgrader = websocket.Upgrader{}	upgrader.CheckOrigin = func(r *http.Request) bool { return true }	c, err := upgrader.Upgrade(w, r, nil)	if err != nil {		log.Print("upgrade:", err)		w.WriteHeader(http.StatusInternalServerError)		w.Write([]byte("500 - Internal Server Error"))		return	}	//Set this realpath as websocket connected	rh.renderingFolder.Store(targetPath, c)	//For each file, serve a cached image preview	errorExists := false	filesWithoutCache := []string{}	//Updates implementation 02/10/2021: Load thumbnail of files first before folder and apply user preference sort mode	if sortmode == "" {		sortmode = "default"	}	pendingFiles := []string{}	pendingFolders := []string{}	for _, file := range files {		if isDir(file) {			pendingFiles = append(pendingFiles, file)		} else {			pendingFolders = append(pendingFolders, file)		}	}	pendingFiles = append(pendingFiles, pendingFolders...)	files = fssort.SortFileList(pendingFiles, sortmode)	//Updated implementation 24/12/2020: Load image with cache first before rendering those without	for _, file := range files {		if !CacheExists(fsh, file) {			//Cache not exists. Render this later			filesWithoutCache = append(filesWithoutCache, file)		} else {			//Cache exists. Send it out first			cachedImage, err := rh.LoadCache(fsh, file, false)			if err != nil {			} else {				jsonString, _ := json.Marshal([]string{filepath.Base(file), cachedImage})				err := c.WriteMessage(1, jsonString)				if err != nil {					//Connection closed					errorExists = true					break				}			}		}	}	retryList := []string{}	//Render the remaining cache files	for _, file := range filesWithoutCache {		//Load the image cache		cachedImage, err := rh.LoadCache(fsh, file, false)		if err != nil {			//Unable to load this file's cache. Push it to retry list			retryList = append(retryList, file)		} else {			jsonString, _ := json.Marshal([]string{filepath.Base(file), cachedImage})			err := c.WriteMessage(1, jsonString)			if err != nil {				//Connection closed				errorExists = true				break			}		}	}	//Process the retry list after some wait time	if len(retryList) > 0 {		time.Sleep(1000 * time.Millisecond)		for _, file := range retryList {			//Load the image cache			cachedImage, err := rh.LoadCache(fsh, file, false)			if err != nil {			} else {				jsonString, _ := json.Marshal([]string{filepath.Base(file), cachedImage})				err := c.WriteMessage(1, jsonString)				if err != nil {					//Connection closed					errorExists = true					break				}			}		}	}	//Clear record from syncmap	if !errorExists {		//This ended normally. Delete the targetPath		rh.renderingFolder.Delete(targetPath)	}	c.Close()}//Check if the cache for a file existsfunc CacheExists(fsh *filesystem.FileSystemHandler, file string) bool {	cacheFolder := filepath.ToSlash(filepath.Join(filepath.Clean(filepath.Dir(file)), "/.metadata/.cache/") + "/")	return fsh.FileSystemAbstraction.FileExists(cacheFolder+filepath.Base(file)+".jpg") || fileExists(cacheFolder+filepath.Base(file)+".png")}//Get cache path for this file, given realpathfunc GetCacheFilePath(fsh *filesystem.FileSystemHandler, file string) (string, error) {	if CacheExists(fsh, file) {		fshAbs := fsh.FileSystemAbstraction		cacheFolder := filepath.ToSlash(filepath.Join(filepath.Clean(filepath.Dir(file)), "/.metadata/.cache/") + "/")		if fshAbs.FileExists(cacheFolder + filepath.Base(file) + ".jpg") {			return cacheFolder + filepath.Base(file) + ".jpg", nil		} else if fshAbs.FileExists(cacheFolder + filepath.Base(file) + ".png") {			return cacheFolder + filepath.Base(file) + ".png", nil		} else {			return "", errors.New("Unable to resolve thumbnail cache location")		}	} else {		return "", errors.New("No thumbnail cached for this file")	}}//Remove cache if exists, given realpathfunc RemoveCache(fsh *filesystem.FileSystemHandler, file string) error {	if CacheExists(fsh, file) {		cachePath, err := GetCacheFilePath(fsh, file)		//log.Println("Removing ", cachePath, err)		if err != nil {			return err		}		//Remove the thumbnail cache		os.Remove(cachePath)		return nil	} else {		//log.Println("Cache not exists: ", file)		return errors.New("Thumbnail cache not exists for this file")	}}func specialGlob(fsh *filesystem.FileSystemHandler, path string) ([]string, error) {	fshAbs := fsh.FileSystemAbstraction	files, err := fshAbs.Glob(path)	if err != nil {		return []string{}, err	}	if strings.Contains(path, "[") == true || strings.Contains(path, "]") == true {		if len(files) == 0 {			//Handle reverse check. Replace all [ and ] with *			newSearchPath := strings.ReplaceAll(path, "[", "?")			newSearchPath = strings.ReplaceAll(newSearchPath, "]", "?")			//Scan with all the similar structure except [ and ]			tmpFilelist, _ := filepath.Glob(newSearchPath)			for _, file := range tmpFilelist {				file = filepath.ToSlash(file)				if strings.Contains(file, filepath.ToSlash(filepath.Dir(path))) {					files = append(files, file)				}			}		}	}	//Convert all filepaths to slash	for i := 0; i < len(files); i++ {		files[i] = filepath.ToSlash(files[i])	}	return files, nil}
 |