mirror of
https://github.com/IBM/fp-go.git
synced 2025-11-27 22:28:29 +02:00
271 lines
7.0 KiB
Go
271 lines
7.0 KiB
Go
// Copyright (c) 2023 IBM Corp.
|
|
// All rights reserved.
|
|
//
|
|
// 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.
|
|
|
|
package builder
|
|
|
|
import (
|
|
"bytes"
|
|
"net/http"
|
|
"net/url"
|
|
"strconv"
|
|
|
|
ENDO "github.com/IBM/fp-go/endomorphism"
|
|
F "github.com/IBM/fp-go/function"
|
|
C "github.com/IBM/fp-go/http/content"
|
|
H "github.com/IBM/fp-go/http/headers"
|
|
IOE "github.com/IBM/fp-go/ioeither"
|
|
IOEH "github.com/IBM/fp-go/ioeither/http"
|
|
J "github.com/IBM/fp-go/json"
|
|
LZ "github.com/IBM/fp-go/lazy"
|
|
L "github.com/IBM/fp-go/optics/lens"
|
|
O "github.com/IBM/fp-go/option"
|
|
S "github.com/IBM/fp-go/string"
|
|
)
|
|
|
|
type (
|
|
Builder struct {
|
|
method O.Option[string]
|
|
url string
|
|
headers http.Header
|
|
body O.Option[IOE.IOEither[error, []byte]]
|
|
}
|
|
|
|
// Endomorphism returns an [ENDO.Endomorphism] that transforms a builder
|
|
Endomorphism = ENDO.Endomorphism[*Builder]
|
|
)
|
|
|
|
var (
|
|
// Default is the default builder
|
|
Default = &Builder{method: O.Some(defaultMethod()), headers: make(http.Header), body: noBody}
|
|
|
|
defaultMethod = F.Constant(http.MethodGet)
|
|
|
|
// Monoid is the [M.Monoid] for the [Endomorphism]
|
|
Monoid = ENDO.Monoid[*Builder]()
|
|
|
|
// Url is a [L.Lens] for the URL
|
|
Url = L.MakeLensRef((*Builder).GetUrl, (*Builder).SetUrl)
|
|
// Method is a [L.Lens] for the HTTP method
|
|
Method = L.MakeLensRef((*Builder).GetMethod, (*Builder).SetMethod)
|
|
// Body is a [L.Lens] for the request body
|
|
Body = L.MakeLensRef((*Builder).GetBody, (*Builder).SetBody)
|
|
// Headers is a [L.Lens] for the complete set of request headers
|
|
Headers = L.MakeLensRef((*Builder).GetHeaders, (*Builder).SetHeaders)
|
|
|
|
getHeader = F.Bind2of2((*Builder).GetHeader)
|
|
delHeader = F.Bind2of2((*Builder).DelHeader)
|
|
setHeader = F.Bind2of3((*Builder).SetHeader)
|
|
|
|
noHeader = O.None[string]()
|
|
noBody = O.None[IOE.IOEither[error, []byte]]()
|
|
|
|
// WithMethod creates a [BuilderBuilder] for a certain method
|
|
WithMethod = Method.Set
|
|
// WithUrl creates a [BuilderBuilder] for a certain method
|
|
WithUrl = Url.Set
|
|
// WithHeaders creates a [BuilderBuilder] for a set of headers
|
|
WithHeaders = Headers.Set
|
|
// WithBody creates a [BuilderBuilder] for a request body
|
|
WithBody = F.Flow2(
|
|
O.Of[IOE.IOEither[error, []byte]],
|
|
Body.Set,
|
|
)
|
|
// WithBytes creates a [BuilderBuilder] for a request body using bytes
|
|
WithBytes = F.Flow2(
|
|
IOE.Of[error, []byte],
|
|
WithBody,
|
|
)
|
|
// WithContentType adds the [H.ContentType] header
|
|
WithContentType = WithHeader(H.ContentType)
|
|
// WithAuthorization adds the [H.Authorization] header
|
|
WithAuthorization = WithHeader(H.Authorization)
|
|
|
|
// WithGet adds the [http.MethodGet] method
|
|
WithGet = WithMethod(http.MethodGet)
|
|
// WithPost adds the [http.MethodPost] method
|
|
WithPost = WithMethod(http.MethodPost)
|
|
// WithPut adds the [http.MethodPut] method
|
|
WithPut = WithMethod(http.MethodPut)
|
|
// WithDelete adds the [http.MethodDelete] method
|
|
WithDelete = WithMethod(http.MethodDelete)
|
|
|
|
// WithBearer creates a [BuilderBuilder] to add a Bearer [H.Authorization] header
|
|
WithBearer = F.Flow2(
|
|
S.Format[string]("Bearer %s"),
|
|
WithAuthorization,
|
|
)
|
|
|
|
// Requester creates a requester from a builder
|
|
Requester = (*Builder).Requester
|
|
|
|
// WithoutBody creates a [BuilderBuilder] to remove the body
|
|
WithoutBody = F.Pipe1(
|
|
noBody,
|
|
Body.Set,
|
|
)
|
|
)
|
|
|
|
func (builder *Builder) clone() *Builder {
|
|
cpy := *builder
|
|
cpy.headers = cpy.headers.Clone()
|
|
return &cpy
|
|
}
|
|
|
|
func (builder *Builder) GetUrl() string {
|
|
return builder.url
|
|
}
|
|
|
|
func (builder *Builder) GetMethod() string {
|
|
return F.Pipe1(
|
|
builder.method,
|
|
O.GetOrElse(defaultMethod),
|
|
)
|
|
}
|
|
|
|
func (builder *Builder) GetHeaders() http.Header {
|
|
return builder.headers
|
|
}
|
|
|
|
func (builder *Builder) GetBody() O.Option[IOE.IOEither[error, []byte]] {
|
|
return builder.body
|
|
}
|
|
|
|
func (builder *Builder) SetMethod(method string) *Builder {
|
|
builder.method = O.Some(method)
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) SetUrl(url string) *Builder {
|
|
builder.url = url
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) SetHeaders(headers http.Header) *Builder {
|
|
builder.headers = headers
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) SetBody(body O.Option[IOE.IOEither[error, []byte]]) *Builder {
|
|
builder.body = body
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) SetHeader(name, value string) *Builder {
|
|
builder.headers.Set(name, value)
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) DelHeader(name string) *Builder {
|
|
builder.headers.Del(name)
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) GetHeader(name string) O.Option[string] {
|
|
return F.Pipe2(
|
|
name,
|
|
builder.headers.Get,
|
|
O.FromPredicate(S.IsNonEmpty),
|
|
)
|
|
}
|
|
|
|
func (builder *Builder) GetHeaderValues(name string) []string {
|
|
return builder.headers.Values(name)
|
|
}
|
|
|
|
func (builder *Builder) AddHeaderHeader(name, value string) *Builder {
|
|
builder.headers.Add(name, value)
|
|
return builder
|
|
}
|
|
|
|
func (builder *Builder) Requester() IOEH.Requester {
|
|
return F.Pipe3(
|
|
builder.GetBody(),
|
|
O.Map(IOE.Map[error](bytes.NewReader)),
|
|
O.GetOrElse(F.Constant(IOE.Of[error, *bytes.Reader](nil))),
|
|
IOE.Chain(func(rdr *bytes.Reader) IOE.IOEither[error, *http.Request] {
|
|
return IOE.TryCatchError(func() (*http.Request, error) {
|
|
req, err := http.NewRequest(builder.GetMethod(), builder.GetUrl(), rdr)
|
|
if err == nil {
|
|
for name, value := range builder.GetHeaders() {
|
|
req.Header[name] = value
|
|
}
|
|
if rdr != nil {
|
|
req.Header.Set(H.ContentLength, strconv.FormatInt(rdr.Size(), 10))
|
|
}
|
|
}
|
|
return req, err
|
|
})
|
|
}),
|
|
)
|
|
}
|
|
|
|
// Header returns a [L.Lens] for a single header
|
|
func Header(name string) L.Lens[*Builder, O.Option[string]] {
|
|
get := getHeader(name)
|
|
set := F.Bind1of2(setHeader(name))
|
|
del := F.Flow2(
|
|
LZ.Of[*Builder],
|
|
LZ.Map(delHeader(name)),
|
|
)
|
|
|
|
return L.MakeLens(get, func(b *Builder, value O.Option[string]) *Builder {
|
|
cpy := b.clone()
|
|
return F.Pipe1(
|
|
value,
|
|
O.Fold(del(cpy), set(cpy)),
|
|
)
|
|
})
|
|
}
|
|
|
|
// WithHeader creates a [BuilderBuilder] for a certain header
|
|
func WithHeader(name string) func(value string) Endomorphism {
|
|
return F.Flow2(
|
|
O.Of[string],
|
|
Header(name).Set,
|
|
)
|
|
}
|
|
|
|
// WithoutHeader creates a [BuilderBuilder] to remove a certain header
|
|
func WithoutHeader(name string) Endomorphism {
|
|
return Header(name).Set(noHeader)
|
|
}
|
|
|
|
// WithFormData creates a [BuilderBuilder] to send form data payload
|
|
func WithFormData(value url.Values) Endomorphism {
|
|
return F.Flow2(
|
|
F.Pipe4(
|
|
value,
|
|
url.Values.Encode,
|
|
S.ToBytes,
|
|
IOE.Of[error, []byte],
|
|
WithBody,
|
|
),
|
|
WithContentType(C.FormEncoded),
|
|
)
|
|
}
|
|
|
|
// WithJson creates a [BuilderBuilder] to send JSON payload
|
|
func WithJson[T any](data T) Endomorphism {
|
|
return F.Flow2(
|
|
F.Pipe3(
|
|
data,
|
|
J.Marshal[T],
|
|
IOE.FromEither[error, []byte],
|
|
WithBody,
|
|
),
|
|
WithContentType(C.Json),
|
|
)
|
|
}
|