2021-01-31 09:08:52 +01:00
|
|
|
package photoprism
|
2021-01-31 07:41:23 +01:00
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2021-02-09 20:17:06 +01:00
|
|
|
"io/ioutil"
|
2021-02-04 06:01:21 +01:00
|
|
|
"net/http"
|
2021-02-09 23:45:36 +01:00
|
|
|
"net/url"
|
2021-02-04 06:01:21 +01:00
|
|
|
"strings"
|
2021-01-31 09:08:52 +01:00
|
|
|
|
2021-02-12 05:14:42 +01:00
|
|
|
v1 "github.com/kris-nova/photoprism-client-go/api/v1"
|
2021-02-04 06:01:21 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// DefaultContentType is the content type header the API expects
|
2021-02-09 23:45:36 +01:00
|
|
|
APIContentType string = "application/json; charset=utf-8"
|
2021-02-04 06:01:21 +01:00
|
|
|
|
|
|
|
// Default Host Configuration
|
2021-02-09 23:45:36 +01:00
|
|
|
APIAuthHeaderKey string = "X-Session-Id"
|
2021-02-04 06:01:21 +01:00
|
|
|
)
|
|
|
|
|
2021-02-09 23:45:36 +01:00
|
|
|
// New is used to create a new Client to authenticate with
|
|
|
|
// Photoprism.
|
2021-02-09 20:17:06 +01:00
|
|
|
func New(connectionString string) *Client {
|
2021-02-13 22:30:08 +01:00
|
|
|
for strings.HasSuffix(connectionString, "/") {
|
|
|
|
connectionString = connectionString[:len(connectionString)-1]
|
|
|
|
}
|
|
|
|
|
2021-02-09 20:17:06 +01:00
|
|
|
c := &Client{
|
2021-02-09 23:45:36 +01:00
|
|
|
contentType: APIContentType,
|
|
|
|
connectionString: connectionString,
|
2021-02-04 06:01:21 +01:00
|
|
|
}
|
2021-02-09 20:17:06 +01:00
|
|
|
return c
|
2021-02-04 06:01:21 +01:00
|
|
|
}
|
|
|
|
|
2021-02-13 22:30:08 +01:00
|
|
|
func (c *Client) ConnectionString() string {
|
|
|
|
return c.connectionString
|
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// Client represents a client to a Photoprism application
|
2021-01-31 09:08:52 +01:00
|
|
|
type Client struct {
|
2021-02-09 23:45:36 +01:00
|
|
|
v1client *v1.V1Client
|
2021-02-04 06:01:21 +01:00
|
|
|
authenticator ClientAuthenticator
|
|
|
|
contentType string
|
|
|
|
connectionString string
|
2021-02-09 23:45:36 +01:00
|
|
|
connectionURL *url.URL
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// ClientAuthenticator is used to store the secret
|
|
|
|
// data for authenticating with the Photoprism API
|
2021-01-31 09:08:52 +01:00
|
|
|
type ClientAuthenticator interface {
|
|
|
|
getKey() string
|
|
|
|
getSecret() string
|
2021-02-04 06:01:21 +01:00
|
|
|
JSON() ([]byte, error)
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
|
2021-02-09 23:45:36 +01:00
|
|
|
// ClientAuthLogin holds secret login information
|
2021-01-31 09:08:52 +01:00
|
|
|
type ClientAuthLogin struct {
|
2021-02-09 23:45:36 +01:00
|
|
|
authPayload
|
|
|
|
}
|
|
|
|
|
|
|
|
type authPayload struct {
|
2021-02-04 06:01:21 +01:00
|
|
|
Username string `json:"username"`
|
|
|
|
Password string `json:"password"`
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// NewClientAuthLogin is used to build a new login struct
|
2021-01-31 09:08:52 +01:00
|
|
|
func NewClientAuthLogin(user, pass string) ClientAuthenticator {
|
|
|
|
return &ClientAuthLogin{
|
2021-02-09 23:45:36 +01:00
|
|
|
authPayload: authPayload{
|
|
|
|
Username: user,
|
|
|
|
Password: pass,
|
|
|
|
},
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-09 23:45:36 +01:00
|
|
|
// getKey is used internally to get the key with any modifiers
|
2021-01-31 09:08:52 +01:00
|
|
|
func (c *ClientAuthLogin) getKey() string {
|
2021-02-09 23:45:36 +01:00
|
|
|
return c.authPayload.Username
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
2021-02-09 23:45:36 +01:00
|
|
|
|
|
|
|
// getKey is used internally to get the secret with any modifiers
|
2021-01-31 09:08:52 +01:00
|
|
|
func (c *ClientAuthLogin) getSecret() string {
|
2021-02-09 23:45:36 +01:00
|
|
|
return c.authPayload.Password
|
2021-01-31 07:41:23 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// JSON is used to marshal the fields to JSON
|
|
|
|
func (c *ClientAuthLogin) JSON() ([]byte, error) {
|
|
|
|
return json.Marshal(c)
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// V1 is used to access the V1 version of the Photoprism API
|
2021-02-09 23:45:36 +01:00
|
|
|
func (c *Client) V1() *v1.V1Client {
|
2021-02-04 06:01:21 +01:00
|
|
|
return c.v1client
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// Login is used to attempt to authenticate with the Photoprism API
|
2021-02-09 20:17:06 +01:00
|
|
|
func (c *Client) Auth(auth ClientAuthenticator) error {
|
|
|
|
c.authenticator = auth
|
2021-02-09 23:45:36 +01:00
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// @kris-nova We are returning V1 by default
|
|
|
|
return c.LoginV1()
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// POST /api/v1/session
|
|
|
|
//
|
|
|
|
// Data: {username: "admin", password: "missy"}
|
|
|
|
func (c *Client) LoginV1() error {
|
2021-02-09 23:45:36 +01:00
|
|
|
// Auth wil also validate the connection string
|
|
|
|
// We do this here so that New() will never return
|
|
|
|
// an error.
|
|
|
|
url, err := url.Parse(c.connectionString)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to parse connection string url [%s]: %v", c.connectionString, err)
|
|
|
|
}
|
|
|
|
c.connectionURL = url
|
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
body, err := c.authenticator.JSON()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("JSON marshal error: %v", err)
|
|
|
|
}
|
|
|
|
buffer := bytes.NewBuffer(body)
|
2021-02-09 20:17:06 +01:00
|
|
|
resp, err := http.Post(c.Endpoint("api/v1/session"), c.contentType, buffer)
|
2021-02-04 06:01:21 +01:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("authentication error: %v", err)
|
|
|
|
}
|
2021-02-09 20:17:06 +01:00
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to parse body for [%d] response: %v", resp.StatusCode, err)
|
|
|
|
}
|
|
|
|
return fmt.Errorf("login error [%d] %s", resp.StatusCode, body)
|
|
|
|
}
|
2021-02-10 03:25:46 +01:00
|
|
|
|
|
|
|
// --- JSON Auth Response on to Options ---
|
|
|
|
cfg := &Config{
|
|
|
|
Config: &Options{},
|
|
|
|
}
|
|
|
|
bytes, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to parse auth body: %v", err)
|
|
|
|
}
|
|
|
|
err = json.Unmarshal(bytes, &cfg)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to json unmarshal auth body: %v", err)
|
|
|
|
}
|
|
|
|
|
2021-02-09 23:45:36 +01:00
|
|
|
token := resp.Header.Get(APIAuthHeaderKey)
|
2021-02-09 20:17:06 +01:00
|
|
|
if token == "" {
|
|
|
|
return fmt.Errorf("missing auth token from successful login")
|
|
|
|
}
|
2021-02-10 03:25:46 +01:00
|
|
|
c.v1client = v1.New(c.connectionURL, token, cfg.Config.DownloadToken)
|
2021-02-04 06:01:21 +01:00
|
|
|
return nil
|
2021-01-31 07:41:23 +01:00
|
|
|
}
|
2021-01-31 09:08:52 +01:00
|
|
|
|
2021-02-04 06:01:21 +01:00
|
|
|
// Endpoint is used to calculate a FQN for a given API endpoint
|
|
|
|
// based on the API version and Host/Port
|
|
|
|
func (c *Client) Endpoint(str string) string {
|
|
|
|
if strings.HasPrefix("/", str) {
|
2021-02-09 23:45:36 +01:00
|
|
|
str = fmt.Sprintf("%s%s", c.connectionString, str)
|
|
|
|
} else {
|
|
|
|
str = fmt.Sprintf("%s/%s", c.connectionString, str)
|
2021-02-04 06:01:21 +01:00
|
|
|
}
|
2021-02-09 23:45:36 +01:00
|
|
|
//logger.Debug(str)
|
|
|
|
return str
|
2021-01-31 09:08:52 +01:00
|
|
|
}
|