Migration Guide
Upgrading to 0.8.0
Version 0.8.0 introduces changes to snapshot serialization, impacting the management of Redisson snapshots.
Postgres Version
The FactCast server now requires at least Postgres 15. If you are using Postgres 14 or below, you have to upgrade your database before migrating to FactCast 0.8.0.
Changes to Projections
Transactional
Please note that, even though the max bulk size of a transaction in Factus can be configured per projection (by
implementing maxBatchSizePerTransaction or by annotation), the maximum size of a transaction bulk is ultimately limited
by the value of
factcast.grpc.client.maxInboundMessageSize
, which defaults to 3.5MB and can be configured up to 32MB max.
@RedisTransactional
For transactional Redis projections Factus now provides abstract classes. Projections that are annotated with
@RedisTransactional
now must extend AbstractRedisTxManagedProjection
or AbstractRedisTxSubscribedProjection
respectively. The annotation itself is optional, if no default settings are overwritten.
Please note that the default bulk size has changed from 50 to 1000.
@RedisBatched
The @RedisBatched
annotation has been removed. Refactor your affected projections to extend the
respective RedisTx
class.
- Further configuration of the batching behaviour is possible via the optional
@RedisTransactional
annotation. - Handlers with a
RBatch
parameter now receive anRTransaction
object instead. If you’ve previously usedbatch.addAsync()
you can now apply the changes directly on the transaction using e.g.transactionalSet.add()
.
Be aware that Redisson transactions do not support all RObjects
(e.g. Multimaps or AtomicLong). In these cases you
have to decide if you can continue without transactionality, or adapt your projection’s implementation.
@SpringTransactional
For transactional Spring projections Factus now provides abstract classes. Projections that are annotated with
@SpringTransactional
now must extend AbstractSpringTxManagedProjection
or AbstractSpringTxSubscribedProjection
respectively. The annotation itself is optional, if no default settings are overwritten.
Please note that the default bulk size has changed from 50 to 1000.
Removal of deprecated field in ProjectionMetaData
The deprecated field serial
field was finally removed from ProjectionMetaData
. Please use revision
instead like in
the following example:
@ProjectionMetaData(revision = 1)
public class MyProjection extends AbstractManagedProjection { ...
}
Removal of deprecated dependencies for Compression
org.xerial.snappy
and net.jpountz.lz4
have been removed with version 0.8.0.
Please add the new factcast-grpc-snappy
and factcast-grpc-lz4
(preferred) dependencies in order to enable
compression via snappy or lz4 respectively.
Further documentation can be found here.
Redisson Snapshots
If you use Factus for aggregates or snapshot projections you now have to configure where to store the SnapshotCache. Further documentation can be found here. Some properties were renamed and will be automatically migrated by Spring for a while.
Ssnapshots are now stored under different keys in Redis using a simpler ByteArrayCodec codec.
The migration to new snapshots happens automatically. Old snapshots will remain in Redis for the duration specified by
the factcast.redis.deleteSnapshotStaleForDays
property (default: 90 days), even though they are no longer read. This
has two implications:
- You can roll back to the previous version of FactCast without losing any snapshots for a certain number of days.
- During this period, Redis will store both old and new snapshots, consuming more memory. You can manually delete the old snapshots if you are certain they are no longer needed.
Upgrading to 0.4.1
This only applies if you used 0.4.0 on you data before.
If you rely on the header meta-attributes ‘_ser’ or ‘_ts’, there was a bug in 0.4.0 that has the consequence, of some events missing the mentioned attributes.
In order to add them where they are missing, you would want to execute the following SQL on your factstore database:
update fact
set header =
jsonb_set(header, '{meta}',
COALESCE(header -> 'meta', '{}')
|| concat('{"_ser":', ser,
', "_ts":', EXTRACT(EPOCH FROM now()::timestamptz(3)) * 1000, '}')::jsonb
, true)
WHERE (header ->> 'meta')::jsonb ->> '_ser' is null
Obviously, the timestamp will not be correct here, but at least there is one. If you have a better indication of the publishing time in the payload of your events, you may want to use that one instead.
Upgrading to 0.4.0
Please make sure you followed the migration guide if your current version is <0.3.10.
Building
Java11 runtime is required for building and running the server
lombok val is disallowed by configuration
Please use Java11’s final var instead.
Server
Java11 runtime is required for building and running the server
Default Postgres version shifted from 9.6 to 11.5
While there is no indication that FactCast won’t run on 9.x or 10.x we test against 11 now, (as 14 already is around the corner)
Property namespace for the store has been changed from
org.factcast.store.pg
toorg.factcast.store
.While 0.4.0 still supports the older namespace, it is deprecated and should be migrated asap. Note that new properties are only added to the new namespace, so please adjust your projects accordingly.
Note that the default catchup strategy was changed from PAGED or TMPPAGED to FETCHING. Make sure your postgres does not timeout connections.
Client
The Subscription default for ‘maxBatchDelayInMs’ changed from 0msec to 10msec
We feel like this is a good compromise between reducing the server load and minimizing latency. If you absolutely need minimum latency, you can still set it to 0 on a per-subscription basis.
The default for
factcast.grpc.client.catchup-batchsize
is set to 50.If, for some weird reason, you don’t want the transfer to be batched, you can set this down to 1 again.
The default for
factcast.grpc.client.id
is${spring.application.name}
The id is used for logging purposes on the server.
Upgrading to 0.3.10
0.3.10 changes the namespaces of the metrics. Also some metric names have been changed for consistency reasons. If you created dashboards for example based on those names, please be prepared to update them accordingly. You can find the current metric names here.
Upgrading to 0.3.0
There is a new module for use in projects that provide ‘EventObjects’ but do not want to depend on factcast-core, which is called “factcast-factus-event”. @Specification and interface EventObject have been moved there - please update your imports accordingly if necessary.
Another new module is factcast-test which is supposed to help with integration tests by adding a capability to dynamically alter namespaces in integration tests. See testing section
factcast.lockGlobally()
has been removed. By adding the new locking capabilities, that define the scope of the
lock not
just by aggregate-ids, but by FactSpecs, ’lockGlobally()’ needed to be deleted, as there is no proper way to provide a
compatible
migration path.
As most of the internals of FactCast rely on namespaces (for instance for authorization), you really need to enumerate
all possible namespaces that are in scope for the lock. Please use factcast.lock(List<FactSpec>)
instead. Sorry for
the inconvenience.
Upgrading to 0.2.1
Reconnection behavior
Reconnecting due to connection/unknown errors has changed from an infinite loop to just try five times (in a window of three seconds) and escalate if they failed. Details can be found here.
Upgrading to 0.2.0 (quite a lot)
header field ‘aggId’ replaced by ‘aggIds’
This change was actually two years ago, but it was not documented well, and if you used certain factory methods of DefaultFact, a field ‘aggId’ was created/read. So, to make it perfectly clear, a Fact can have an ‘aggIds’ header field, which is an array of uuids.
A FactSpec, however has an optional ‘aggId’ (without the ’s’), as it is not an array, but a single one. We encourage you to update any header in your postgres, that contains ‘aggId’, rather than ‘aggIds’ in the course of migration to 0.2.0
basic-auth setup has changed
If you used a ‘factcast-security.json’ before, you will be please to learn that FactCast was extended to support role/namespace based authorisation. Also the filename changed to ‘factcast-access.json’.
see basicauth usage
basic-auth setup is enforced
By default, when executing without security enabled, you need to supply a property ‘factcast.security.enabled=false’ via commandline or propertyfile to get away with just a warning. If you don’t, FactCast will exit with errorcode 1.
fetching facts by ID has been extended
Next to fetchById(UUID) (asking for the event exactly the way it was published), there is a new fetchByIdAndVersion( UUID,int), that makes FactCast try to transform the event up/down to the requested version. Of course, the usage of a schema-registry is a precondition, as it provides the code to do that transformation.
Upgrading to 0.1.0
unique_identifier
If you used the uniqueidentifier feature before, it was removed. It was only a rouge hack that was used to coordinate two instance in case of publishing. By now, coordination can be done via optimistic locking, so that the need for _unique_identifier is no longer there.
optimistic locking
There is a section on optimistic locking as a new api feature.
Postgres module uuid-ossp
The Postgres module uuid-ossp is necessary for the new optimistic locking api feature. In order to install this extension, the user performing the Liquibase operations requires Postgres superuser permissions.
GRPC Protocol Version
The GRPC Protocol Version shifted from 1.0.0 to 1.1.0. That means, in order to talk to a FactCast server with version 0.1.0, you can use and client from 0.0.30 on, but in order to use a 0.1.0 client, you’d need to talk to a FactCast server with at least the same protocol version than your client. So the idea is: first update your servers, then update the clients.
GRPC Addresses, Hosts, Ports
We updated to yidongnan/grpc-spring-boot-starter. In order to direct your client to a particular target address of a FactCast server, you might have specified:
grpc.client.factstore.port=9443
grpc.client.factstore.host=localhost
this was replaced by
grpc.client.factstore.address=static://localhost:9443
or
grpc.client.factstore.address=dns:///some.host:9443
see https://github.com/yidongnan/grpc-spring-boot-starter for details
Upgrading to 0.0.30
Spring Boot 2
If you use Spring boot, please note, that all projects now depend on Spring Boot 2 artifacts. Support for Spring Boot 1.x was removed.
Plaintext vs TLS
There was a dependency upgrade of grpc-spring-boot-starter in order to support TLS. Note that the default client configuration is now switched to TLS. That means, if you want to continue communicating in an unencrypted fashion, you need to set an application property of ’ grpc.client.factstore.negotiation-type=PLAINTEXT’.
Testcontainers / Building and Testing
In order to run integration tests, that need a Postgres to run, FactCast now uses Testcontainers in order to download and run an ephemeral Postgres. For this to work, the machine that runs test must have docker installed and the current user needs to be able to run and stop docker containers.
You can still override this behavior by supplying an Environment-Variable ‘pg_url’ to use a particular postgres instead. This might be important for build agents that themselves run within docker and do not provide Docker-in-Docker.
Upgrading to 0.0.14
- Incompatible change in GRPC API
The GRPC API has changed to enable non-breaking changes later. (Version endpoint added) The result is, that you have to use > 0.0.14 on Client and Server consistently.
Noteworthy 0.0.12
Note that the jersey impl of the REST interface has its own place on github now. and got new coordinates: * *org.factcast:factcast-server-rest-jersey:0.0.12.** If you use the REST Server, you’ll need to change your dependencies accordingly
There is a BOM within FactCast at org.factcast:factcast-bom:0.0.12 you can use to conveniently pin versions - remember that factcast-server-rest-jersey might not be available for every milestone and is not part of the BOM