2020-12-04 13:10:32 +01:00
|
|
|
package hub
|
2020-10-04 22:22:53 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"runtime"
|
|
|
|
"time"
|
2020-10-08 08:52:03 +02:00
|
|
|
|
|
|
|
"github.com/photoprism/photoprism/internal/form"
|
2024-01-10 12:21:43 +01:00
|
|
|
"github.com/photoprism/photoprism/pkg/header"
|
2020-10-08 08:52:03 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
2020-10-04 22:22:53 +02:00
|
|
|
)
|
|
|
|
|
2020-12-04 13:10:32 +01:00
|
|
|
var FeedbackURL = ServiceURL + "/%s/feedback"
|
2020-10-04 22:22:53 +02:00
|
|
|
|
|
|
|
type Feedback struct {
|
|
|
|
Category string `json:"Category"`
|
2020-10-05 07:40:11 +02:00
|
|
|
Subject string `json:"Subject"`
|
2020-10-04 22:22:53 +02:00
|
|
|
Message string `json:"Message"`
|
|
|
|
UserName string `json:"UserName"`
|
|
|
|
UserEmail string `json:"UserEmail"`
|
|
|
|
UserAgent string `json:"UserAgent"`
|
2020-10-05 07:40:11 +02:00
|
|
|
ApiKey string `json:"ApiKey"`
|
|
|
|
ClientVersion string `json:"ClientVersion"`
|
2020-12-05 06:21:16 +01:00
|
|
|
ClientSerial string `json:"ClientSerial"`
|
2020-10-05 07:40:11 +02:00
|
|
|
ClientOS string `json:"ClientOS"`
|
|
|
|
ClientArch string `json:"ClientArch"`
|
|
|
|
ClientCPU int `json:"ClientCPU"`
|
2022-03-02 14:16:49 +01:00
|
|
|
ClientEnv string `json:"ClientEnv"`
|
|
|
|
PartnerID string `json:"PartnerID"`
|
2020-10-04 22:22:53 +02:00
|
|
|
}
|
|
|
|
|
2020-12-04 13:10:32 +01:00
|
|
|
// NewFeedback creates a new hub feedback instance.
|
2022-03-02 14:16:49 +01:00
|
|
|
func NewFeedback(version, serial, env, partnerId string) *Feedback {
|
2020-10-04 22:22:53 +02:00
|
|
|
return &Feedback{
|
|
|
|
ClientVersion: version,
|
2020-12-05 06:21:16 +01:00
|
|
|
ClientSerial: serial,
|
2020-10-04 22:22:53 +02:00
|
|
|
ClientOS: runtime.GOOS,
|
|
|
|
ClientArch: runtime.GOARCH,
|
|
|
|
ClientCPU: runtime.NumCPU(),
|
2022-03-02 14:16:49 +01:00
|
|
|
ClientEnv: env,
|
|
|
|
PartnerID: partnerId,
|
2020-10-04 22:22:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-02 14:16:49 +01:00
|
|
|
// SendFeedback sends a feedback message.
|
2020-10-04 22:22:53 +02:00
|
|
|
func (c *Config) SendFeedback(f form.Feedback) (err error) {
|
2022-03-02 14:16:49 +01:00
|
|
|
feedback := NewFeedback(c.Version, c.Serial, c.Env, c.PartnerID)
|
2020-10-04 22:22:53 +02:00
|
|
|
feedback.Category = f.Category
|
2021-09-23 23:46:17 +02:00
|
|
|
feedback.Subject = txt.Shorten(f.Message, 50, "...")
|
2020-10-04 22:22:53 +02:00
|
|
|
feedback.Message = f.Message
|
|
|
|
feedback.UserName = f.UserName
|
|
|
|
feedback.UserEmail = f.UserEmail
|
|
|
|
feedback.UserAgent = f.UserAgent
|
2020-10-05 07:40:11 +02:00
|
|
|
feedback.ApiKey = c.Key
|
2020-10-04 22:22:53 +02:00
|
|
|
|
2022-03-27 21:37:11 +02:00
|
|
|
// Create new http.Client instance.
|
|
|
|
//
|
|
|
|
// NOTE: Timeout specifies a time limit for requests made by
|
|
|
|
// this Client. The timeout includes connection time, any
|
|
|
|
// redirects, and reading the response body. The timer remains
|
|
|
|
// running after Get, Head, Post, or Do return and will
|
|
|
|
// interrupt reading of the Response.Body.
|
2020-10-04 22:22:53 +02:00
|
|
|
client := &http.Client{Timeout: 60 * time.Second}
|
2022-03-27 21:37:11 +02:00
|
|
|
|
2020-10-04 22:22:53 +02:00
|
|
|
url := fmt.Sprintf(FeedbackURL, c.Key)
|
|
|
|
method := http.MethodPost
|
2021-11-20 19:14:00 +01:00
|
|
|
|
2020-10-04 22:22:53 +02:00
|
|
|
var req *http.Request
|
|
|
|
|
2020-10-08 08:52:03 +02:00
|
|
|
log.Debugf("sending feedback to %s", ApiHost())
|
2020-10-04 22:22:53 +02:00
|
|
|
|
|
|
|
if j, err := json.Marshal(feedback); err != nil {
|
|
|
|
return err
|
|
|
|
} else if req, err = http.NewRequest(method, url, bytes.NewReader(j)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-03-02 14:16:49 +01:00
|
|
|
// Set user agent.
|
|
|
|
if c.UserAgent != "" {
|
|
|
|
req.Header.Set("User-Agent", c.UserAgent)
|
|
|
|
} else {
|
|
|
|
req.Header.Set("User-Agent", "PhotoPrism/Test")
|
|
|
|
}
|
|
|
|
|
2020-10-04 22:22:53 +02:00
|
|
|
req.Header.Add("Accept-Language", f.UserLocales)
|
2024-01-10 12:21:43 +01:00
|
|
|
req.Header.Add(header.ContentType, header.ContentTypeJson)
|
2020-10-04 22:22:53 +02:00
|
|
|
|
|
|
|
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("sending feedback to %s failed (error %d)", ApiHost(), r.StatusCode)
|
2020-10-04 22:22:53 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = json.NewDecoder(r.Body).Decode(c)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|