2020-03-27 11:01:41 +01:00
|
|
|
/*
|
|
|
|
Package entity implementing sharing with WebDAV servers.
|
|
|
|
|
|
|
|
Additional information can be found in our Developer Guide:
|
|
|
|
|
|
|
|
https://github.com/photoprism/photoprism/wiki
|
|
|
|
*/
|
|
|
|
package webdav
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path"
|
2020-04-24 14:11:17 +02:00
|
|
|
"time"
|
2020-03-27 11:01:41 +01:00
|
|
|
|
|
|
|
"github.com/photoprism/photoprism/internal/event"
|
2020-04-02 18:17:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
2020-03-27 11:01:41 +01:00
|
|
|
"github.com/studio-b12/gowebdav"
|
|
|
|
)
|
|
|
|
|
|
|
|
var log = event.Log
|
|
|
|
|
|
|
|
type Client struct {
|
|
|
|
client *gowebdav.Client
|
|
|
|
}
|
|
|
|
|
2020-04-03 18:08:49 +02:00
|
|
|
// New creates a new WebDAV client.
|
|
|
|
func New(url, user, pass string) Client {
|
2020-03-27 11:01:41 +01:00
|
|
|
clt := gowebdav.NewClient(url, user, pass)
|
|
|
|
|
2020-04-24 14:11:17 +02:00
|
|
|
clt.SetTimeout(10 * time.Minute) // TODO: Change timeout if needed
|
|
|
|
|
2020-03-27 11:01:41 +01:00
|
|
|
result := Client{client: clt}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-03-27 11:56:24 +01:00
|
|
|
func (c Client) readDir(path string) ([]os.FileInfo, error) {
|
|
|
|
if path == "" {
|
|
|
|
path = "/"
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.client.ReadDir(path)
|
|
|
|
}
|
|
|
|
|
2020-03-27 11:01:41 +01:00
|
|
|
// Files returns all files in path as string slice.
|
2020-04-02 18:17:07 +02:00
|
|
|
func (c Client) Files(dir string) (result fs.FileInfos, err error) {
|
2020-03-31 17:26:25 +02:00
|
|
|
files, err := c.readDir(dir)
|
2020-03-27 11:01:41 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, file := range files {
|
2020-03-27 18:17:07 +01:00
|
|
|
if !file.Mode().IsRegular() {
|
|
|
|
continue
|
|
|
|
}
|
2020-04-02 18:17:07 +02:00
|
|
|
|
|
|
|
info := fs.NewFileInfo(file, dir)
|
|
|
|
|
|
|
|
result = append(result, info)
|
2020-03-27 11:01:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Directories returns all sub directories in path as string slice.
|
2020-04-02 18:17:07 +02:00
|
|
|
func (c Client) Directories(root string, recursive bool) (result fs.FileInfos, err error) {
|
2020-03-31 17:26:25 +02:00
|
|
|
files, err := c.readDir(root)
|
2020-03-27 11:01:41 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
2020-03-31 17:26:25 +02:00
|
|
|
if root == "/" {
|
|
|
|
root = ""
|
|
|
|
}
|
|
|
|
|
2020-03-27 11:01:41 +01:00
|
|
|
for _, file := range files {
|
2020-03-27 18:17:07 +01:00
|
|
|
if !file.Mode().IsDir() {
|
|
|
|
continue
|
|
|
|
}
|
2020-03-27 11:56:24 +01:00
|
|
|
|
2020-04-02 18:17:07 +02:00
|
|
|
info := fs.NewFileInfo(file, root)
|
2020-03-31 17:26:25 +02:00
|
|
|
|
2020-04-02 18:17:07 +02:00
|
|
|
result = append(result, info)
|
2020-03-27 11:56:24 +01:00
|
|
|
|
|
|
|
if recursive {
|
2020-04-02 18:17:07 +02:00
|
|
|
subDirs, err := c.Directories(info.Abs, true)
|
2020-03-27 11:56:24 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
result = append(result, subDirs...)
|
|
|
|
}
|
2020-03-27 11:01:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Download downloads a single file to the given location.
|
2020-04-06 16:34:29 +02:00
|
|
|
func (c Client) Download(from, to string, force bool) error {
|
|
|
|
if _, err := os.Stat(to); err == nil && !force {
|
|
|
|
return fmt.Errorf("webdav: download skipped, %s already exists", to)
|
|
|
|
}
|
|
|
|
|
2020-03-27 11:01:41 +01:00
|
|
|
dir := path.Dir(to)
|
|
|
|
dirInfo, err := os.Stat(dir)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
// Create directory
|
|
|
|
if err := os.MkdirAll(dir, os.ModePerm); err != nil {
|
|
|
|
return fmt.Errorf("webdav: can't create %s (%s)", dir, err)
|
|
|
|
}
|
|
|
|
} else if !dirInfo.IsDir() {
|
2020-05-07 12:33:09 +02:00
|
|
|
return fmt.Errorf("webdav: %s is not a folder", dir)
|
2020-03-27 11:01:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bytes, err := c.client.Read(from)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ioutil.WriteFile(to, bytes, 0644)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DownloadDir downloads all files from a remote to a local directory.
|
2020-04-06 16:34:29 +02:00
|
|
|
func (c Client) DownloadDir(from, to string, recursive, force bool) (errs []error) {
|
2020-03-27 11:01:41 +01:00
|
|
|
files, err := c.Files(from)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return append(errs, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, file := range files {
|
2020-04-02 18:17:07 +02:00
|
|
|
dest := to + string(os.PathSeparator) + file.Abs
|
2020-03-27 11:01:41 +01:00
|
|
|
|
|
|
|
if _, err := os.Stat(dest); err == nil {
|
|
|
|
// File exists
|
|
|
|
msg := fmt.Errorf("webdav: %s exists", dest)
|
|
|
|
errs = append(errs, msg)
|
|
|
|
log.Error(msg)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-04-06 16:34:29 +02:00
|
|
|
if err := c.Download(file.Abs, dest, force); err != nil {
|
2020-03-27 11:01:41 +01:00
|
|
|
msg := fmt.Errorf("webdav: %s", err)
|
|
|
|
errs = append(errs, msg)
|
|
|
|
log.Error(msg)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !recursive {
|
|
|
|
return errs
|
|
|
|
}
|
|
|
|
|
2020-03-27 11:56:24 +01:00
|
|
|
dirs, err := c.Directories(from, false)
|
2020-03-27 11:01:41 +01:00
|
|
|
|
|
|
|
for _, dir := range dirs {
|
2020-04-06 16:34:29 +02:00
|
|
|
errs = append(errs, c.DownloadDir(dir.Abs, to, true, force)...)
|
2020-03-27 11:01:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return errs
|
|
|
|
}
|
|
|
|
|
2020-04-01 12:00:45 +02:00
|
|
|
// CreateDir recursively creates directories if they don't exist.
|
|
|
|
func (c Client) CreateDir(dir string) error {
|
2020-04-07 12:51:01 +02:00
|
|
|
if dir == "" || dir == "/" || dir == "." {
|
2020-04-01 12:00:45 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.client.MkdirAll(dir, os.ModePerm)
|
|
|
|
}
|
|
|
|
|
2020-03-27 11:01:41 +01:00
|
|
|
// Upload uploads a single file to the remote server.
|
|
|
|
func (c Client) Upload(from, to string) error {
|
|
|
|
file, err := os.Open(from)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
return c.client.WriteStream(to, file, 0644)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete deletes a single file or directory on a remote server.
|
|
|
|
func (c Client) Delete(path string) error {
|
|
|
|
return c.client.Remove(path)
|
|
|
|
}
|