mirror of
https://github.com/davidallendj/magellan.git
synced 2025-12-20 11:37:01 -07:00
Refactored how clients work to reduce hard-coded dependencies
This commit is contained in:
parent
8e59885f55
commit
c5a348562b
3 changed files with 100 additions and 57 deletions
|
|
@ -13,38 +13,36 @@ import (
|
||||||
"github.com/OpenCHAMI/magellan/internal/util"
|
"github.com/OpenCHAMI/magellan/internal/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Option func(*Client)
|
type Option[T Client] func(client T)
|
||||||
|
|
||||||
// The 'Client' struct is a wrapper around the default http.Client
|
// The 'Client' struct is a wrapper around the default http.Client
|
||||||
// that provides an extended API to work with functional options.
|
// that provides an extended API to work with functional options.
|
||||||
// It also provides functions that work with `collect` data.
|
// It also provides functions that work with `collect` data.
|
||||||
type Client struct {
|
type Client interface {
|
||||||
*http.Client
|
Name() string
|
||||||
|
GetClient() *http.Client
|
||||||
|
RootEndpoint(endpoint string) string
|
||||||
|
|
||||||
|
// functions needed to make request
|
||||||
|
Add(data util.HTTPBody, headers util.HTTPHeader) error
|
||||||
|
Update(data util.HTTPBody, headers util.HTTPHeader) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewClient() creates a new client
|
// NewClient() creates a new client
|
||||||
func NewClient(opts ...Option) *Client {
|
func NewClient[T Client](opts ...func(T)) T {
|
||||||
client := &Client{
|
client := new(T)
|
||||||
Client: http.DefaultClient,
|
|
||||||
}
|
|
||||||
for _, opt := range opts {
|
for _, opt := range opts {
|
||||||
opt(client)
|
opt(*client)
|
||||||
}
|
}
|
||||||
return client
|
return *client
|
||||||
}
|
}
|
||||||
|
|
||||||
func WithHttpClient(httpClient *http.Client) Option {
|
func WithCertPool[T Client](certPool *x509.CertPool) func(T) {
|
||||||
return func(c *Client) {
|
|
||||||
c.Client = httpClient
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func WithCertPool(certPool *x509.CertPool) Option {
|
|
||||||
if certPool == nil {
|
if certPool == nil {
|
||||||
return func(c *Client) {}
|
return func(client T) {}
|
||||||
}
|
}
|
||||||
return func(c *Client) {
|
return func(client T) {
|
||||||
c.Client.Transport = &http.Transport{
|
client.GetClient().Transport = &http.Transport{
|
||||||
TLSClientConfig: &tls.Config{
|
TLSClientConfig: &tls.Config{
|
||||||
RootCAs: certPool,
|
RootCAs: certPool,
|
||||||
InsecureSkipVerify: true,
|
InsecureSkipVerify: true,
|
||||||
|
|
@ -60,20 +58,20 @@ func WithCertPool(certPool *x509.CertPool) Option {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func WithSecureTLS(certPath string) Option {
|
func WithSecureTLS[T Client](certPath string) func(T) {
|
||||||
cacert, err := os.ReadFile(certPath)
|
cacert, err := os.ReadFile(certPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return func(c *Client) {}
|
return func(client T) {}
|
||||||
}
|
}
|
||||||
certPool := x509.NewCertPool()
|
certPool := x509.NewCertPool()
|
||||||
certPool.AppendCertsFromPEM(cacert)
|
certPool.AppendCertsFromPEM(cacert)
|
||||||
return WithCertPool(certPool)
|
return WithCertPool[T](certPool)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Post() is a simplified wrapper function that packages all of the
|
// Post() is a simplified wrapper function that packages all of the
|
||||||
// that marshals a mapper into a JSON-formatted byte array, and then performs
|
// that marshals a mapper into a JSON-formatted byte array, and then performs
|
||||||
// a request to the specified URL.
|
// a request to the specified URL.
|
||||||
func (c *Client) Post(url string, data map[string]any, header util.HTTPHeader) (*http.Response, util.HTTPBody, error) {
|
func (c *MagellanClient) Post(url string, data map[string]any, header util.HTTPHeader) (*http.Response, util.HTTPBody, error) {
|
||||||
// serialize data into byte array
|
// serialize data into byte array
|
||||||
body, err := json.Marshal(data)
|
body, err := json.Marshal(data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
@ -81,7 +79,3 @@ func (c *Client) Post(url string, data map[string]any, header util.HTTPHeader) (
|
||||||
}
|
}
|
||||||
return util.MakeRequest(c.Client, url, http.MethodPost, body, header)
|
return util.MakeRequest(c.Client, url, http.MethodPost, body, header)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) MakeRequest(url string, method string, body util.HTTPBody, header util.HTTPHeader) (*http.Response, util.HTTPBody, error) {
|
|
||||||
return util.MakeRequest(c.Client, url, method, body, header)
|
|
||||||
}
|
|
||||||
|
|
|
||||||
55
pkg/client/default.go
Normal file
55
pkg/client/default.go
Normal file
|
|
@ -0,0 +1,55 @@
|
||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/OpenCHAMI/magellan/internal/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MagellanClient struct {
|
||||||
|
*http.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *MagellanClient) Name() string {
|
||||||
|
return "default"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add() is the default function that is called with a client with no implementation.
|
||||||
|
// This function will simply make a HTTP request including all the data passed as
|
||||||
|
// the first argument with no data processing or manipulation. The function sends
|
||||||
|
// the data to a set callback URL (which may be changed to use a configurable value
|
||||||
|
// instead).
|
||||||
|
func (c *MagellanClient) Add(data util.HTTPBody, headers util.HTTPHeader) error {
|
||||||
|
if data == nil {
|
||||||
|
return fmt.Errorf("no data found")
|
||||||
|
}
|
||||||
|
|
||||||
|
path := "/inventory/add"
|
||||||
|
res, body, err := util.MakeRequest(c.Client, path, http.MethodPost, data, headers)
|
||||||
|
if res != nil {
|
||||||
|
statusOk := res.StatusCode >= 200 && res.StatusCode < 300
|
||||||
|
if !statusOk {
|
||||||
|
return fmt.Errorf("returned status code %d when POST'ing to endpoint", res.StatusCode)
|
||||||
|
}
|
||||||
|
fmt.Printf("%v (%v)\n%s\n", path, res.Status, string(body))
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *MagellanClient) Update(data util.HTTPBody, headers util.HTTPHeader) error {
|
||||||
|
if data == nil {
|
||||||
|
return fmt.Errorf("no data found")
|
||||||
|
}
|
||||||
|
|
||||||
|
path := "/inventory/update"
|
||||||
|
res, body, err := util.MakeRequest(c.Client, path, http.MethodPut, data, headers)
|
||||||
|
if res != nil {
|
||||||
|
statusOk := res.StatusCode >= 200 && res.StatusCode < 300
|
||||||
|
if !statusOk {
|
||||||
|
return fmt.Errorf("returned status code %d when PUT'ing to endpoint", res.StatusCode)
|
||||||
|
}
|
||||||
|
fmt.Printf("%v (%v)\n%s\n", path, res.Status, string(body))
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
@ -11,41 +11,39 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
Host = "http://localhost"
|
Host = "http://localhost:27779"
|
||||||
BaseEndpoint = "/hsm/v2"
|
BaseEndpoint = "/hsm/v2"
|
||||||
Port = 27779
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func (c *Client) GetRedfishEndpoints(header util.HTTPHeader) error {
|
type SmdClient struct {
|
||||||
url := makeEndpointUrl("/Inventory/RedfishEndpoints")
|
*http.Client
|
||||||
_, body, err := util.MakeRequest(c.Client, url, http.MethodGet, nil, header)
|
Host string
|
||||||
if err != nil {
|
Xname string
|
||||||
return fmt.Errorf("failed to get endpoint: %v", err)
|
|
||||||
}
|
|
||||||
// fmt.Println(res)
|
|
||||||
fmt.Println(string(body))
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) GetComponentEndpoint(xname string) error {
|
func (c SmdClient) Name() string {
|
||||||
url := makeEndpointUrl("/Inventory/ComponentsEndpoints/" + xname)
|
return "smd"
|
||||||
res, body, err := c.MakeRequest(url, "GET", nil, nil)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to get endpoint: %v", err)
|
|
||||||
}
|
|
||||||
fmt.Println(res)
|
|
||||||
fmt.Println(string(body))
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) AddRedfishEndpoint(data map[string]any, headers util.HTTPHeader) error {
|
func (c SmdClient) RootEndpoint(endpoint string) string {
|
||||||
|
return fmt.Sprintf("/hsm/v2/%s%s", Host, endpoint)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c SmdClient) GetClient() *http.Client {
|
||||||
|
return c.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add() has a similar function definition to that of the default implementation,
|
||||||
|
// but also allows further customization and data/header manipulation that would
|
||||||
|
// be specific and/or unique to SMD's API.
|
||||||
|
func (c SmdClient) Add(data util.HTTPBody, headers util.HTTPHeader) error {
|
||||||
if data == nil {
|
if data == nil {
|
||||||
return fmt.Errorf("failed to add redfish endpoint: no data found")
|
return fmt.Errorf("failed to add redfish endpoint: no data found")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add redfish endpoint via POST `/hsm/v2/Inventory/RedfishEndpoints` endpoint
|
// Add redfish endpoint via POST `/hsm/v2/Inventory/RedfishEndpoints` endpoint
|
||||||
url := makeEndpointUrl("/Inventory/RedfishEndpoints")
|
url := c.RootEndpoint("/Inventory/RedfishEndpoints")
|
||||||
res, body, err := c.Post(url, data, headers)
|
res, body, err := util.MakeRequest(c.Client, url, http.MethodPost, data, headers)
|
||||||
if res != nil {
|
if res != nil {
|
||||||
statusOk := res.StatusCode >= 200 && res.StatusCode < 300
|
statusOk := res.StatusCode >= 200 && res.StatusCode < 300
|
||||||
if !statusOk {
|
if !statusOk {
|
||||||
|
|
@ -56,13 +54,13 @@ func (c *Client) AddRedfishEndpoint(data map[string]any, headers util.HTTPHeader
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) UpdateRedfishEndpoint(xname string, data []byte, headers map[string]string) error {
|
func (c SmdClient) Update(data util.HTTPBody, headers util.HTTPHeader) error {
|
||||||
if data == nil {
|
if data == nil {
|
||||||
return fmt.Errorf("failed to add redfish endpoint: no data found")
|
return fmt.Errorf("failed to add redfish endpoint: no data found")
|
||||||
}
|
}
|
||||||
// Update redfish endpoint via PUT `/hsm/v2/Inventory/RedfishEndpoints` endpoint
|
// Update redfish endpoint via PUT `/hsm/v2/Inventory/RedfishEndpoints` endpoint
|
||||||
url := makeEndpointUrl("/Inventory/RedfishEndpoints/" + xname)
|
url := c.RootEndpoint("/Inventory/RedfishEndpoints/" + c.Xname)
|
||||||
res, body, err := c.MakeRequest(url, "PUT", data, headers)
|
res, body, err := util.MakeRequest(c.Client, url, http.MethodPut, data, headers)
|
||||||
fmt.Printf("%v (%v)\n%s\n", url, res.Status, string(body))
|
fmt.Printf("%v (%v)\n%s\n", url, res.Status, string(body))
|
||||||
if res != nil {
|
if res != nil {
|
||||||
statusOk := res.StatusCode >= 200 && res.StatusCode < 300
|
statusOk := res.StatusCode >= 200 && res.StatusCode < 300
|
||||||
|
|
@ -72,7 +70,3 @@ func (c *Client) UpdateRedfishEndpoint(xname string, data []byte, headers map[st
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeEndpointUrl(endpoint string) string {
|
|
||||||
return Host + ":" + fmt.Sprint(Port) + BaseEndpoint + endpoint
|
|
||||||
}
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue