Initial Commit
This commit is contained in:
commit
74fd90c850
83
asink.go
Normal file
83
asink.go
Normal file
@ -0,0 +1,83 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"flag"
|
||||
"path"
|
||||
"os/user"
|
||||
"github.com/howeyc/fsnotify"
|
||||
"code.google.com/p/goconf/conf"
|
||||
)
|
||||
|
||||
var configFileName string
|
||||
|
||||
func init() {
|
||||
const config_usage = "Config File to use"
|
||||
userHomeDir := "~"
|
||||
|
||||
u, err := user.Current()
|
||||
if err == nil {
|
||||
userHomeDir = u.HomeDir
|
||||
}
|
||||
|
||||
flag.StringVar(&configFileName, "config", path.Join(userHomeDir, ".asink", "config"), config_usage)
|
||||
flag.StringVar(&configFileName, "c", path.Join(userHomeDir, ".asink", "config"), config_usage+" (shorthand)")
|
||||
}
|
||||
|
||||
func main() {
|
||||
flag.Parse()
|
||||
fmt.Println("config file:", configFileName)
|
||||
|
||||
config, err := conf.ReadConfigFile(configFileName)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
fmt.Println("Error reading config file at ", configFileName, ". Does it exist?")
|
||||
return
|
||||
}
|
||||
|
||||
storage, err := GetStorage(config)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
return
|
||||
}
|
||||
|
||||
syncdir, err := config.GetString("local", "syncdir")
|
||||
cachedir, err := config.GetString("local", "cachedir")
|
||||
|
||||
fmt.Println(syncdir)
|
||||
fmt.Println(cachedir)
|
||||
fmt.Println(storage)
|
||||
|
||||
setup_watchers()
|
||||
}
|
||||
|
||||
func setup_watchers() {
|
||||
watcher, err := fsnotify.NewWatcher()
|
||||
if err != nil {
|
||||
fmt.Println("Failed to create fsnotify watcher")
|
||||
return
|
||||
}
|
||||
fmt.Println("Created new fsnotify watcher!")
|
||||
|
||||
err = watcher.Watch("/home/aclindsa/.asink")
|
||||
if err != nil {
|
||||
fmt.Println("Failed to watch /home/aclindsa/.asink")
|
||||
return
|
||||
}
|
||||
|
||||
for {
|
||||
select {
|
||||
case ev := <-watcher.Event:
|
||||
fmt.Println("event:", ev)
|
||||
hash, err := HashFile(ev.Name)
|
||||
//TODO if creating a directory, start watching it (and then initiate a full scan of it so we're sure nothing slipped through the cracks)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
} else {
|
||||
fmt.Println(hash)
|
||||
}
|
||||
case err := <-watcher.Error:
|
||||
fmt.Println("error:", err)
|
||||
}
|
||||
}
|
||||
}
|
21
hash.go
Normal file
21
hash.go
Normal file
@ -0,0 +1,21 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"io"
|
||||
"os"
|
||||
"fmt"
|
||||
"crypto/sha256"
|
||||
)
|
||||
|
||||
func HashFile(filename string) (string, error) {
|
||||
hashfn := sha256.New()
|
||||
|
||||
infile, err := os.Open(filename)
|
||||
if err != nil { return "", err }
|
||||
defer infile.Close()
|
||||
|
||||
_, err = io.Copy(hashfn, infile)
|
||||
if err != nil { return "", err }
|
||||
|
||||
return fmt.Sprintf("%x", hashfn.Sum(nil)), nil
|
||||
}
|
32
storage.go
Normal file
32
storage.go
Normal file
@ -0,0 +1,32 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"code.google.com/p/goconf/conf"
|
||||
)
|
||||
|
||||
type Storage interface {
|
||||
Put(filename string, hash string) error
|
||||
Get(filename string, hash string) error
|
||||
}
|
||||
|
||||
func GetStorage(config *conf.ConfigFile) (*Storage, error) {
|
||||
storageMethod, err := config.GetString("storage", "method")
|
||||
if err != nil {
|
||||
return nil, errors.New("Error: storage method not specified in config file.")
|
||||
}
|
||||
|
||||
var storage Storage
|
||||
|
||||
switch storageMethod {
|
||||
case "local":
|
||||
storage, err = NewLocalStorage(config)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
default:
|
||||
return nil, errors.New("Error: storage method '" + storageMethod + "' not implemented.")
|
||||
}
|
||||
|
||||
return &storage, nil
|
||||
}
|
69
storage_local.go
Normal file
69
storage_local.go
Normal file
@ -0,0 +1,69 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"os"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"path"
|
||||
"errors"
|
||||
"code.google.com/p/goconf/conf"
|
||||
)
|
||||
|
||||
type LocalStorage struct {
|
||||
storageDir string
|
||||
tmpSubdir string
|
||||
}
|
||||
|
||||
func NewLocalStorage(config *conf.ConfigFile) (*LocalStorage, error) {
|
||||
storageDir, err := config.GetString("storage", "dir")
|
||||
if err != nil {
|
||||
return nil, errors.New("Error: LocalStorage indicated in config file, but lacking local storage directory ('dir = some/dir').")
|
||||
}
|
||||
|
||||
ls := new(LocalStorage)
|
||||
ls.storageDir = storageDir
|
||||
ls.tmpSubdir = path.Join(storageDir, ".asink-tmpdir")
|
||||
return ls, nil
|
||||
}
|
||||
|
||||
func (ls *LocalStorage) copyToTmp(src string) (string, error) {
|
||||
infile, err := os.Open(src)
|
||||
if err != nil { return "", err }
|
||||
defer infile.Close()
|
||||
|
||||
outfile, err := ioutil.TempFile(ls.tmpSubdir, "asink")
|
||||
if err != nil { return "", err }
|
||||
defer outfile.Close()
|
||||
|
||||
_, err = io.Copy(outfile, infile)
|
||||
if err != nil { return "", err }
|
||||
|
||||
return outfile.Name(), nil
|
||||
}
|
||||
|
||||
func (ls *LocalStorage) Put(filename string, hash string) (e error) {
|
||||
tmpfile, err := ls.copyToTmp(filename)
|
||||
if err != nil { return err }
|
||||
defer func() {
|
||||
err := os.Remove(tmpfile)
|
||||
if err != nil && e == nil { e = err }
|
||||
}()
|
||||
|
||||
err = os.Rename(tmpfile, path.Join(ls.storageDir, hash))
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (ls *LocalStorage) Get(filename string, hash string) error {
|
||||
infile, err := os.Open(path.Join(ls.storageDir, hash))
|
||||
if err != nil { return err }
|
||||
defer infile.Close()
|
||||
|
||||
outfile, err := os.Open(filename)
|
||||
if err != nil { return err }
|
||||
defer outfile.Close()
|
||||
|
||||
_, err = io.Copy(outfile, infile)
|
||||
|
||||
return err
|
||||
}
|
Loading…
Reference in New Issue
Block a user