aboutsummaryrefslogtreecommitdiff
path: root/internal/feed/cache/cache.go
diff options
context:
space:
mode:
Diffstat (limited to 'internal/feed/cache/cache.go')
-rw-r--r--internal/feed/cache/cache.go179
1 files changed, 179 insertions, 0 deletions
diff --git a/internal/feed/cache/cache.go b/internal/feed/cache/cache.go
new file mode 100644
index 0000000..e7fe673
--- /dev/null
+++ b/internal/feed/cache/cache.go
@@ -0,0 +1,179 @@
+package cache
+
+import (
+ "bufio"
+ "encoding/gob"
+ "errors"
+ "fmt"
+ "os"
+ "path/filepath"
+ "time"
+
+ "github.com/nightlyone/lockfile"
+
+ "github.com/Necoro/feed2imap-go/internal/feed"
+ "github.com/Necoro/feed2imap-go/pkg/log"
+)
+
+type Version byte
+
+const (
+ currentVersion Version = 1
+)
+
+type Impl interface {
+ cachedFeed(*feed.Feed) CachedFeed
+ transformToCurrent() (Impl, error)
+ Version() Version
+ Info() string
+ SpecificInfo(interface{}) string
+}
+
+type Cache struct {
+ Impl
+ lock lockfile.Lockfile
+ locked bool
+}
+
+type CachedFeed interface {
+ // Checked marks the feed as being a failure or a success on last check.
+ Checked(withFailure bool)
+ // Failures of this feed up to now.
+ Failures() int
+ // The Last time, this feed has been checked
+ Last() time.Time
+ // Filter the given items against the cached items.
+ Filter(items []feed.Item, ignoreHash bool, alwaysNew bool) []feed.Item
+ // Commit any changes done to the cache state.
+ Commit()
+ // The Feed, that is cached.
+ Feed() *feed.Feed
+}
+
+func cacheForVersion(version Version) (Impl, error) {
+ switch version {
+ case v1Version:
+ return newV1Cache(), nil
+ default:
+ return nil, fmt.Errorf("unknown cache version '%d'", version)
+ }
+}
+
+func lockName(fileName string) (string, error) {
+ return filepath.Abs(fileName + ".lck")
+}
+
+func lock(fileName string) (lock lockfile.Lockfile, err error) {
+ var lockFile string
+
+ if lockFile, err = lockName(fileName); err != nil {
+ return
+ }
+ log.Debugf("Handling lock file '%s'", lockFile)
+
+ if lock, err = lockfile.New(lockFile); err != nil {
+ err = fmt.Errorf("Creating lock file: %w", err)
+ return
+ }
+
+ if err = lock.TryLock(); err != nil {
+ err = fmt.Errorf("Locking cache: %w", err)
+ return
+ }
+
+ return
+}
+
+func (cache *Cache) store(fileName string) error {
+ if cache.Impl == nil {
+ return fmt.Errorf("trying to store nil cache")
+ }
+ if cache.Version() != currentVersion {
+ return fmt.Errorf("trying to store cache with unsupported version '%d' (current: '%d')", cache.Version(), currentVersion)
+ }
+
+ f, err := os.Create(fileName)
+ if err != nil {
+ return fmt.Errorf("trying to store cache to '%s': %w", fileName, err)
+ }
+ defer f.Close()
+
+ writer := bufio.NewWriter(f)
+ if err = writer.WriteByte(byte(currentVersion)); err != nil {
+ return fmt.Errorf("writing to '%s': %w", fileName, err)
+ }
+
+ encoder := gob.NewEncoder(writer)
+ if err = encoder.Encode(cache.Impl); err != nil {
+ return fmt.Errorf("encoding cache: %w", err)
+ }
+
+ writer.Flush()
+ log.Printf("Stored cache to '%s'.", fileName)
+
+ return cache.Unlock()
+}
+
+func (cache *Cache) Unlock() error {
+ if cache.locked {
+ if err := cache.lock.Unlock(); err != nil {
+ return fmt.Errorf("Unlocking cache: %w", err)
+ }
+ }
+ cache.locked = false
+ return nil
+}
+
+func newCache() (Cache, error) {
+ cache, err := cacheForVersion(currentVersion)
+ if err != nil {
+ return Cache{}, err
+ }
+ return Cache{
+ Impl: cache,
+ locked: false,
+ }, nil
+}
+
+func LoadCache(fileName string) (Cache, error) {
+ f, err := os.Open(fileName)
+ if err != nil {
+ if errors.Is(err, os.ErrNotExist) {
+ // no cache there yet -- make new
+ return newCache()
+ }
+ return Cache{}, fmt.Errorf("opening cache at '%s': %w", fileName, err)
+ }
+ defer f.Close()
+
+ lock, err := lock(fileName)
+ if err != nil {
+ return Cache{}, err
+ }
+
+ log.Printf("Loading cache from '%s'", fileName)
+
+ reader := bufio.NewReader(f)
+ version, err := reader.ReadByte()
+ if err != nil {
+ return Cache{}, fmt.Errorf("reading from '%s': %w", fileName, err)
+ }
+
+ cache, err := cacheForVersion(Version(version))
+ if err != nil {
+ return Cache{}, err
+ }
+
+ decoder := gob.NewDecoder(reader)
+ if err = decoder.Decode(cache); err != nil {
+ return Cache{}, fmt.Errorf("decoding for version '%d' from '%s': %w", version, fileName, err)
+ }
+
+ if cache, err = cache.transformToCurrent(); err != nil {
+ return Cache{}, fmt.Errorf("cannot transform from version %d to %d: %w", version, currentVersion, err)
+ }
+
+ log.Printf("Loaded cache (version %d), transformed to version %d.", version, currentVersion)
+
+ return Cache{cache, lock, true}, nil
+}