2020-10-03 13:50:30 +02:00
|
|
|
package pro
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/aes"
|
|
|
|
"crypto/cipher"
|
|
|
|
"crypto/sha1"
|
|
|
|
"crypto/sha256"
|
|
|
|
"encoding/hex"
|
|
|
|
"encoding/json"
|
2020-10-07 12:47:12 +02:00
|
|
|
"errors"
|
2020-10-03 13:50:30 +02:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
2020-10-07 12:22:45 +02:00
|
|
|
"path/filepath"
|
2020-10-03 13:50:30 +02:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/photoprism/photoprism/internal/pro/places"
|
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Config represents photoprism.pro api credentials for maps & geodata.
|
|
|
|
type Config struct {
|
2020-10-07 12:22:45 +02:00
|
|
|
Key string `json:"key" yaml:"key"`
|
|
|
|
Secret string `json:"secret" yaml:"secret"`
|
|
|
|
Session string `json:"session" yaml:"session"`
|
|
|
|
Status string `json:"status" yaml:"status"`
|
|
|
|
Version string `json:"version" yaml:"version"`
|
|
|
|
FileName string `json:"-" yaml:"-"`
|
2020-10-03 13:50:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewConfig creates a new photoprism.pro api credentials instance.
|
2020-10-07 12:22:45 +02:00
|
|
|
func NewConfig(version string, fileName string) *Config {
|
2020-10-03 13:50:30 +02:00
|
|
|
return &Config{
|
2020-10-07 12:22:45 +02:00
|
|
|
Key: "",
|
|
|
|
Secret: "",
|
|
|
|
Session: "",
|
|
|
|
Status: "",
|
|
|
|
Version: version,
|
|
|
|
FileName: fileName,
|
2020-10-03 13:50:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
// MapKey returns the maps api key.
|
|
|
|
func (c *Config) MapKey() string {
|
|
|
|
if sess, err := c.DecodeSession(); err != nil {
|
|
|
|
return ""
|
|
|
|
} else {
|
|
|
|
return sess.MapKey
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-03 13:50:30 +02:00
|
|
|
// Propagate updates photoprism.pro api credentials in other packages.
|
2020-10-04 04:47:54 +02:00
|
|
|
func (c *Config) Propagate() {
|
|
|
|
places.Key = c.Key
|
|
|
|
places.Secret = c.Secret
|
2020-10-03 13:50:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sanitize verifies and sanitizes photoprism.pro api credentials.
|
2020-10-04 04:47:54 +02:00
|
|
|
func (c *Config) Sanitize() {
|
|
|
|
c.Key = strings.ToLower(c.Key)
|
|
|
|
|
|
|
|
if c.Secret != "" {
|
|
|
|
if c.Key != fmt.Sprintf("%x", sha1.Sum([]byte(c.Secret))) {
|
|
|
|
c.Key = ""
|
|
|
|
c.Secret = ""
|
|
|
|
c.Session = ""
|
|
|
|
c.Status = ""
|
2020-10-03 13:50:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeSession decodes photoprism.pro api session data.
|
2020-10-04 04:47:54 +02:00
|
|
|
func (c *Config) DecodeSession() (Session, error) {
|
|
|
|
c.Sanitize()
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
result := Session{}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
if c.Session == "" {
|
2020-10-03 13:50:30 +02:00
|
|
|
return result, fmt.Errorf("empty session")
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
s, err := hex.DecodeString(c.Session)
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hash := sha256.New()
|
2020-10-04 04:47:54 +02:00
|
|
|
hash.Write([]byte(c.Secret))
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
var b []byte
|
|
|
|
|
|
|
|
block, err := aes.NewCipher(hash.Sum(b))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
iv := s[:aes.BlockSize]
|
|
|
|
|
|
|
|
plaintext := make([]byte, len(s))
|
|
|
|
|
|
|
|
stream := cipher.NewCTR(block, iv)
|
|
|
|
stream.XORKeyStream(plaintext, s[aes.BlockSize:])
|
|
|
|
|
|
|
|
plaintext = bytes.Trim(plaintext, "\x00")
|
|
|
|
|
|
|
|
if err := json.Unmarshal(plaintext, &result); err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Refresh updates photoprism.pro api credentials.
|
2020-10-04 04:47:54 +02:00
|
|
|
func (c *Config) Refresh() (err error) {
|
|
|
|
mutex.Lock()
|
|
|
|
defer mutex.Unlock()
|
|
|
|
|
2020-10-07 12:22:45 +02:00
|
|
|
if err := os.MkdirAll(filepath.Dir(c.FileName), os.ModePerm); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
c.Sanitize()
|
2020-10-03 13:50:30 +02:00
|
|
|
client := &http.Client{Timeout: 60 * time.Second}
|
|
|
|
url := ApiURL
|
|
|
|
method := http.MethodPost
|
|
|
|
var req *http.Request
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
if c.Key != "" {
|
|
|
|
url = fmt.Sprintf(ApiURL+"/%s", c.Key)
|
2020-10-03 13:50:30 +02:00
|
|
|
method = http.MethodPut
|
2020-10-08 08:52:03 +02:00
|
|
|
log.Debugf("getting updated api key for maps & places from %s", ApiHost())
|
2020-10-03 13:50:30 +02:00
|
|
|
} else {
|
2020-10-08 08:52:03 +02:00
|
|
|
log.Debugf("requesting api key for maps & places from %s", ApiHost())
|
2020-10-03 13:50:30 +02:00
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
if j, err := json.Marshal(NewRequest(c.Version)); err != nil {
|
2020-10-03 13:50:30 +02:00
|
|
|
return err
|
|
|
|
} else if req, err = http.NewRequest(method, url, bytes.NewReader(j)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Header.Add("Content-Type", "application/json")
|
|
|
|
|
|
|
|
var r *http.Response
|
|
|
|
|
|
|
|
for i := 0; i < 3; i++ {
|
|
|
|
r, err = client.Do(req)
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
} else if r.StatusCode >= 400 {
|
2020-10-08 08:52:03 +02:00
|
|
|
err = fmt.Errorf("getting api key from %s failed (error %d)", ApiHost(), r.StatusCode)
|
2020-10-03 13:50:30 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
err = json.NewDecoder(r.Body).Decode(c)
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load photoprism.pro api credentials from a YAML file.
|
2020-10-07 12:22:45 +02:00
|
|
|
func (c *Config) Load() error {
|
|
|
|
if !fs.FileExists(c.FileName) {
|
2020-10-08 08:52:03 +02:00
|
|
|
return fmt.Errorf("settings file not found: %s", txt.Quote(c.FileName))
|
2020-10-03 13:50:30 +02:00
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
mutex.Lock()
|
|
|
|
defer mutex.Unlock()
|
|
|
|
|
2020-10-07 12:22:45 +02:00
|
|
|
yamlConfig, err := ioutil.ReadFile(c.FileName)
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
if err := yaml.Unmarshal(yamlConfig, c); err != nil {
|
2020-10-03 13:50:30 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
c.Sanitize()
|
|
|
|
c.Propagate()
|
2020-10-03 13:50:30 +02:00
|
|
|
|
2020-10-07 12:47:12 +02:00
|
|
|
if sess, err := c.DecodeSession(); err != nil {
|
|
|
|
return err
|
|
|
|
} else if sess.Expired() {
|
|
|
|
return errors.New("session expired")
|
|
|
|
}
|
|
|
|
|
2020-10-03 13:50:30 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save photoprism.pro api credentials to a YAML file.
|
2020-10-07 12:22:45 +02:00
|
|
|
func (c *Config) Save() error {
|
2020-10-04 04:47:54 +02:00
|
|
|
mutex.Lock()
|
|
|
|
defer mutex.Unlock()
|
|
|
|
|
|
|
|
c.Sanitize()
|
2020-10-03 13:50:30 +02:00
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
data, err := yaml.Marshal(c)
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
c.Propagate()
|
2020-10-03 13:50:30 +02:00
|
|
|
|
2020-10-07 12:22:45 +02:00
|
|
|
if err := os.MkdirAll(filepath.Dir(c.FileName), os.ModePerm); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := ioutil.WriteFile(c.FileName, data, os.ModePerm); err != nil {
|
2020-10-03 13:50:30 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 04:47:54 +02:00
|
|
|
c.Propagate()
|
2020-10-03 13:50:30 +02:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|