mirror of
https://github.com/IBM/fp-go.git
synced 2025-08-10 22:31:32 +02:00
* doc: add presentation Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * fix: add some more examples Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * doc: update presentation Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * fix: update presentation Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * fix: add presentation and samples Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * fix: benchmarks Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * fix: upload presentation Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> * doc: add presentation Signed-off-by: Carsten Leue <carsten.leue@de.ibm.com> * doc: add link to video Signed-off-by: Carsten Leue <carsten.leue@de.ibm.com> --------- Signed-off-by: Dr. Carsten Leue <carsten.leue@de.ibm.com> Signed-off-by: Carsten Leue <carsten.leue@de.ibm.com>
124 lines
3.2 KiB
Go
124 lines
3.2 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 benchmarks
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"io"
|
|
"testing"
|
|
|
|
HTTP "net/http"
|
|
|
|
A "github.com/IBM/fp-go/array"
|
|
R "github.com/IBM/fp-go/context/readerioeither"
|
|
H "github.com/IBM/fp-go/context/readerioeither/http"
|
|
F "github.com/IBM/fp-go/function"
|
|
T "github.com/IBM/fp-go/tuple"
|
|
)
|
|
|
|
type PostItem struct {
|
|
UserId uint `json:"userId"`
|
|
Id uint `json:"id"`
|
|
Title string `json:"title"`
|
|
Body string `json:"body"`
|
|
}
|
|
|
|
type CatFact struct {
|
|
Fact string `json:"fact"`
|
|
}
|
|
|
|
func heterogeneousHttpRequests(count int) R.ReaderIOEither[[]T.Tuple2[PostItem, CatFact]] {
|
|
// prepare the http client
|
|
client := H.MakeClient(HTTP.DefaultClient)
|
|
// readSinglePost sends a GET request and parses the response as [PostItem]
|
|
readSinglePost := H.ReadJson[PostItem](client)
|
|
// readSingleCatFact sends a GET request and parses the response as [CatFact]
|
|
readSingleCatFact := H.ReadJson[CatFact](client)
|
|
|
|
single := F.Pipe2(
|
|
T.MakeTuple2("https://jsonplaceholder.typicode.com/posts/1", "https://catfact.ninja/fact"),
|
|
T.Map2(H.MakeGetRequest, H.MakeGetRequest),
|
|
R.TraverseTuple2(
|
|
readSinglePost,
|
|
readSingleCatFact,
|
|
),
|
|
)
|
|
|
|
return F.Pipe1(
|
|
A.Replicate(count, single),
|
|
R.SequenceArray[T.Tuple2[PostItem, CatFact]],
|
|
)
|
|
}
|
|
|
|
func heterogeneousHttpRequestsIdiomatic(count int) ([]T.Tuple2[PostItem, CatFact], error) {
|
|
// prepare the http client
|
|
var result []T.Tuple2[PostItem, CatFact]
|
|
|
|
for i := 0; i < count; i++ {
|
|
resp, err := HTTP.Get("https://jsonplaceholder.typicode.com/posts/1")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
body, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var item PostItem
|
|
err = json.Unmarshal(body, &item)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
resp, err = HTTP.Get("https://catfact.ninja/fact")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
body, err = io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var fact CatFact
|
|
err = json.Unmarshal(body, &item)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
result = append(result, T.MakeTuple2(item, fact))
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
// BenchmarkHeterogeneousHttpRequests shows how to execute multiple HTTP requests in parallel when
|
|
// the response structure of these requests is different. We use [R.TraverseTuple2] to account for the different types
|
|
func BenchmarkHeterogeneousHttpRequests(b *testing.B) {
|
|
|
|
count := 100
|
|
var benchResults any
|
|
|
|
b.Run("functional", func(b *testing.B) {
|
|
for n := 0; n < b.N; n++ {
|
|
benchResults = heterogeneousHttpRequests(count)(context.Background())()
|
|
}
|
|
})
|
|
|
|
b.Run("idiomatic", func(b *testing.B) {
|
|
for n := 0; n < b.N; n++ {
|
|
benchResults, _ = heterogeneousHttpRequestsIdiomatic(count)
|
|
}
|
|
})
|
|
|
|
globalResult = benchResults
|
|
}
|