158 lines
4.0 KiB
Go
158 lines
4.0 KiB
Go
// Copyright (c) 2024 Joshua Rich <joshua.rich@gmail.com>
|
|
//
|
|
// This software is released under the MIT License.
|
|
// https://opensource.org/licenses/MIT
|
|
|
|
package hass
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"github.com/go-resty/resty/v2"
|
|
)
|
|
|
|
var (
|
|
ErrInvalidURL = errors.New("invalid URL")
|
|
ErrInvalidClient = errors.New("invalid client")
|
|
ErrResponseMalformed = errors.New("malformed response")
|
|
ErrNoPrefs = errors.New("loading preferences failed")
|
|
defaultTimeout = 30 * time.Second
|
|
defaultRetry = func(r *resty.Response, _ error) bool {
|
|
return r.StatusCode() == http.StatusTooManyRequests
|
|
}
|
|
)
|
|
|
|
// APIError represents an error returned either by the HTTP layer or by the Home
|
|
// Assistant API. The StatusCode reflects the HTTP status code returned while
|
|
// Message and Code are additional and optional values returned from the Home
|
|
// Assistant API.
|
|
type APIError struct {
|
|
Code any `json:"code,omitempty"`
|
|
Message string `json:"message,omitempty"`
|
|
StatusCode int `json:"-"`
|
|
}
|
|
|
|
func (e *APIError) Error() string {
|
|
switch {
|
|
case e.Code != nil:
|
|
return fmt.Sprintf("%v: %s", e.Code, e.Message)
|
|
case e.StatusCode > 0:
|
|
return fmt.Sprintf("Status: %d", e.StatusCode)
|
|
default:
|
|
return e.Message
|
|
}
|
|
}
|
|
|
|
// GetRequest is a HTTP GET request.
|
|
type GetRequest any
|
|
|
|
// PostRequest is a HTTP POST request with the request body provided by Body().
|
|
//
|
|
//go:generate moq -out mock_PostRequest_test.go . PostRequest
|
|
type PostRequest interface {
|
|
RequestBody() json.RawMessage
|
|
}
|
|
|
|
// Authenticated represents a request that requires passing an authentication
|
|
// header with the value returned by Auth().
|
|
type Authenticated interface {
|
|
Auth() string
|
|
}
|
|
|
|
// Encrypted represents a request that should be encrypted with the secret
|
|
// provided by Secret().
|
|
type Encrypted interface {
|
|
Secret() string
|
|
}
|
|
|
|
//go:generate moq -out mock_Response_test.go . Response
|
|
type Response interface {
|
|
json.Unmarshaler
|
|
}
|
|
|
|
// ExecuteRequest sends an API request to Home Assistant. It supports either the
|
|
// REST or WebSocket API. By default and at a minimum, request are sent as GET
|
|
// requests and need to satisfy the GetRequest interface. To send a POST,
|
|
// satisfy the PostRequest interface. To add authentication where required,
|
|
// satisfy the Auth interface. To send an encrypted request, satisfy the Secret
|
|
// interface.
|
|
func ExecuteRequest(ctx context.Context, request any, response Response) error {
|
|
// TODO: handle nil response here
|
|
url := ContextGetURL(ctx)
|
|
if url == "" {
|
|
return ErrInvalidURL
|
|
}
|
|
|
|
client := ContextGetClient(ctx)
|
|
if client == nil {
|
|
return ErrInvalidClient
|
|
}
|
|
|
|
var responseErr *APIError
|
|
var resp *resty.Response
|
|
var err error
|
|
cl := client.R().
|
|
SetContext(ctx).
|
|
SetError(&responseErr)
|
|
if a, ok := request.(Authenticated); ok {
|
|
cl = cl.SetAuthToken(a.Auth())
|
|
}
|
|
switch r := request.(type) {
|
|
case PostRequest:
|
|
log.Trace().
|
|
Str("method", "POST").
|
|
RawJSON("body", r.RequestBody()).
|
|
Time("sent_at", time.Now()).
|
|
Msg("Sending request.")
|
|
resp, err = cl.
|
|
SetBody(r.RequestBody()).
|
|
Post(url)
|
|
case GetRequest:
|
|
log.Trace().
|
|
Str("method", "GET").
|
|
Time("sent_at", time.Now()).
|
|
Msg("Sending request.")
|
|
resp, err = cl.
|
|
Get(url)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
log.Trace().Err(err).
|
|
Int("statuscode", resp.StatusCode()).
|
|
Str("status", resp.Status()).
|
|
Str("protocol", resp.Proto()).
|
|
Dur("time", resp.Time()).
|
|
Time("received_at", resp.ReceivedAt()).
|
|
RawJSON("body", resp.Body()).
|
|
Msg("Response received.")
|
|
if resp.IsError() {
|
|
if responseErr != nil {
|
|
responseErr.StatusCode = resp.StatusCode()
|
|
return responseErr
|
|
} else {
|
|
return &APIError{
|
|
StatusCode: resp.StatusCode(),
|
|
Message: resp.Status(),
|
|
}
|
|
}
|
|
}
|
|
if err := response.UnmarshalJSON(resp.Body()); err != nil {
|
|
return errors.Join(ErrResponseMalformed, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func NewDefaultHTTPClient() *resty.Client {
|
|
return resty.New().
|
|
SetTimeout(defaultTimeout).
|
|
AddRetryCondition(defaultRetry)
|
|
}
|