Files
photoprism/internal/commands/cluster_helpers.go
2025-09-19 06:52:45 +02:00

56 lines
1.7 KiB
Go

package commands
import (
"bytes"
"encoding/json"
"fmt"
"net/http"
"net/url"
"strings"
"github.com/photoprism/photoprism/internal/service/cluster"
"github.com/photoprism/photoprism/pkg/service/http/header"
)
// obtainClientCredentialsViaRegister calls the portal register endpoint using a join token
// to (re)register the node, rotating the secret when necessary, and returns client id/secret.
func obtainClientCredentialsViaRegister(portalURL, joinToken, nodeName string) (id, secret string, err error) {
u, err := url.Parse(strings.TrimRight(portalURL, "/"))
if err != nil || u.Scheme == "" || u.Host == "" {
return "", "", fmt.Errorf("invalid portal-url: %s", portalURL)
}
endpoint := *u
endpoint.Path = strings.TrimRight(endpoint.Path, "/") + "/api/v1/cluster/nodes/register"
reqBody := map[string]any{
"nodeName": nodeName,
"nodeRole": cluster.RoleInstance,
"rotateSecret": true,
}
b, _ := json.Marshal(reqBody)
req, _ := http.NewRequest(http.MethodPost, endpoint.String(), bytes.NewReader(b))
req.Header.Set("Content-Type", "application/json")
header.SetAuthorization(req, joinToken)
resp, err := (&http.Client{}).Do(req)
if err != nil {
return "", "", err
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusCreated && resp.StatusCode != http.StatusConflict {
return "", "", fmt.Errorf("%s", resp.Status)
}
var regResp cluster.RegisterResponse
if err := json.NewDecoder(resp.Body).Decode(&regResp); err != nil {
return "", "", err
}
id = regResp.Node.ID
if regResp.Secrets != nil {
secret = regResp.Secrets.NodeSecret
}
if id == "" || secret == "" {
return "", "", fmt.Errorf("missing client credentials in response")
}
return id, secret, nil
}