Sharpino.Core
2.0.4
See the version list below for details.
dotnet add package Sharpino.Core --version 2.0.4
NuGet\Install-Package Sharpino.Core -Version 2.0.4
<PackageReference Include="Sharpino.Core" Version="2.0.4" />
paket add Sharpino.Core --version 2.0.4
#r "nuget: Sharpino.Core, 2.0.4"
// Install Sharpino.Core as a Cake Addin #addin nuget:?package=Sharpino.Core&version=2.0.4 // Install Sharpino.Core as a Cake Tool #tool nuget:?package=Sharpino.Core&version=2.0.4
Sharpino
<img src="ico/sharpino.png" alt="drawing" width="50"/>
A little F# event-sourcing library
What is it?
Support for Event-sourcing in F#.
Features
- Supports in memory and Postgres event store. Supports EventStoreDB (only for the LightCommandHandler).
- Support publishing events to Apache Kafka Event Broker.
- Example application with tests including Kafka subscriber. (unstable at the moment)
- Contexts represent sets of collections of entities (e.g. a collection of todos, a collection of tags, a collection of categories, etc.) associated with events.
- Aggregates are the same as contexts with many instances identified by Id (Guid).
- A specific technique helps refactoring (migration) between different versions of the application.
Absent features (for now)
- Sensible data (GDPR)
- Stable and working integration with an Event Broker (Apache Kafka)
Projects
Sharpino.Lib.Core:
- Core.fs: Abstract definition of Events, Commands and Undoer (the reverse of a command to be used if the used event store lacks transaction between streams). Definition of EvolveUnForgivingErrors and "normal" Evolve. The former raises an error in processing invalid events. The latter just skip those events. Some other types and functions are defined here. In a "SAFE stack" like project we may "safely" include the Core in a shared project (Fable Remoting style) in the sense that Fable can transpile it to JavaScript.
Sharpino.Lib:
- CommandHandler.fs: Gets and stores snapshots, executes commands, and produces and stores events using the event store.
- LightCommandHandler.fs: gets and stores snapshots, executes commands, produces and stores events using an event store that supports publish/subscribe ( EventStoreDB).
- PgEventStore.fs and MemoryStorage.fs: Manages persistency in Postgres or in-memory respectively.
- Cache.fs. Caches the current state of contexts or aggregates.
Sharpino.Sample: You need a user called 'safe' with password 'safe' in your Postgres (if you want to use Postgres as Eventstore).
It is an example of a library for managing todos with tags and categories. There are two versions in the sense of two different configurations concerning the distribution of the models (collection of entities) between the contexts. There is a strategy to test the migration between versions (contexts refactoring) that is described in the code (See: AppVersions.fs and MultiVersionsTests.fs ) .
contexts (e.g. TodosContext) controls a partition of the collection of the entities and provides members to handle them.
contexts members have corresponding events (e.g. TagsEvents) that are Discriminated Unions cases. Event types implement the Process interface.
contexts are related to Commands (e.g. TagCommand) that are Discriminated Unions cases that can return lists of events by implementing the Executable interface.
Commands defines also undoers are functions that can undo the commands to reverse action in a multiple-stream operation for storage that doesn't support multiple-stream transactions (see LightCommandHandler).
A Storage stores and retrieves events and snapshots.
The api layer functions provide business logic involving one or more clusters by accessing their state, and by building one or more commands and sending them to the CommandHandler.
An example of how to handle multiple versions of the application to help refactoring and migration between different versions: application versions.
aggregates works as context and I can handle multiple instances of an aggregate identified by Id (Guid). The example is in the sample application 4 (see below).
Sharpino.Sample.tests
- tests for the sample application Sharpino.Lib.fsproj Sharpino.Sample.Kafka
- scripts to set up Kafka topics corresponding to the contexts of the sample application.
How to use it
- You can run the sample application as a rest service by running the following command from Sharpino.Sample folder:
dotnet run
- You can run the client Fable/Elmish sample application by running the following command from the Sharpino.Sample.Client folder:
npm install
npm start
- Just use ordinary dotnet command line tools for building the solution. Particularly you can run tests of the sample application by using the following command:
dotnet test
You can also run the tests by the following command from Sharpino.Sample.Tests folder:
dotnet run
In the latter case, you get the output from Expecto test runner (in this case the console shows eventual standard output/printf).
By default, the tests run only the in-memory implementation of the storage. You can set up the Postgres tables and db by using dbmate. In the Sharpino.Sample folder you can run the following command to set up the Postgres database:
dbmate -e up
(see the .env to set up the DATABASE_URL environment variable to connect to the Postgres database with a connection string). If you have Eventstore the standard configuration should work. (I have tested it with Eventstore 20.10.2-alpha on M2 Apple Silicon chip under Docker).
SAMPLE 4
Sample 4 uses SAFE stack (Fable/Elmish). The aggregates are rows of seats that I can book. A skeleton of Elmish client is provided.
SAMPLE 5
Sample 5 will just replace SAMPLE 4. uses SAFE stack (Fable/Elmish). Use the common SAFE stack basic operation for running and testing (under the directory Sharpino.Sample.5) Please just look at the domain, and don't care that much about the u.i. (Elmish part).
Tests on eventstoredb. EventStroreDb is not mantained that much at the moment
The following line needs to stay commented out.
// (AppVersions.evSApp, AppVersions.evSApp, fun () -> () |> Result.Ok)
Sample application 2 is a problem of booking seats in two rows of five seats:
- Booking seats among multiple rows (where those rows are aggregates) in an event-sourcing way.
- Booking seats in a single row by two concurrent commands that singularly do not violate any invariant rule and yet the final state is potentially invalid.
Problem 1
I have two rows of seats related to two different streams of events. Each row has 5 seats. I want to book a seat in row 1 and a seat in row 2. I want to do this in a single transaction so that if just one of the claimed seats is already booked then the entire multiple-row transaction fails and no seats are booked at all.
Questions:
- can you do this in a transactional way? Answer: yes because in-memory and Postgres event-store implementation as single sources of truth are transactional. The runTwoCommands in the Command Handler is transactional.
- Can it handle more rows? up to tre. (runThreeCommands in the Command Handler)
- Is feasible to scale to thousands of seats/hundreds of rows (even though we know that few rows will be actually involved in a single booking operation)? Not yet.
- Is Apache Kafka integration included in this example? No.
- Is EventStoreDb integration included in this example? Not yet (it will show the "undo" feature of commands to do rollback commands on multiple streams of events).
Problem 2
There is an invariant rule that says that no booking can end up in leaving the only middle seat free in a row. This invariant rule must be preserved even if two concurrent transactions try to book the two left seats and the two right seats independently so violating (together) this invariant.
Questions:
- can you just use a lock on any row to solve this problem? Answer: yes by setting PessimisticLocking to true in appSettings.json that will force single-threaded execution of any command involving the same "aggregate"
- can you solve this problem without using locks? Answer: yes. if I set PessimisticLocking to false then parallel command processing is allowed and invalid events can be stored in the eventstore. However they will be skipped by the "evolve" function anyway.
- Where are more info about how to test this behavior? Answer: See the testList called hackingEventInStorageTest. It will simply add invalid events and show that the current state is not affected by them.
- You also need to give timely feedback to the user. How can you achieve that if you satisfy invariants by skipping the events? Answer: you can't. The user may need to do some refresh or wait a confirmation (open a link that is not immediately generated). There is no immediate consistency in this case.
Sample application 4
The domain Sample application 4 is the same of the Sample 2 and uses aggregates to be able to create an arbitrary number of seat rows. Invariants can be represented by quoted expressions so that, ideally, this may allow us to move toward a DSL (Example: "no booking can end up in leaving the only middle seat free in a row").
Faq:
- Why the name "Sharpino"?
- It's a mix of "Sharp" (as the '#' of C# or F#) and fino (Italian for "thin"). "sciarpino" (same pronunciation) in Italian means also "little scarf".
- Why another event-sourcing library?
- I wanted to study the subject and it ended up in a tiny little framework.
- Why F#?
- Any functional language of the ML family language in my opinion is a good fit for the following reasons:
- Events are immutable, building the state of the context is a function of those events.
- Discriminated Unions are suitable to represent events and commands.
- The use of the lambda expression is a nice trick for the undoers (the under is returned as a lambda that retrieves the context for applying the undo and returns another lambda that actually can "undo" the command).
- It is a .net language, so you can use everything in the .net ecosystem (including C# libraries).
- Any functional language of the ML family language in my opinion is a good fit for the following reasons:
- How to use it
- add the nuget package Sharpino to your project.
- note: on my side, when I added Sharpino as a library into a web app, then I had to add the following line to the web app project file to avoid a false error (the error was "A function labeled with the 'EntryPointAttribute' attribute must be the last declaration")
<GenerateProgramFile>false</GenerateProgramFile>
Useful info:
Examples 4 and 5 are using the SAFE stack. To run the tests use the common SAFE way (dotnet run
and dotnet run -- RunTests
from their root dir )
help wanted:
- Rewrite from scratch the Kafka integration making it work as is supposed to (i.e. Kafka "viewers" can be passed to application instances as in the examples)
- Adapt the examples to the new version of the library (2.0.0)
News
Version 2.1.3: added local fork of FsKafka (with library dependencies updated) to be able to use it in the project.
Version 2.1.0: going to remove newtonsoft, introduced FsPickler, FsKafka, changed kafka publisher way (binary and textencoding). Removed Kafkareceiver. Preparing to replace it with one based on FSKafka
I am porting the examples to use the newer version (2.0.6). The porting of the first example(Sharpino.Sample) is incomplete (At the moment I disabled the "migrate between version" function in that example).
version 2.0.7: a fix in runThreeCommand. CommandHandler will just use fresh data ignoring the viewer that has been passed.
version 2.0.6:
- eventstore checks the eventId of the state that produces any event before adding them. That will ensure that events are added in the right order and cannot fail (so the optimistic lock stateId will be superfluous and will be dropped). Serialization can be binary or JSON (see appSettings.json). The default is binary. The samples use Fspickler to serialize/deserialize events and snapshots. There is no use of ISerializer interface which was by default newtonsoft. Therefore the user needs to provide a serializer (pickler is ok for binary and for json as well). Pickler will not work with jsonb fields in Postgres as the indexes change their order in jsonb and pickler doesn't want it, so they must be text. Kafka is still not working on the read part. The write part is ok even though any read test has been dropped for incompatibility and will be rewritten.
version 2.0.3: changes from "now()" to utcNow() format in eventstores (Postgres and inMemory) and Sql template scripts.
published version 2.0.0 supporting binary serialization for events and snapshots on Postgres. Note: the current examples provided are still referencing the previous 1.6.6 version. Here is an example compatible with 2.0.0. with binary serialization
added some videos (in Italian) about the library and the sample applications. https://youtu.be/OQKD5uluFPc https://youtu.be/ToZ_I_xRA-g https://youtu.be/WtGEQqznPnQ https://youtu.be/j2XoLkCt31c
added a few new examples (can be used for dojos) pub system
version 1.6.6: can use plain text instead of JSON data type for database (see scripts in SqlTemplate dir). The appSettings has a new settings for it:
{
"LockType":{"Case":"Optimistic"},
"RefreshTimeout": 100,
"CacheAggregateSize": 100,
"PgSqlJsonFormat":{"Case":"PlainText"}
}
The other option is:
"PgSqlJsonFormat":{"Case":"PgJson"}
the tables should be coherent: use PlainText when json fields are of type text and PgJson when they are of type json or jsonb. Why bother? In this example I use FsPickler to serialize/deserialize https://github.com/tonyx/shoppingCartWithSharpino It won't work with jsonb fields because it needs the same order of fields in the json string whereas json/jsonb fields are stored in a way that doesn't preserve the order of fields.
version 1.6.0: starting removing kafka for aggregates (will be replaced somehow). Use eventstore (postgres) based state viewers instead. New sample: started an example of Restaurant/Pub management. (Sample 6)
Version 1.5.8: fix in adding events with stateId when adding more events (only the first stateId matters in adding many events, so the rest are new generated on the fly)
Version 1.5.7:
Added runInitAndCommand that creates a new aggregate and a command context in a single transaction.
Changed the signature of runAggregate and runNAggregate (simplified the viewer passed as parameter avoiding a labmda)
Version 1.5.5:
fixed a key problem in dictionary keys in memory based eventstore (MemoryStorage). Note it is supposed to be used only for dev and testing.
WARNING: Kafka publishing is ok but Kafka client integration needs heavy refactoring and fixing, particularly about aggregate viewer.T That means that any program that tries to build the state using KafkaStateViewer may have some inefficiencies of even errors. Just use the storage base state viewers for now (or build your own state viewers by subscribing to the Kafka topic and building the state locally). I am ready to refactor now because I have an elmish sample app for (manual) testing. An example of hot try it is by taking a look in the src/Server/server.fs in Sample4 (commented code with different ways to instantiate the "bookingsystem" sample app)
Added SqlTempate dir with template examples for creating table relate to events and snapshots for aggregates and contexts.
Addes sample4 witch is almost the same as sample3 but using SAFE stack (Fable/Elmish) as envelope (going to ditch sample3 because it is going to be messy)
Version 1.5.4 Replaces version 1.5.3 (that was deprecated )
- Aggregate snapshots added (in addition to contexts snapshots)
Version 1.5.3 (don't use it: it is deprecated: missing packages)
Version 1.5.1: -
- Added a new configuration file named appSettings.json in the root of the project with the following content:
{
"LockType":{"Case":"Optimistic"},
"RefreshTimeout": 100,
"CacheAggregateSize": 100
}
Moreover, the specific variant of optimitic lock (classic or more permissive) can be changed by code (evenstore method). The "more permissive" optmistic lock will skip aggregate state version control so it allows that events generated in the same moment can be stored and published. Nevertheless if they end up in a violation of the invariant rule the core will skip them anyway. The more permissive optimistic lock cannot ensure that multiple aggregate transactions are handled properly: if I book seats from multiple rows it is theoretically possible that only seats from one row are booked.
See the sql script of sample 4: it includes the steps related to change "on the fly" (by application) the db level constraints that inhibit adding two events with the same aggregate stateid.
(in the same stream) in the event store.
Current version 1.5.0: - Kafka integration for fine-grained aggregates (identified by Id) is included.
Version 1.4.8: streams of events can relate to proper aggregate identified by id (and not only context). I can run commands for an arbitrary number of aggregates of some specific type. See Sample4 (booking seats of rows where rows are aggregates of a context which is a stadium). Integration with Kafka for those "fine" aggregates identified by Id is not included in this version.
Booking seat example: https://github.com/tonyx/seatsLockWithSharpinoExample (it shows some scalability issues that will be fixed in future releases)
Version 1.4.7: contains sample app that builds state of contexts by using Kafka subscriber (receives and processes events to build locally the state of those contexts, despite can still access the "souce of truth", which is the db/event store, when something goes wrong in processing its state, i.e. out of sync events).
Version 1.4.6: fix bug in Postgres AddEvents
Version 1.4.5: Upgrade to net8.0
Info: in adding new features I may risk breaking backward compatibility. At the moment I am handling in this simple way: if I change the signature of a function I add a new function with the new signature and I deprecate the old one. I will keep the old one for a while. I will remove it only if I am sure that nobody is using it.
Version: 1.4.4 Postgres tables of events need a new column: kafkaoffset of type BigInt. It is used to store the offset/position of the event in the Kafka topic. See the new four last alter_ Db script in Sharpino.Sample app. This feature is Not backward compatible: You need your equivalent script to update the tables of your stream of events. (Error handling can be improved in writing/reading Kafka event info there). Those data will be used in the future to feed the "kafkaViewer" on initialization.
From Version 1.4.1 CommandHandler changed: runCommand requires a further parameter: todoViewer of type (stateViewer: unit → Result<EventId * 'A, string>). It can be obtained by the CommandHandler module itself.getStorageStateViewera<'A, 'E> (for database event-store based state viewer.)
Version 1.4.1: little change in Kafka consumer. Can use DeliveryResults to optimize tests
Version 1.4.0: runCommand instead of Result<unit, string> returns, under result, info about event-store created IDs (Postgres based) of new events and eventually Kafka Delivery result (if Kafka is configured).
Version 1.3.9: Repository interface changed (using Result type when it is needed). Note: the new Repository interface (and implementation) is not compatible with the one introduced in Version 1.3.8!
Version 1.3.8: can use a new Repository type instead of lists (even though they are still implemented as plain lists at the moment) to handle collections of entities.
Version 1.3.5: the library is split into two nuget packages: Sharpino.Core and Sharpino.Lib. the Sharpino.Core can be included in a Shared project in the Fable Remoting style. The collections of the entities used in the Sharpino.Sample are not lists anymore but use Repository data type (which at the moment uses plain lists anyway).
Version 1.3.4 there is the possibility to choose a pessimistic lock (or not) in command processing. Needed a configuration file named appSettings.json in the root of the project with the following content: don't use this because the configuration is changed in version 1.5.1
"SharpinoConfig": {
"PessimisticLock": false // or true
}
More documentation (a little bit out of date. Will fix it soon) (Sharpino gitbook)
<a href="https://www.buymeacoffee.com/Now7pmK92m" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 60px !important;width: 217px !important;" ></a>
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net7.0 is compatible. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
-
net7.0
- Expecto (>= 10.2.1)
- Fable.Core (>= 4.3.0)
- Farmer (>= 1.8.9)
- FSharpPlus (>= 1.6.1)
- FsPickler (>= 5.3.2)
- FsPickler.Json (>= 5.3.2)
- FsToolkit.ErrorHandling (>= 4.15.2)
- Log4net (>= 2.0.17)
- Newtonsoft.Json (>= 13.0.3)
-
net8.0
- Expecto (>= 10.2.1)
- Fable.Core (>= 4.3.0)
- Farmer (>= 1.8.9)
- FSharpPlus (>= 1.6.1)
- FsPickler (>= 5.3.2)
- FsPickler.Json (>= 5.3.2)
- FsToolkit.ErrorHandling (>= 4.15.2)
- Log4net (>= 2.0.17)
- Newtonsoft.Json (>= 13.0.3)
NuGet packages (1)
Showing the top 1 NuGet packages that depend on Sharpino.Core:
Package | Downloads |
---|---|
Sharpino
Simple F# Event-Sourcing Library |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated | |
---|---|---|---|
2.1.7 | 105 | 12/7/2024 | |
2.1.6 | 132 | 11/14/2024 | |
2.1.5 | 141 | 10/25/2024 | |
2.1.4 | 150 | 10/20/2024 | |
2.1.3 | 368 | 8/13/2024 | |
2.1.2 | 125 | 8/9/2024 | |
2.1.1 | 126 | 7/19/2024 | |
2.1.0 | 175 | 7/6/2024 | |
2.0.4 | 225 | 5/25/2024 | |
2.0.3 | 155 | 5/19/2024 | |
2.0.2 | 145 | 5/14/2024 | |
2.0.1 | 125 | 5/14/2024 | |
2.0.0 | 189 | 5/2/2024 | |
1.1.4 | 213 | 4/6/2024 | |
1.1.3 | 120 | 4/6/2024 | |
1.1.2 | 152 | 3/17/2024 | |
1.1.1 | 142 | 3/11/2024 | |
1.1.0 | 179 | 3/4/2024 | |
1.0.9 | 307 | 2/7/2024 | |
1.0.8 | 159 | 1/31/2024 | |
1.0.7 | 178 | 1/25/2024 | |
1.0.6 | 140 | 1/9/2024 | |
1.0.5 | 151 | 1/1/2024 | |
1.0.4 | 225 | 12/19/2023 | |
1.0.3 | 114 | 12/19/2023 | |
1.0.2 | 142 | 12/15/2023 | |
1.0.1 | 649 | 12/8/2023 | |
1.0.0 | 495 | 11/28/2023 |