Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adding Seq.traverse & sequence functions #277

Merged
merged 10 commits into from
Sep 23, 2024
Merged
18 changes: 18 additions & 0 deletions .vscode/launch.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
{
"version": "0.2.0",
"configurations": [
{
"name": "benchmarks",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is nice ❤️

"type": "coreclr",
"request": "launch",
"program": "${workspaceFolder}/benchmarks/bin/Release/net7.0/benchmarks.exe",
"args": [],
"env": {
"ASPNETCORE_ENVIRONMENT": "Development"
},
"console": "integratedTerminal",
"preLaunchTask": "build release",
"cwd": "${workspaceFolder}/benchmarks/bin/Release/net7.0/"
}
]
}
2 changes: 0 additions & 2 deletions .vscode/settings.json
Original file line number Diff line number Diff line change
@@ -1,13 +1,11 @@
{
"editor.inlayHints.enabled": "off",
"FSharp.enableAdaptiveLspServer": true,
"FSharp.enableMSBuildProjectGraph": true,
"editor.formatOnSave": true,
"FSharp.notifications.trace": false,
"FSharp.notifications.traceNamespaces": [
"BoundModel.TypeCheck",
"BackgroundCompiler."
],
"FSharp.fsac.conserveMemory": true,
"FSharp.fsac.parallelReferenceResolution": false
}
18 changes: 18 additions & 0 deletions .vscode/tasks.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
{
"version": "2.0.0",
"tasks": [
{
"label": "build release",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

❤️

"type": "process",
"command": "dotnet",
"args": [
"build",
"-c",
"Release",
"${workspaceFolder}/FsToolkit.ErrorHandling.sln",
"/property:GenerateFullPaths=true",
"/consoleloggerparameters:NoSummary"
]
}
]
}
13 changes: 8 additions & 5 deletions benchmarks/Program.fs
Original file line number Diff line number Diff line change
Expand Up @@ -3,23 +3,26 @@ open BenchmarkDotNet.Running
open benchmarks
open BenchmarkDotNet.Configs
open BenchmarkDotNet.Jobs
open BenchmarkDotNet.Columns
open BenchmarkDotNet.Environments
open BenchmarkDotNet.Reports
open FsToolkit.ErrorHandling.Benchmarks
open ApplicativeTests

[<EntryPoint>]
let main argv =

let cfg =
DefaultConfig
.Instance
DefaultConfig.Instance
// .AddJob(Job.Default.WithRuntime(CoreRuntime.Core50))
.AddJob(Job.Default.WithRuntime(CoreRuntime.Core60))
.AddJob(Job.Default.WithRuntime(CoreRuntime.Core70))
.AddColumn(StatisticColumn.P80, StatisticColumn.P95)
.WithSummaryStyle(SummaryStyle.Default.WithRatioStyle(RatioStyle.Trend))
// BenchmarkRunner.Run<EitherMapBenchmarks>() |> ignore
// BenchmarkRunner.Run<TaskResult_BindCEBenchmarks>(cfg) |> ignore
// BenchmarkRunner.Run<BindSameBenchmarks>() |> ignore

BenchmarkRunner.Run<Result_BindvsAndCEBenchmarks>(cfg)
BenchmarkRunner.Run<SeqTests.SeqBenchmarks>(cfg, argv)
|> ignore
//

0 // return an integer exit code
213 changes: 213 additions & 0 deletions benchmarks/SeqTests.fs
Original file line number Diff line number Diff line change
@@ -0,0 +1,213 @@
module SeqTests

open BenchmarkDotNet.Attributes
open BenchmarkDotNet.Order
open BenchmarkDotNet.Mathematics
open BenchmarkDotNet.Configs

module sequenceResultMTests =

module v1 =

let sequenceResultM (xs: seq<Result<'t, 'e>>) : Result<'t seq, 'e> =
let rec loop xs ts =
match Seq.tryHead xs with
| Some x ->
x
|> Result.bind (fun t -> loop (Seq.tail xs) (t :: ts))
| None ->
Ok(
List.rev ts
|> List.toSeq
)

// Seq.cache prevents double evaluation in Seq.tail
loop (Seq.cache xs) []

module v2 =

let traverseResultM' state (f: 'okInput -> Result<'okOutput, 'error>) xs =
let folder state x =
match state, f x with
| Error e, _ -> Error e
| Ok oks, Ok ok ->
Seq.singleton ok
|> Seq.append oks
|> Ok
| Ok _, Error e -> Error e

Seq.fold folder state xs
|> Result.map Seq.rev

