local lfs = require("libs/libkoreader-lfs") local DataStorage = require("datastorage") local dump = require("dump") local purgeDir = require("ffi/util").purgeDir local DocSettings = {} local HISTORY_DIR = DataStorage:getHistoryDir() local function buildCandidate(file_path) if lfs.attributes(file_path, "mode") == "file" then return { file_path, lfs.attributes(file_path, "modification") } else return nil end end -- Sidecar directory is the file without _last_ suffix. function DocSettings:getSidecarDir(doc_path) if doc_path == nil or doc_path == '' then return '' end local file_without_suffix = doc_path:match("(.*)%.") if file_without_suffix then return file_without_suffix..".sdr" end -- We shouldn't be called with anything but files with registered -- extensions, but in case we are, return something useful return doc_path..".sdr" end function DocSettings:getSidecarFile(doc_path) if doc_path == nil or doc_path == '' then return '' end return self:getSidecarDir(doc_path) .. "/metadata." .. doc_path:match(".*%.(.+)") .. ".lua" end function DocSettings:hasSidecarDir(doc_path) -- We may be called with items from FileManager, which may not be a document if lfs.attributes(doc_path, "mode") == "directory" then return false end return lfs.attributes(self:getSidecarDir(doc_path), "mode") == "directory" end function DocSettings:getHistoryPath(fullpath) return HISTORY_DIR .. "/[" .. fullpath:gsub("(.*/)([^/]+)","%1] %2"):gsub("/","#") .. ".lua" end function DocSettings:getPathFromHistory(hist_name) if hist_name == nil or hist_name == '' then return '' end -- 1. select everything included in brackets local s = string.match(hist_name,"%b[]") if s == nil or s == '' then return '' end -- 2. crop the bracket-sign from both sides -- 3. and finally replace decorative signs '#' to dir-char '/' return string.gsub(string.sub(s,2,-3),"#","/") end function DocSettings:getNameFromHistory(hist_name) if hist_name == nil or hist_name == '' then return '' end local s = string.match(hist_name, "%b[]") if s == nil or s == '' then return '' end -- at first, search for path length -- and return the rest of string without 4 last characters (".lua") return string.sub(hist_name, string.len(s)+2, -5) end function DocSettings:ensureSidecar(sidecar) if lfs.attributes(sidecar, "mode") ~= "directory" then lfs.mkdir(sidecar) end end function DocSettings:open(docfile) -- TODO(zijiehe): Remove history_path, use only sidecar. local new = {} local ok, stored new.history_file = self:getHistoryPath(docfile) local sidecar = self:getSidecarDir(docfile) new.sidecar = sidecar DocSettings:ensureSidecar(sidecar) -- If there is a file which has a same name as the sidecar directory, or -- the file system is read-only, we should not waste time to read it. if lfs.attributes(sidecar, "mode") == "directory" then -- New sidecar file name is metadata.{file last suffix}.lua. So we -- can handle two files with only different suffixes. new.sidecar_file = self:getSidecarFile(docfile) new.legacy_sidecar_file = sidecar.."/".. docfile:match("([^%/]+%..+)")..".lua" end local candidates = {} -- New sidecar file table.insert(candidates, buildCandidate(new.sidecar_file)) -- Legacy sidecar file table.insert(candidates, buildCandidate(new.legacy_sidecar_file)) -- Legacy history folder table.insert(candidates, buildCandidate(new.history_file)) -- Legacy kpdfview setting table.insert(candidates, buildCandidate(docfile..".kpdfview.lua")) table.sort(candidates, function(l, r) if l == nil then return false elseif r == nil then return true else return l[2] > r[2] end end) for _, k in pairs(candidates) do ok, stored = pcall(dofile, k[1]) if ok then break end end if ok and stored then new.data = stored new.candidates = candidates else new.data = {} end return setmetatable(new, {__index = DocSettings}) end function DocSettings:readSetting(key) return self.data[key] end function DocSettings:saveSetting(key, value) self.data[key] = value end function DocSettings:delSetting(key) self.data[key] = nil end function DocSettings:flush() -- write serialized version of the data table into one of -- i) sidecar directory in the same directory of the document or -- ii) history directory in root directory of KOReader if not self.history_file and not self.sidecar_file then return end -- If we can write to sidecar_file, we do not need to write to history_file -- anymore. local serials = { self.sidecar_file, self.history_file } self:ensureSidecar(self.sidecar) local s_out = dump(self.data) os.setlocale('C', 'numeric') for _, f in pairs(serials) do local f_out = io.open(f, "w") if f_out ~= nil then f_out:write("-- we can read Lua syntax here!\nreturn ") f_out:write(s_out) f_out:write("\n") f_out:close() if self.candidates ~= nil and not G_reader_settings:readSetting( "preserve_legacy_docsetting") then for _, k in pairs(self.candidates) do if k[1] ~= f then os.remove(k[1]) -- We should not remove sidecar folder, as it may -- contain Kindle history files. end end end break end end end function DocSettings:close() self:flush() end function DocSettings:purge() if self.history_file then os.remove(self.history_file) end if lfs.attributes(self.sidecar, "mode") == "directory" then purgeDir(self.sidecar) end self.data = {} end return DocSettings