mirror of
https://github.com/derfenix/photocatalog.git
synced 2026-03-11 21:35:34 +03:00
Initial commit
This commit is contained in:
10
pkg/core/managemode.go
Normal file
10
pkg/core/managemode.go
Normal file
@@ -0,0 +1,10 @@
|
||||
package core
|
||||
|
||||
//go:generate stringer -type ManageMode
|
||||
|
||||
type ManageMode uint8
|
||||
|
||||
const (
|
||||
Copy ManageMode = iota
|
||||
Hardlink
|
||||
)
|
||||
24
pkg/core/managemode_string.go
Normal file
24
pkg/core/managemode_string.go
Normal file
@@ -0,0 +1,24 @@
|
||||
// Code generated by "stringer -type ManageMode"; DO NOT EDIT.
|
||||
|
||||
package core
|
||||
|
||||
import "strconv"
|
||||
|
||||
func _() {
|
||||
// An "invalid array index" compiler error signifies that the constant values have changed.
|
||||
// Re-run the stringer command to generate them again.
|
||||
var x [1]struct{}
|
||||
_ = x[Copy-0]
|
||||
_ = x[Hardlink-1]
|
||||
}
|
||||
|
||||
const _ManageMode_name = "CopyHardlink"
|
||||
|
||||
var _ManageMode_index = [...]uint8{0, 4, 12}
|
||||
|
||||
func (i ManageMode) String() string {
|
||||
if i >= ManageMode(len(_ManageMode_index)-1) {
|
||||
return "ManageMode(" + strconv.FormatInt(int64(i), 10) + ")"
|
||||
}
|
||||
return _ManageMode_name[_ManageMode_index[i]:_ManageMode_index[i+1]]
|
||||
}
|
||||
134
pkg/core/manager.go
Normal file
134
pkg/core/manager.go
Normal file
@@ -0,0 +1,134 @@
|
||||
package core
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"log"
|
||||
"os"
|
||||
"os/exec"
|
||||
"path"
|
||||
"strings"
|
||||
|
||||
"github.com/pkg/errors"
|
||||
|
||||
"photocatalog/pkg/metadata"
|
||||
)
|
||||
|
||||
type Manager struct {
|
||||
TargetPath string
|
||||
Mode ManageMode
|
||||
|
||||
processor func(fp, targetDir string) (string, error)
|
||||
extractorsCache map[string]metadata.Extractor
|
||||
}
|
||||
|
||||
func NewManager(target string, mode ManageMode) (*Manager, error) {
|
||||
manager := Manager{
|
||||
TargetPath: target,
|
||||
Mode: mode,
|
||||
processor: nil,
|
||||
}
|
||||
if err := manager.initProcessor(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
manager.extractorsCache = map[string]metadata.Extractor{}
|
||||
return &manager, nil
|
||||
}
|
||||
|
||||
func (m *Manager) buildTarget(meta *metadata.Metadata) (string, error) {
|
||||
dir, err := m.dirPathFromMeta(meta)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
dirPath := path.Join(m.TargetPath, dir)
|
||||
err = os.MkdirAll(dirPath, os.FileMode(0770))
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return dirPath, nil
|
||||
}
|
||||
|
||||
func (m *Manager) dirPathFromMeta(meta *metadata.Metadata) (string, error) {
|
||||
t := meta.Time
|
||||
year := t.Format("2006")
|
||||
month := t.Format("01")
|
||||
day := t.Format("02")
|
||||
return path.Join(year, month, day), nil
|
||||
}
|
||||
|
||||
func (m *Manager) getMetadataExtractor(fp string) metadata.Extractor {
|
||||
switch strings.ToLower(path.Ext(fp)) {
|
||||
case ".jpeg", ".jpg":
|
||||
if _, ok := m.extractorsCache["jpeg"]; !ok {
|
||||
m.extractorsCache["jpeg"] = metadata.NewJpegExtractor()
|
||||
}
|
||||
return m.extractorsCache["jpeg"]
|
||||
default:
|
||||
if _, ok := m.extractorsCache["default"]; !ok {
|
||||
m.extractorsCache["default"] = metadata.NewDefaultExtractor()
|
||||
}
|
||||
return m.extractorsCache["default"]
|
||||
}
|
||||
}
|
||||
|
||||
func (m *Manager) initProcessor() error {
|
||||
switch m.Mode {
|
||||
case Copy:
|
||||
m.processor = func(fp, targetDir string) (string, error) {
|
||||
_, fn := path.Split(fp)
|
||||
target := path.Join(targetDir, fn)
|
||||
cmd := exec.Command("cp", "-f", "--reflink=auto", fp, target)
|
||||
return target, cmd.Run()
|
||||
}
|
||||
case Hardlink:
|
||||
m.processor = func(fp, targetDir string) (string, error) {
|
||||
_, fn := path.Split(fp)
|
||||
target := path.Join(targetDir, fn)
|
||||
cmd := exec.Command("ln", "-f", fp, target)
|
||||
return target, cmd.Run()
|
||||
}
|
||||
default:
|
||||
return fmt.Errorf("failed to init processor: invalid Mode value")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Manager) Manage(fp string) error {
|
||||
if m.processor == nil {
|
||||
return fmt.Errorf("no processor initialized")
|
||||
}
|
||||
// Skip hidden files
|
||||
if strings.HasPrefix(path.Base(fp), ".") {
|
||||
return nil
|
||||
}
|
||||
|
||||
log.Println("processing", fp)
|
||||
|
||||
extractor := m.getMetadataExtractor(fp)
|
||||
if extractor == nil {
|
||||
return fmt.Errorf("failed to get md extractor for %s", fp)
|
||||
}
|
||||
|
||||
md, err := extractor.Extract(fp)
|
||||
if err != nil {
|
||||
return errors.WithMessagef(err, "failed to extract md from %s", fp)
|
||||
}
|
||||
|
||||
targetDir, err := m.buildTarget(&md)
|
||||
if err != nil {
|
||||
return errors.WithMessagef(err, "failed to create dir for %s", fp)
|
||||
}
|
||||
|
||||
target, err := m.processor(fp, targetDir)
|
||||
if err != nil {
|
||||
return errors.WithMessagef(err, "failed to process %s to %s", fp, targetDir)
|
||||
}
|
||||
|
||||
if m.Mode == Hardlink {
|
||||
log.Println(fp, "linked to", target)
|
||||
} else if m.Mode == Copy {
|
||||
log.Println(fp, "copied to", target)
|
||||
}
|
||||
|
||||
log.Println("success")
|
||||
return nil
|
||||
}
|
||||
Reference in New Issue
Block a user