Effect App is a light-weight framework to build resilient and fully type-safe services and apps with Effect and FP-TS, in a statically typed functional style in the latest of Typescript. Compiler extensions through ts-plus are recommended, yet not required. It allows for a compiler/type driven development, where often, if it compiles, your program is probably quite correct :) Reducing the need for boilerplate testing and providing you a level of confidence out of the box.
We aim to provide defaults in the box which are useful to 80% of the apps you want to build.
All data modeling is Schema based. It powers the strong domain modelling, RPC (client+api controllers), database schema, input/form schemas, providing a single source of truth for all models and refined primitives.
Your UI and services will never be out of sync, and even the fully type-safe api client (incl. runtime client side input and response validation) to use in your apps is automatically derived from your resource specifications, also guiding the implementation of your api controllers.
Sketch..
- Applications
- API - private business logic, endpoints with resources for UI
- UI - views and public business logic, uses derived clients to talk to endpoints on the API
- Messages - talk between API/Worker services
- Resources - talk between UI and API - think REST resources, actions, view models.
- Models - domain core types, many others are derived from these
TBD
TBD
in the box are adapters for:
- in memory (
mem://
): great for testing locally, no resilience, only works within a single process (but you can of course compose multiple service entrypoints into a single app). - Azure ServiceBus: (
Endpoint=
) - adding your own is a piece of cake
(All of these are based on established community or vendor libraries)
TBD
TBD
- Published to UI via Server Sent Events
Configuration is specified in API config.ts
files, and leverages the built-in Effect Config.
Logging defaults to the logfmt format. It's recommended to use humanlog for improved formatting and highlighting, especially of embedded JSON payloads.
The default storage adapters and setup is for document databases. There is the Persistence Model (how does the data look like within the data store), Encoded type (often the same as persistence model) and Model type (fully validated and strong domain types).
In the box Store
are adapters for
- memory (
mem://
): good for local development, tests, single instance, no need for migrations - disk (
disk://.data
): good for local development, tests, single instance, migrations optional - redis (
redis://
...): good for dev instances, single instance, migrations optional - CosmosDB (
AccountEndpoint=
): production ready, multi instance support - Mongo (WIP) (
mongo://
...): production ready, multi instance support - adding your own is a piece of cake
(All of these are based on established community or vendor libraries)
While basic transaction support for ACID operations is possible, it is often limited to one collection and perhaps 100 records at a time, therefore it is recommended to design mutations in non corruptive ways, and consider reconciliation options.
e.g:
- remove links before removing the actual record
- design in a way where you can process batches
Data migrations are manual and explicit.
Empty values should be null
, so omitted/undefined
values in the data are usually considered suspicious and an error.
So if you use a persistent database adapter like diskdb, cosmos, redis or mongo, you need to make sure you migrate your data, via:
- just in time migration within the repository, when converting from Persistence Model, to Encoded type, before parsing
- implementing a migration strategy that runs at start up or at specific command request
- manually writin/executing migration queries for the particular database.
locally when the data is considered disposable, you could also increment the STORAGE_VERSION
constant,
which will ignore previous data, and store under a new namespace.
By default, optimistic concurrency is used via an _etag
field. If the _etag
on a record changed between reading the record, and updating the record,
an OptimisticConcurrencyException
is thrown. You will need to handle it as appropriate for your case.
During saving operations you can include events (domain or integration events), each repository decides what events to support and what to do with them. Often an Event needs to be published on a Queue or MessageBus, after the data has been successfuly saved. To make this even more durable, you can save the events within the same database transaction, so that if reaching a queue fails after saving or the process is terminated, you can reconsile this by replaying unprocessed events from the database.
The Pure type helps you to build logic that builds or updates state, and logs events to be picked up by the save and publish. It composes nicely. Which is inspired by ZPure
- Boilerplate
- The actual framework