blob: a7582c2ba71150e5a9608792a2ef88892c1f480d [file] [log] [blame]
// -
// ========================LICENSE_START=================================
// O-RAN-SC
// %%
// Copyright (C) 2021: Nordix Foundation
// %%
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// ========================LICENSE_END===================================
//
package restclient
import (
"bytes"
"crypto/tls"
"fmt"
"io"
"math"
"net/http"
"net/url"
"time"
"github.com/hashicorp/go-retryablehttp"
log "github.com/sirupsen/logrus"
)
const ContentTypeJSON = "application/json"
const ContentTypePlain = "text/plain"
// HTTPClient interface
type HTTPClient interface {
Get(url string) (*http.Response, error)
Do(*http.Request) (*http.Response, error)
}
type RequestError struct {
StatusCode int
Body []byte
}
func (pe RequestError) Error() string {
return fmt.Sprintf("Request failed due to error response with status: %v and body: %v", pe.StatusCode, string(pe.Body))
}
func Get(url string, client HTTPClient) ([]byte, error) {
if response, err := client.Get(url); err == nil {
if isResponseSuccess(response.StatusCode) {
defer response.Body.Close()
if responseData, err := io.ReadAll(response.Body); err == nil {
return responseData, nil
} else {
return nil, err
}
} else {
return nil, getRequestError(response)
}
} else {
return nil, err
}
}
func Put(url string, body []byte, client HTTPClient) error {
return do(http.MethodPut, url, body, ContentTypeJSON, client)
}
func Post(url string, body []byte, contentType string, client HTTPClient) error {
return do(http.MethodPost, url, body, contentType, client)
}
func do(method string, url string, body []byte, contentType string, client HTTPClient) error {
if req, reqErr := http.NewRequest(method, url, bytes.NewBuffer(body)); reqErr == nil {
req.Header.Set("Content-Type", contentType)
if response, respErr := client.Do(req); respErr == nil {
if isResponseSuccess(response.StatusCode) {
return nil
} else {
return getRequestError(response)
}
} else {
return respErr
}
} else {
return reqErr
}
}
func isResponseSuccess(statusCode int) bool {
return statusCode >= http.StatusOK && statusCode <= 299
}
func getRequestError(response *http.Response) RequestError {
defer response.Body.Close()
responseData, _ := io.ReadAll(response.Body)
putError := RequestError{
StatusCode: response.StatusCode,
Body: responseData,
}
return putError
}
func CreateClientCertificate(certPath string, keyPath string) (tls.Certificate, error) {
if cert, err := tls.LoadX509KeyPair(certPath, keyPath); err == nil {
return cert, nil
} else {
return tls.Certificate{}, fmt.Errorf("cannot create x509 keypair from cert file %s and key file %s due to: %v", certPath, keyPath, err)
}
}
func CreateRetryClient(cert tls.Certificate) *http.Client {
rawRetryClient := retryablehttp.NewClient()
rawRetryClient.Logger = leveledLogger{}
rawRetryClient.RetryWaitMax = time.Minute
rawRetryClient.RetryMax = math.MaxInt
rawRetryClient.HTTPClient.Transport = getSecureTransportWithoutVerify(cert)
client := rawRetryClient.StandardClient()
return client
}
func CreateClientWithoutRetry(cert tls.Certificate, timeout time.Duration) *http.Client {
return &http.Client{
Timeout: timeout,
Transport: getSecureTransportWithoutVerify(cert),
}
}
func getSecureTransportWithoutVerify(cert tls.Certificate) *http.Transport {
return &http.Transport{
TLSClientConfig: &tls.Config{
Certificates: []tls.Certificate{
cert,
},
InsecureSkipVerify: true,
},
}
}
func IsUrlSecure(configUrl string) bool {
u, _ := url.Parse(configUrl)
return u.Scheme == "https"
}
// Used to get leveled logging in the RetryClient
type leveledLogger struct {
}
func (ll leveledLogger) Error(msg string, keysAndValues ...interface{}) {
log.WithFields(getFields(keysAndValues)).Error(msg)
}
func (ll leveledLogger) Info(msg string, keysAndValues ...interface{}) {
log.WithFields(getFields(keysAndValues)).Info(msg)
}
func (ll leveledLogger) Debug(msg string, keysAndValues ...interface{}) {
log.WithFields(getFields(keysAndValues)).Debug(msg)
}
func (ll leveledLogger) Warn(msg string, keysAndValues ...interface{}) {
log.WithFields(getFields(keysAndValues)).Warn(msg)
}
func getFields(keysAndValues []interface{}) log.Fields {
fields := log.Fields{}
for i := 0; i < len(keysAndValues); i = i + 2 {
fields[fmt.Sprint(keysAndValues[i])] = keysAndValues[i+1]
}
return fields
}