let traverseResultM (f: 'okInput -> Result<'okOutput, 'error>) xs =
traverseResultM' (Ok Seq.empty) f xs

let sequenceResultM xs = traverseResultM id xs

module v3 =

let inline traverseResultM'
state
([<InlineIfLambda>] f: 'okInput -> Result<'okOutput, 'error>)
xs
=
let folder state x =
match state, f x with
| Error e, _ -> Error e
| Ok oks, Ok ok ->
Seq.singleton ok
|> Seq.append oks
|> Ok
| Ok _, Error e -> Error e

Seq.fold folder state xs
|> Result.map Seq.rev

let traverseResultM (f: 'okInput -> Result<'okOutput, 'error>) xs =
traverseResultM' (Ok Seq.empty) f xs

let sequenceResultM xs = traverseResultM id xs

module v4 =

let traverseResultM' initialState (f: 'okInput -> Result<'okOutput, 'error>) xs =
(initialState, 0)
|> Seq.unfold (fun (state, i) ->
xs
|> Seq.tryItem i
|> Option.bind (fun x ->
match state, f x with
| Error _, _ -> None
| Ok oks, Ok ok ->
let newState =
Seq.singleton ok
|> Seq.append oks
|> Ok

Some(newState, (newState, i + 1))
| Ok _, Error e -> Some(Error e, (Error e, i + 1))
)
)
|> Seq.last

let traverseResultM f xs = traverseResultM' (Ok Seq.empty) f xs
let sequenceResultM xs = traverseResultM id xs

module v5 =

let traverseResultM' state (f: 'okInput -> Result<'okOutput, 'error>) (xs: seq<'okInput>) =
let mutable state = state

let enumerator = xs.GetEnumerator()

while enumerator.MoveNext() do
match state, f enumerator.Current with
| Error _, _ -> ()
| Ok oks, Ok ok -> state <- Ok(Seq.append oks (Seq.singleton ok))
| Ok _, Error e -> state <- Error e

state

let traverseResultM f xs = traverseResultM' (Ok Seq.empty) f xs
let sequenceResultM xs = traverseResultM id xs

module v6 =

let inline traverseResultM'
state
([<InlineIfLambda>] f: 'okInput -> Result<'okOutput, 'error>)
(xs: seq<'okInput>)
=
let mutable state = state

let enumerator = xs.GetEnumerator()

while enumerator.MoveNext() do
match state, f enumerator.Current with
| Error _, _ -> ()
| Ok oks, Ok ok -> state <- Ok(Seq.append oks (Seq.singleton ok))
| Ok _, Error e -> state <- Error e

state

let traverseResultM f xs = traverseResultM' (Ok Seq.empty) f xs
let sequenceResultM xs = traverseResultM id xs


[<MemoryDiagnoser>]
[<Orderer(SummaryOrderPolicy.FastestToSlowest)>]
[<RankColumn(NumeralSystem.Stars)>]
[<MinColumn; MaxColumn; MedianColumn; MeanColumn; CategoriesColumn>]
[<GroupBenchmarksBy(BenchmarkLogicalGroupRule.ByCategory)>]
type SeqBenchmarks() =

member _.GetPartialOkSeq size =
seq {
for i in 1u .. size do
if i = size / 2u then Error "error" else Ok i
}

member _.SmallSize = 1000u

member _.LargeSize = 500_000u

[<Benchmark(Baseline = true, Description = "original")>]
[<BenchmarkCategory("Small")>]
member this.original() =
sequenceResultMTests.v1.sequenceResultM (this.GetPartialOkSeq this.SmallSize)
|> ignore

[<Benchmark(Description = "Seq.fold")>]
[<BenchmarkCategory("Small")>]
member this.seqFoldSmall() =
sequenceResultMTests.v2.sequenceResultM (this.GetPartialOkSeq this.SmallSize)
|> ignore

[<Benchmark(Description = "inlined Seq.fold")>]
[<BenchmarkCategory("Small")>]
member this.inlineSeqFoldSmall() =
sequenceResultMTests.v3.sequenceResultM (this.GetPartialOkSeq this.SmallSize)
|> ignore

[<Benchmark(Description = "Seq.unfold")>]
[<BenchmarkCategory("Small")>]
member this.seqUnfoldSmall() =
sequenceResultMTests.v4.sequenceResultM (this.GetPartialOkSeq this.SmallSize)
|> ignore

[<Benchmark(Description = "GetEnumerator w/ mutability")>]
[<BenchmarkCategory("Small")>]
member this.getEnumeratorSmall() =
sequenceResultMTests.v5.sequenceResultM (this.GetPartialOkSeq this.SmallSize)
|> ignore

[<Benchmark(Description = "inlined GetEnumerator w/ mutability")>]
[<BenchmarkCategory("Small")>]
member this.inlineGetEnumeratorSmall() =
sequenceResultMTests.v6.sequenceResultM (this.GetPartialOkSeq this.SmallSize)
|> ignore

// made this baseline for this category since unfold and original were so unperformant for this size of data
[<Benchmark(Baseline = true, Description = "Seq.fold")>]
[<BenchmarkCategory("Large")>]
member this.seqFoldLarge() =
sequenceResultMTests.v2.sequenceResultM (this.GetPartialOkSeq this.LargeSize)
|> ignore

[<Benchmark(Description = "inlined Seq.fold")>]
[<BenchmarkCategory("Large")>]
member this.inlineSeqFoldLarge() =
sequenceResultMTests.v3.sequenceResultM (this.GetPartialOkSeq this.LargeSize)
|> ignore

[<Benchmark(Description = "GetEnumerator w/ mutability")>]
[<BenchmarkCategory("Large")>]
member this.getEnumeratorLarge() =
sequenceResultMTests.v5.sequenceResultM (this.GetPartialOkSeq this.LargeSize)
|> ignore

[<Benchmark(Description = "inlined GetEnumerator w/ mutability")>]
[<BenchmarkCategory("Large")>]
member this.inlineGetEnumeratorLarge() =
sequenceResultMTests.v6.sequenceResultM (this.GetPartialOkSeq this.LargeSize)
|> ignore
1 change: 1 addition & 0 deletions benchmarks/benchmarks.fsproj
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
<Tailcalls>true</Tailcalls>
</PropertyGroup>
<ItemGroup>
<Compile Include="SeqTests.fs" />
<Compile Include="ApplicativeTests.fs" />
<Compile Include="Benchmarks.fs" />
<Compile Include="AsyncResultCE.fs" />
Expand Down
15 changes: 15 additions & 0 deletions gitbook/SUMMARY.md
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,11 @@
* [sequenceResultM](list/sequenceResultM.md)
* [traverseResultA](list/traverseResultA.md)
* [sequenceResultA](list/sequenceResultA.md)
* Sequences
* [traverseResultM](seq/traverseResultM.md)
* [sequenceResultM](seq/sequenceResultM.md)
* [traverseResultA](seq/traverseResultA.md)
* [sequenceResultA](seq/sequenceResultA.md)
* Transforms
* [ofChoice](result/ofChoice.md)

Expand All @@ -45,6 +50,11 @@
* [sequenceOptionM](option/sequenceOptionM.md)
* [traverseVOptionM](option/traverseVOptionM.md)
* [sequenceVOptionM](option/sequenceVOptionM.md)
* Sequences
* [traverseOptionM](seq/traverseOptionM.md)
* [sequenceOptionM](seq/sequenceOptionM.md)
* [traverseVOptionM](seq/traverseVOptionM.md)
* [sequenceVOptionM](seq/sequenceVOptionM.md)
* Transforms
* [ofNull](option/ofNull.md)
* [ofPair](option/ofPair.md)
Expand Down Expand Up @@ -89,6 +99,11 @@
* [sequenceAsyncResultM](list/sequenceAsyncResultM.md)
* [traverseAsyncResultA](list/traverseAsyncResultA.md)
* [sequenceAsyncResultA](list/sequenceAsyncResultA.md)
* Sequences
* [traverseAsyncResultM](seq/traverseAsyncResultM.md)
* [sequenceAsyncResultM](seq/sequenceAsyncResultM.md)
* [traverseAsyncResultA](seq/traverseAsyncResultA.md)
* [sequenceAsyncResultA](seq/sequenceAsyncResultA.md)
* Transforms
* [ofAsync](asyncResult/ofAsync.md)
* [ofResult](asyncResult/ofResult.md)
Expand Down
20 changes: 20 additions & 0 deletions gitbook/seq/sequenceAsyncResultA.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
## Seq.sequenceAsyncResultA

Namespace: `FsToolkit.ErrorHandling`

Function Signature:

```
Async<Result<'a, 'b>> seq -> Async<Result<'a seq, 'b seq>>
```

Note that `sequence` is the same as `traverse id`. See also [Seq.traverseAsyncResultA](traverseAsyncResultA.md).

This is applicative, collecting all errors.

This is the same as [sequenceResultA](sequenceResultA.md) except that it uses `Async<Result<_,_>>` instead of `Result<_,_>`.

See also Scott Wlaschin's [Understanding traverse and sequence](https://fsharpforfunandprofit.com/posts/elevated-world-4/).

## Examples

19 changes: 19 additions & 0 deletions gitbook/seq/sequenceAsyncResultM.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
## Seq.sequenceAsyncResultM

Namespace: `FsToolkit.ErrorHandling`

Function Signature:

```
Async<Result<'a, 'b>> seq -> Async<Result<'a seq, 'b>>
```

Note that `sequence` is the same as `traverse id`. See also [Seq.traverseAsyncResultM](traverseAsyncResultM.md).

This is monadic, stopping on the first error.

This is the same as [sequenceResultM](sequenceResultM.md) except that it uses `Async<Result<_,_>>` instead of `Result<_,_>`.

See also Scott Wlaschin's [Understanding traverse and sequence](https://fsharpforfunandprofit.com/posts/elevated-world-4/).

## Examples
Loading
Loading