| // - |
| // ========================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 |
| } |