Canton CANTON_VERSION has been released on RELEASE_DATE. You can download the Daml Open Source edition from the Daml Connect Github Release Section. The Enterprise edition is available on Artifactory. Please also consult the full documentation of this release.
- We introduced contract key prefetching / bulk loading to improve workloads that fetch many contract keys.
- Domain renaming
- domain id -> synchronizer id
- domain alias -> synchronizer alias
- domain projects (e.g., community-domain) -> synchronizer projects
- The GetTransactionByEventId and the GetTransactionTreeByEventId endpoints of the lapi update service have been
replaced by the GetTransactionByOffset and the GetTransactionTreeByOffset respectively.
- As a consequence, the GetTransactionByEventIdRequest has been replaced by the GetTransactionByOffsetRequest message.
- The GetTransactionByOffsetRequest contains the offset of the transaction or the transaction tree to be fetched and the requesting parties.
- The json endpoints have been adapted accordingly
Refactored domain connectivity service to have endpoints with limited responsibilities:
- Add: ReconnectDomain to be able to reconnect to a registered domain
- Add: DisconnectAllDomains to disconnect from all connected domains
- Change: RegisterDomain does not allow to fully connect to a domain anymore (only registration and potentially handshake): if you want to connect to a domain, use the other endpoint
- Change: ConnectDomain takes a domain config so that it can be used to connect to a domain for the first time
- Rename: ListConfiguredDomains to ListRegisteredDomains for consistency (and in general: configure(d) -> register(ed))
A memory check has been introduced when starting the node. This check compares the memory allocated to the container with the -Xmx JVM option. The goal is to ensure that the container has sufficient memory to run the application. To configure the memory check behavior, add one of the following to your configuration:
canton.parameters.startup-memory-check-config.reporting-level = warn // Default behavior: Logs a warning.
canton.parameters.startup-memory-check-config.reporting-level = crash // Terminates the node if the check fails.
canton.parameters.startup-memory-check-config.reporting-level = ignore // Skips the memory check entirely.
- Removed parameters
sequencer.writer.event-write-batch-max-duration
andsequencer.writer.payload-write-batch-max-duration
as these are not used anymore. - Introduced parameter
sequencer.writer.event-write-max-concurrency
(default: 2) to configure the maximum number of events batches that can be written at a time. - [Breaking Change]:
TopologyManagerReadService.ExportTopologySnapshot
andTopologyManagerWriteService.ImportTopologySnapshot
are now streaming services for exporting and importing a topology snapshot respectively.
- Added offset (int64) and node-id (int32) fields in all the event types in the ledger api.
The following messages have the additional fields:
- CreatedEvent
- ArchivedEvent
- ExercisedEvent
- Accordingly the java bindings and json schema were augmented to include the new fields.
-
Deduplication Offset extension to accept participant begin
Before, only absolute offsets were allowed to define the deduplication periods by offset. After the change participant-begin offsets are also supported for defining deduplication periods. The participant-begin deduplication period (defined as zero value in API) is only valid to be used if the participant was not pruned yet. Otherwise, as in the other cases where the deduplication offset is earlier than the last pruned offset, an error informing that deduplication period starts too early will be returned.
-
Index DB schema changed in a non-backwards compatible fashion.
The offset-related fields (e.g. ledger_offset, ledger_end) that were previously stored as
VARCHAR(4000)
for H2 andtext
for Postgres are now stored asBIGINT
(for both db types).- If the offset column can take the value of the participant begin then the column should be null-able and null should be stored as the offset value (i.e. no zero values are used to represent the participant begin).
- Only exception to it is the deduplication_offset of the lapi_command_completions which will take the zero value when the participant begin must be stored as deduplication offset, since null is used to signify the absence of this field.
-
Changed DeduplicationPeriod's offset field type to
int64
in participant_transaction.proto in a non-backwards compatible fashion.The type of the offset field changed from
bytes
toint64
to be compatible with the newly introduced intefer offset type.
- [Breaking Change] renamed configuration parameter
session-key-cache-config
tosession-encryption-key-cache
. sequencer_authentication_service
RPCs return failures as gRPC errors instead of a dedicated failure message with status OK.
- display_name is no longer a part of Party data, so is removed from party allocation and update requests in the ledger api and daml script
PartyNameManagement
service was removed from the ledger api
- When a Grpc channel is open or closed on the Ledger API, a message is logged at a debug level:
[..] DEBUG c.d.c.p.a.GrpcConnectionLogger:participant=participant - Grpc connection open: {io.grpc.Grpc.TRANSPORT_ATTR_LOCAL_ADDR=/127.0.0.1:5001, io.grpc.internal.GrpcAttributes.securityLevel=NONE, io.grpc.Grpc.TRANSPORT_ATTR_REMOTE_ADDR=/127.0.0.1:49944}
[..] DEBUG c.d.c.p.a.GrpcConnectionLogger:participant=participant - Grpc connection closed: {io.grpc.Grpc.TRANSPORT_ATTR_LOCAL_ADDR=/127.0.0.1:5001, io.grpc.internal.GrpcAttributes.securityLevel=NONE, io.grpc.Grpc.TRANSPORT_ATTR_REMOTE_ADDR=/127.0.0.1:49944}
- The keep alive behavior of the Ledger API can be configured through
canton.participants.participant.ledger-api.keep-alive-server.*
- The default values of the keep alive configuration for the ledger api has been set to
time: 10m
timeout: 20s
permitKeepAliveTime: 10s
permitKeepAliveWithoutCalls: false
- The effective settings are reported by the Participant Node at the initialization time with a logline:
2024-10-31 18:09:34,258 [canton-env-ec-35] INFO c.d.c.p.a.LedgerApiService:participant=participant - Listening on localhost:5001 over plain text with LedgerApiKeepAliveServerConfig(10m,20s,10s,true).
-
New parameter value for
permitKeepAliveWithoutCalls
has been introduced to all keep alive configurations. When set, it allows the clients to send keep alive signals outside any ongoing grpc call. -
Identical implementations
EnterpriseCantonStatus
andCommunityCantonStatus
have been merged into a single classCantonStatus
. -
A participant will now crash in exceptional cases during transaction validation instead of remaining in a failed state
- Addition of a
submissionTimeRecordTimeTolerance
dynamic domain parameter, which defaults to the value ofledgerTimRecordTimeTolerance
ledgerTimRecordTimeTolerance
is no longer unsafe to increase, however,submissionTimeRecordTimeTolerance
now is, within the same restrictions asledgerTimRecordTimeTolerance
was before- Use of the flag
LedgerTimeRecordTimeToleranceIncrease
is now deprecated - A new flag
SubmissionTimeRecordTimeToleranceIncrease
has been added to forcefully increase thesubmissionTimeRecordTimeTolerance
instead
- Split the current signing schemes into a key
([Encryption/Signing]KeySpec)
and algorithm([Encryption/Signing]AlgorithmSpec)
specifications. We also changed the way this is configured in Canton, for example,signing.default = ec-dsa-p-256
is now represented as:signing.algorithms.default = ec-dsa-sha-256
andsigning.keys.default = ec-p-256
. This is not a breaking change because the old schemes are still accepted. - [Breaking Change] changed the
name
parameter ofrotate_node_key
fromOption
toString
. - Added a
name: String
parameter torotate_kms_node_key
, allowing operators to specify a name for the new key.
- Console commands use now integer offsets. The affected commands are the following:
- ledger_api.updates.{trees, trees_with_tx_filter, subscribe_trees}
- ledger_api.updates.{flat, flat_with_tx_filter, subscribe_flat}
- ledger_api.state.end
- ledger_api.state.acs.{of_party, active_contracts_of_party, incomplete_unassigned_of_party, incomplete_assigned_of_party, of_all}
- ledger_api.completions.{list, subscribe}
- ledger_api.javaapi.updates.{trees, flat, flat_with_tx_filter}
- pruning.{prune, find_safe_offset, get_offset_by_time, prune_internally}
- testing.state_inspection.lookupPublicationTime
- In the canton's pruning and inspection services we used strings to represent the offset of a participant.
The integer approach replaces string representation in:
- pruning service:
- PruneRequest message: with int64
- GetSafePruningOffsetRequest message: with int64
- GetSafePruningOffsetResponse message: with int64
- inspection service:
- LookupOffsetByTime.Response: with optional int64.
- If specified, it must be a valid absolute offset (positive integer).
- If not set, no offset corresponding to the timestamp given exists.
- LookupOffsetByTime.Response: with optional int64.
- pruning service:
- Index DB schema changed in a non-backwards compatible fashion.
- gRPC requests that are aborted due to shutdown server-side return
CANCELLED
instead ofFAILED_PRECONDITION
. - Added auto vacuuming defaults for sequencer tables for Postgres (will be set using database schema migrations).
- Removed support for Postgres 11, 12
- Made Postgres 14 default in the CI
- Don't fetch payloads for events with
eventCounter < subscriptionStartCounter
. - Payloads are fetched behind a Caffeine cache.
canton.sequencers.<sequencer>.parameters.caching {
sequencer-payload-cache {
expire-after-access="1 minute" // default value
maximum-size="1000" // default value
}
}
- Payload fetching can be configured with the following config settings:
canton.sequencers.<sequencer>.sequencer.block.reader {
// max number of payloads to fetch from the datastore in one page
payload-batch-size = 10 // default value
// max time window to wait for more payloads before fetching the current batch from the datastore
payload-batch-window = "5ms" // default value
// how many batches of payloads will be fetched in parallel
payload-fetch-parallelism = 2 // default value
// how many events will be generated from the fetched payloads in parallel
event-generation-parallelism = 4 // default value
}
- Added sequencer in-memory fan out. Sequencer now holds last configurable number of events it has processed in memory. In practice this is 1-5 seconds worth of data with the default max buffer size of 2000 events. If the read request for a member subscription is within the fan out range, the sequencer will serve the event directly from memory, not performing any database queries. This feature is enabled by default and can be configured with the following settings:
canton.sequencers.<sequencer>.sequencer.writer {
type = high-throughput // NB: this is required for the writer config to be parsed properly
max-buffered-events-size = 2000 // Default value
}
This feature greatly improves scalability of sequencer in the number of concurrent subscription, under an assumption that members are reading events in a timely manner. If the fan out range is exceeded, the sequencer will fall back to reading from the database. Longer fan out range can be configured, trading off memory usage for database load reduction.
- CommandService.SubmitAndWaitForUpdateId becomes CommandService.SubmitAndWait in terms of semantics and request/response payloads. The legacy SubmitAndWait form that returns an Empty response is removed from the CommandService
- Improved logging in case of sequencer connectivity problems as requested by Canton Network.
- The block sequencer is now configurable under
canton.sequencers.<sequencer>.block
, including new checkpoint settings:
// how often checkpoints should be written
block.writer.checkpoint-interval = "30s"
// how many checkpoints should be written when backfilling checkpoints at startup
block.writer.checkpoint-backfill-parallelism = 2
IdentityInitializationService.CurrentTimeResponse
returns the current time in microseconds since epoch instead of a Google protobuf timestamp.- Commands.DisclosedContract is enriched with
domain_id
which specifies the ID of the domain where the contract is currently assigned. This field is currently optional to ensure backwards compatibility. When specified, the domain-id of the disclosed contracts that are used in command interpretation is used to route the command submission to the specified domain-id. In case of domain-id mismatches, the possible errors are reported as command rejections with the eitherDISCLOSED_CONTRACTS_DOMAIN_ID_MISMATCH
orPRESCRIBED_DOMAIN_ID_MISMATCH
self-service error codes.
- New config option
parameters.timeouts.processing.sequenced-event-processing-bound
allows to specify a timeout for processing sequenced events. When processing takes longer on a node, the node will log an error or crash (depending on theexit-on-fatal-failures
parameter). - Fixed a crash recovery bug in unified sequencer, when it can miss events in the recovery process. Now it will start from the correct earlier block height in these situations.
- Removed party-level group addressing.
parallel_indexer
metrics have been renamed to simplyindexer
, i.e.daml_participant_api_parallel_indexer_inputmapping_batch_size_bucket
becomesdaml_participant_api_indexer_inputmapping_batch_size_bucket
- Completely removed leftovers in the code of Oracle support.
- Pruning and scheduled pruning along with pruning configuration have moved from enterprise to community. One slight caveat is scheduled sequencer pruning which is currently only wired up in the enterprise database sequencer.
- Sequencer types
type = external
andtype = BFT
can now configure the underlying block sequencer in the config sectioncanton.sequencers.<sequencer>.block
and uses the samereader
andwriter
configuration as thetype = database
sequencer.
canton {
sequencers {
sequencer1 {
type = external
config = {
// config for external sequencer (eg CometBFT)
}
block {
writer.checkpoint-interval = "10s"
checkpoint-backfill-parallelism = 2
reader.read-batch-size = 50
}
}
}
}
- Improve organization and layout of Ledger API Reference docs.
In the ledger api protobufs we used strings to represent the offset of a participant. The integer approach replaces string representation in:
- OffsetCheckpoint message: with int64
- CompletionStreamRequest message of command completion service: with int64.
- If specified, it must be a valid absolute offset (positive integer) or zero (ledger begin offset)..
- If not set, the ledger uses the ledger begin offset instead.
- GetLedgerEndResponse message: with int64
- It will always be a non-negative integer.
- If zero, the participant view of the ledger is empty.
- If positive, the absolute offset of the ledger as viewed by the participant.
- GetLatestPrunedOffsetsResponse message: with int64
- If positive, it is a valid absolute offset (positive integer).
- If zero, no pruning has happened yet.
- SubmitAndWaitForUpdateIdResponse message: with int64
- PruneRequest message (prune_up_to): with int64
- Reassignment, TransactionTree, Transaction and Completion (offset, deduplication_offset) message: with int64
- Commands message (deduplication_offset): with int64
- GetActiveContractsRequest message (active_at_offset): with int64 (non-negative offset expected)
- If zero, the empty set will be returned
- Note that previously if this field was not set the current ledger end was implicitly derived. This is no longer possible.
- GetActiveContractsResponse message: removed the offset field
- GetUpdatesRequest message,
- begin_exclusive: with int64 (non-negative offset expected)
- end_inclusive: with optional int64
- If specified, it must be a valid absolute offset (positive integer).
- If not set, the stream will not terminate.
- Re-onboarding members results in a rejection of the
DomainTrustCertificate
,SequencerDomainState
, orMediatorDomainState
with the errorMEMBER_CANNOT_REJOIN_DOMAIN
.
- Console.bootstrap.domain has new parameter domainThreshold, the minimum number of domain owners that need to authorize on behalf of the domain's namespace.
- [Breaking change]: added a new mandatory
usage: SigningKeyUsage
parameter for theregister_kms_signing_key()
and thegenerate_signing_key()
commands. This new parameter is used to specify the type of usage the new key will have. It can take the following usage types:Namespace
: the root namespace key that defines a node's identity and signs topology requests;IdentityDelegation
: a signing key that acts as a delegation key for the root namespace and that can also be used to sign topology requests;SequencerAuthentication
: a signing key that authenticates members of the network towards a sequencer;Protocol
: a signing key that deals with all the signing that happens as part of the protocol. This separation makes our system more robust in case of a compromised key.
- google.protobuf.XValue wrapper messages are replaced by
optional X
in the protobuf definitions. Incompatibility for manually crafted Protobuf messages and wire formats. Protobuf bindings must be regenerated, but should remain compatible. - Started the renaming transfer -> reassignment
- transferExclusivityTimeout -> assignmentExclusivityTimeout
- Added periodic generation of sequencer counter checkpoints to the sequencer and reworked SQL queries.
- This should improve performance for sequencer snapshotting and pruning and reduce database load.
- The checkpoint interval is configurable under
canton.sequencers.<sequencer>.writer.checkpoint-interval
(default: 30s):
writer {
checkpoint-interval = "30s"
}
- The
ParticipantOffset
message was removed since it was already replaced by a simpler string representation and was not used anymore.
- the DomainId field has been removed from the following topology mapping:
OwnerToKeyMapping
,VettedPackages
,PartyToParticipant
andAuthorityOf
. Those fields were not handled properly, so we decide to remove them. - two new endpoints added to
GrpcInspectionService
to inspect the state of sent and received ACS commitments on participants.lookupSentAcsCommitments
to retrieve sent ACS Commitments and their stateslookupReceivedAcsCommitments
to retrieve received ACS commitments and their states
- When not specifying
AuthorizeRequest.signed_by
orSignTransactionsRequest.signed_by
, suitable signing keys available to the node are selected automatically.
- The
GetLedgerApiVersion
method of theVersionService
contains newfeatures.offset_checkpoint
field within the returnedGetLedgerApiVersionResponse
message. It exposes themax_offset_checkpoint_emission_delay
which is the maximum time needed to emit a new OffsetCheckpoint.
- Error INVALID_SUBMITTER is changed to INVALID_READER
- Config of the jwt token leeway has been moved from
participants.participant.parameters.ledger-api-server.jwt-timestamp-leeway
toparticipants.participant.ledger-api.jwt-timestamp-leeway
- Creating a
MediatorDomainState
fails if a mediator is both in theactive
and theobservers
lists. - Creating a
SequencerDomainState
fails if a sequencer is both in theactive
and theobservers
lists.
In case it is suspected that a member's authentication tokens for the public sequencer API have been leaked or somehow compromised, we introduced new administration commands that allow an operator to revoke all the authentication tokens for a member and close the sequencer connections. The legitimate member then automatically reconnects and obtains new tokens. The commands are accessible via the console as, for example:
participant1.domains.logout(myDomainAlias)
mediator1.sequencer_connections.logout()
We have introduced additional package vetting validations that may result in package rejections:
- You cannot unvet a package unless you provide the force flag: FORCE_FLAG_ALLOW_UNVET_PACKAGE.
- You cannot vet a package that has not yet been uploaded unless you provide the force flag: FORCE_FLAG_ALLOW_UNKNOWN_PACKAGE.
- You cannot vet a package if its dependencies have not yet been vetted, unless you provide the force flag: FORCE_FLAG_ALLOW_UNVETTED_DEPENDENCIES.
Add mediators to multiple groups results in a rejection with error MEDIATORS_ALREADY_IN_OTHER_GROUPS
.
SetTrafficPurchased requests return immediately and no longer return the max sequencing time.
- Removed the GrpcTransferService
- Renamed metric
daml_sequencer_client_handler_delay
=>daml_block_delay
(sequencer block processing delay relative to sequencers local wall clock) - Added new metric
daml_sequencer_db_watermark_delay
(database sequencer watermark delay relative to sequencers local wall clock)
To support OffsetCheckpoints in completions stream changes are made to command completion service protobuf definitions.
- The Checkpoint message and the domain_id have been deleted from CompletionStreamResponse message. The domain id, offset
and record time are now encapsulated in Completion in the following way:
- an additional offset field to hold the offset
- an additional domain_time field to hold the (domain_id, record_time) pair
- The CompletionStreamResponse has been converted to oneof Completion and OffsetCheckpoint in the following way:
to
message CompletionStreamResponse { Checkpoint checkpoint = 1; Completion completion = 2; string domain_id = 3; }
message CompletionStreamResponse { oneof completion_response { Completion completion = 1; OffsetCheckpoint offset_checkpoint = 2; } }
- The
jwt-rs-256-jwks
auth service type in theparticipant.ledger-api.auth-services
configuration has been changed tojwt-jwks
to better represent the generic nature of the JWKS authorization.
- Additive change: new
CommandInspectionService
- CommandInspectionService added to
v2/admin
- Change in
VersionService.GetVersion
, the response extended withExperimentalCommandInspectionService
signalling presence of the new service
- CommandInspectionService added to
- Additive change:
PackageManagementService
extended with new methodValidateDarFile
- Additive change: Paging added to
ListKnownParties
of thePartyManagementService
- New fields in
ListKnownPartiesRequest
- New fields in
ListKnownPartiesResponse
- Change in
VersionService.GetVersion
, the response extended withPartyManagementFeature
signalling paging support and max page size
- New fields in
- Additive change: User management rights extended with a new claim
CanReadAsAnyParty
- Additive change: Party wildcard supported in
TransactionFilter
throughfilters_for_any_party
- Breaking change: Complete rewrite of the filtering in the
TransactionFilter
- Filters message changed
InclusiveFilters inclusive
becomesrepeated CumulativeFilter cumulative
InclusiveFilters
message removed in favor ofCumulativeFilter
WildcardFilter
message addedFilters
message cannot be empty
- Filters message changed
The following changes are not included into release-line-3.1.
In the ledger api protobufs we used both ParticipantOffset message and strings to represent the offset of a participant. The simpler string approach replaces ParticipantOffset in:
- GetLedgerEndResponse message, where an empty string denotes the participant begin offset
- GetLatestPrunedOffsetsResponse message, where an empty string denotes that participant is not pruned so far
- GetUpdatesRequest message,
- begin_exclusive is now a string where previous participant-offset values are mapped in the following manner:
ParticipantOffset.ParticipantBoundary.PARTICIPANT_BOUNDARY_BEGIN
is represented by an empty stringParticipantOffset.Absolute
is represented by a populated stringParticipantOffset.ParticipantBoundary.PARTICIPANT_BOUNDARY_END
cannot be represented anymore and previous references should be replaced by a prior call to retrieve the ledger end- absence of a value was invalid
- end_inclusive is now a string where previous participant-offset values are mapped in the following manner:
ParticipantOffset.ParticipantBoundary.PARTICIPANT_BOUNDARY_BEGIN
cannot be represented anymoreParticipantOffset.Absolute
is represented by a populated stringParticipantOffset.ParticipantBoundary.PARTICIPANT_BOUNDARY_END
cannot be represented anymore and previous references should be replaced by a prior call to retrieve the ledger end- absence of a value signifying an open-ended tailing stream is represented by an empty string
- begin_exclusive is now a string where previous participant-offset values are mapped in the following manner:
- The endpoint to download the genesis state for the sequencer is now available on all nodes, and it has been removed from the sequencer admin commands.
- To download the genesis state use:
sequencer1.topology.transactions.genesis_state()
instead ofsequencer.setup.genesis_state_for_sequencer()
- To download the genesis state use:
- A config option to randomize token life
canton.sequencers.<sequencer>.public-api.use-exponential-random-token-expiration = true|false
(defaults tofalse
). When enabled, it samples token life duration from an exponential distribution with scale ofmaxTokenExpirationInterval
, with the values truncated (re-sampled) to fit into an interval[maxTokenExpirationInterval / 2, maxTokenExpirationInterval]
, so the token will be between half and the value specified inmaxTokenExpirationInterval
. - Config option renamed to prevent confusion:
canton.sequencers.<sequencer>.public-api.token-expiration-time
=>canton.sequencers.<sequencer>.public-api.max-token-expiration-interval
canton.sequencers.<sequencer>.public-api.nonce-expiration-time
=>canton.sequencers.<sequencer>.public-api.nonce-expiration-interval
- Submission request amplification delays resending the submission request for a configurable patience. The sequencer connections' parameter
submission_request_amplification
is now a structured message of the previous factor and the patience. - Paging in Party Management
- The
ListKnownParties
method on thePartyManagementService
now takes two additional parameters. The newpage_size
field determines the maximum number of results to be returned by the server. The newpage_token
field on the other hand is a continuation token that signals to the server to fetch the next page containing the results. EachListKnownPartiesResponse
response contains a page of parties and anext_page_token
field that can be used to populate thepage_token
field for a subsequent request. When the last page is reached, thenext_page_token
is empty. The parties on each page are sorted in ascending order according to their ids. The pages themselves are sorted as well. - The
GetLedgerApiVersion
method of theVersionService
contains newfeatures.party_management
field within the returnedGetLedgerApiVersionResponse
message. It describes the capabilities of the party management through a sub-message calledPartyManagementFeature
. At the moment it contains just one field themax_parties_page_size
which specifies the maximum number of parties that will be sent per page by default. - The default maximum size of the page returned by the participant in response to the
ListKnownParties
call has been set to 10'0000. It can be modified through themax-parties-page-size
entry:
canton.participants.participant.ledger-api.party-management-service.max-parties-page-size=777
- The
- Mediator initialization cleanup
- Removed
InitializeMediatorRequest.domain_parameters
- Removed
MediatorDomainConfiguration.initialKeyFingerprint
and corresponding entry in the database - The static parameters are determined from the set of sequencers provided during initialization via
mediator.setup.assign(...)
or the grpc admin api callMediatorInitializationService.InitializeMediator
.
- Removed
- Canton Node initialization cleanup
- Renamed to remove
X
fromcom.digitalasset.canton.topology.admin.v30.IdentityInitializationXService
- Renamed to remove
- Daml Logging works again, logging by default during phase 1 at Debug log level.
- The
NO_INTERNAL_PARTICIPANT_DATA_BEFORE
error code is introduced and returned whenparticipant.pruning.find_safe_offset
is invoked with a timestamp before the earliest known internal participant data. Before this changefind_safe_offset
used to returnNone
in this case thus making it impossible to distinguish the situation from no safe offset existing. Whenfind_safe_offset
returnsNO_INTERNAL_PARTICIPANT_DATA_BEFORE
, it is safe to invokeparticipant.pruning.prune
with an offset corresponding to the timestamp passed tofind_safe_offset
. vetted_packages.propose_delta
no longer allows specifying aserial
parameter, and instead increments the serial relative to the last authorized topology transaction.- The new repair method
participant.repair.purge_deactivated_domain
allows removing data from the deactivated domain after a hard domain migration. - Repair method
participant.repair.migrate_domain
features a newforce
flag. When settrue
it forces a domain migration ignoring in-flight transactions. - Removed the protobuf message field
BaseQuery.filterOperation
. Setting the fieldBaseQuery.operation
will use it as filter criteria. - Sequencer subscription now will not return
InvalidCounter(...)
when sequencer cannot sign the event, now it will always return a tombstone with aTombstoneEncountered
error. This can happen when a newly onboarded sequencer cannot sign a submission originated before it was bootstrapped or if manually initialized sequencer cannot find its keys. - When connecting to sequencer nodes, participants and mediators return once
sequencerTrustThreshold * 2 + 1
sequencers return valid endpoints unlessSequencerConnectionValidation.ALL
is requested.
In the ledger api protobufs we used both ParticipantOffset message and strings to represent the offset of a participant. The simpler string approach replaces ParticipantOffset in:
- Checkpoint message
- CompletionStreamRequest of command completion service. In particular, the
begin_exclusive
field have been converted to string. Before, the absence of this field was denoting the participant end, while currently the empty string means the participant begin. Thus, if the completion stream starting from the participant end is needed the begin_exclusive offset has to be explicitly given by first querying for the participant end.
In the protobufs, we use participant_id
to sometimes contain PAR::uid
and sometimes only uid
, without
the three-letter code and similar for the other member IDs. Moreover, mediator
contains sometimes a uid
and sometimes the mediator group. The goal is to make it explicit what the field contains:
- Use _uid suffix if the field does not contain the three-letters code
- Use member if it can be any member (with the three-letters code)
Changed field: SequencerConnect.GetDomainIdResponse.sequencer_id -> sequencer_uid (format changed, code removed) SequencerNodeStatus.connected_participants -> connected_participant_uids (format changed, code removed) OrderingRequest.sequencer_id -> OrderingRequest.sequencer_uid (format changed, code removed) ListPartiesResponse.Result.ParticipantDomains.participant -> participant_uid (format changed, code removed) OnboardingStateRequest.sequencer_id -> sequencer_uid (format changed, code removed)
The Ledger API and Admin API gRPC services used for package management now use the same backend logic and storage. There is no Ledger/Admin API client impact, but the following changes are breaking compatibility:
par_daml_packages
is extended withpackage_size
anduploaded_at
, both non-null. A fresh re-upload of all packages is required to conform.ledger_sync_event.proto
drops the package notification ledger events:public_package_upload
andpublic_package_upload_rejected
canton.participants.participant.parameters.ledger-api-server.indexer.package-metadata-view
has been moved tocanton.participants.participant.parameters.package-metadata-view
.com.digitalasset.canton.admin.participant.v30.PackageService
RemoveDar
andRemovePackage
operations become dangerous and are not recommended for production usage anymore. Unadvised usage can lead to broken Ledger API if packages are removed for non-pruned events referencing them. Additionally, as relevant but non-impacting changes:- Ledger API Index database drops all references to package data. The Ledger API uses
par_daml_packages
orpar_dars
for all package/DARs operations.
In order to improve debugging of failed commands, the participant now stores the last few commands (successes, failures and pending) in memory for debug inspection. The data is accessible through the command inspection service on the ledger api.
Before we combined keys and crypto algorithms into a single key scheme, for example EciesP256HkdfHmacSha256Aes128Gcm and EciesP256HmacSha256Aes128Cbc.
The underlying EC key is on the P-256 curve and could be used with both AES-128-GCM and -CBC as part of a hybrid encryption scheme.
Therefore, we decided to split this scheme into a key (EncryptionKeySpec)
and algorithm (EncryptionAlgorithmSpec)
specifications.
We also changed the way this is configured in Canton, for example:
-
encryption.default = rsa-2048-oaep-sha-256
is now represented as:encryption.algorithms.default = rsa-oaep-sha-256
encryption.keys.default = rsa-2048
When a participant replica becomes active, it does not refresh the package dependency cache. If a vetting attempt is made on the participant that fails because the package is not uploaded, the "missing package" response is cached. If the package is then uploaded to another replica, and we switch to the original participant, this package service cache will still record the package as nonexistent. When the package is used in a transaction, we will get a local model conformance error as the transaction validator cannot find the package, whereas other parts of the participant that don't use the package service can successfully locate it.
Participant
3.0, 3.1
Replica crashes during transaction validation.
Validating participant emits warning:
LOCAL_VERDICT_FAILED_MODEL_CONFORMANCE_CHECK(5,a2b60642): Rejected transaction due to a failed model conformance check: UnvettedPackages
And then emits an error:
An internal error has occurred.
java.lang.IllegalStateException: Mediator approved a request that we have locally rejected
Restart recently active replica
Likely to happen in any replicated participant setup with frequent vetting attempts and switches between active and passive replicated participants between those vetting attempts.
Users are advised to upgrade to the next minor release (3.2) during their maintenance window.
(24-015, Minor): Pointwise flat transaction Ledger API queries can unexpectedly return TRANSACTION_NOT_FOUND
When a party submits a command that has no events for contracts whose stakeholders are amongst the submitters, the resulted transaction cannot be queried by pointwise flat transaction Ledger API queries. This impacts GetTransactionById, GetTransactionByEventId and SubmitAndWaitForTransaction gRPC endpoints.
Participant
User might perceive that a command was not successful even if it was.
TRANSACTION_NOT_FOUND is returned on a query that is expected to succeed.
Query instead the transaction tree by transaction-id to get the transaction details.
Lower likelihood as commands usually have events whose contracts' stakeholders are amongst the submitting parties.
Users are advised to upgrade to the next patch release during their maintenance window.
When a participant handles a malformed request (for instance because topology changed during the request processing and a party was added, causing the recipient list to be invalid), it will attempt to send a response to the mediator. If the sending fails (for instance because max sequencing time has elapsed), the request never gets cleaned up. This is not fixed by crash recovery because the same thing will happen again as max sequencing time is still elapsed, and therefore the request stays dirty.
Participant
An affected participant cannot be pruned above the last dirty request and crash recovery will take longer as it restarts from that request as well.
The number of dirty requests reported by the participant never reaches 0.
No workaround exists. You need to upgrade to a version not affected by this issue.
Not very likely as only triggered by specific malformed events followed by a failure to send the response the sequencer. Concurrent topology changes and participant lagging behind the domain increase the odds of it happening.
Upgrade during your next maintenance window to a patch version not affected by this issue.
- TransactionFilters have been extended to hold filters for party-wildcards:
TransactionFilter message changed from
message TransactionFilter {
map<string, Filters> filters_by_party = 1;
}
to
message TransactionFilter {
map<string, Filters> filters_by_party = 1;
Filters filters_for_any_party = 2;
}
- Filters changed to include a list of cumulative filters:
Filters message changed from
message Filters {
// Optional
InclusiveFilters inclusive = 1;
}
to
message Filters {
// Optional
repeated CumulativeFilter cumulative = 1;
}
- Inclusive filters where changed to cumulative filter which support a Wildcard filter that matches all the templates (template-wildcard). Every filter in the cumulative list expands the scope of the resulting stream. Each interface, template or wildcard filter means additional events that will match the query.
InclusiveFilters message changed from
message InclusiveFilters {
// Optional
repeated InterfaceFilter interface_filters = 1;
// Optional
repeated TemplateFilter template_filters = 2;
}
to
message CumulativeFilter {
oneof identifier_filter {
// Optional
WildcardFilter wildcard_filter = 1;
// Optional
InterfaceFilter interface_filter = 2;
// Optional
TemplateFilter template_filter = 3;
}
}
- The new wildcard filter that is used to match all the templates (template-wildcard) includes the
include_created_event_blob
flag to control the presence of thecreated_event_blob
in the returnedCreatedEvent
.
WildcardFilter message added:
message WildcardFilter {
// Optional
bool include_created_event_blob = 1;
}
- We changed the retry policy for checking the creation of KMS crypto keys to use exponential backoff, so the configuration for the
retry-config.create-key-check
is now done similarly as theretry-config.failures
canton.participants.participant1.crypto.kms.retries.create-key-check { initial-delay = "0.1s", max-delay = "10 seconds", max-retries = 20, }
-
health.running
is renamed tohealth.is_running
-
AcsCommitmentsCatchUpConfig
is removed fromStaticDomainParameters
in proto files -
When an access token expires and ledger api stream is terminated an
ABORTED(ACCESS_TOKEN_EXPIRED)
error is returned instead ofUNAUTHENTICATED(ACCESS_TOKEN_EXPIRED)
. -
The participant.domains.connect* methods have been modified in order to accommodate a new sequencer connection validation argument, which caused the existing commands to no longer work due to ambiguous default arguments. The connect methods will likely be reworked in the future to improve consistency and usability, as right now, there are too many of them with different capabilities and user experience.
-
The
MetricsConfig
has been altered. The boolean argumentreport-jvm-metrics
has been replaced with a more finegrained control over the available jvm metrics. Usejvm-metrics.enabled = true
to recover the previous metrics. -
Many metrics have been renamed and restructured. In particular, labelled metrics are used now instead of the older ones where the node name was included in the metric name.
-
The Jaeger trace exporter is no longer supported, as OpenTelemetry and Jaeger suggest to configure Jaeger using the otlp exporter instead of the custom Jaeger exporter.
-
The arguments of the RateLimitConfig have been renamed, changing
maxDirtyRequests
tomaxInflightValidationRequests
andmaxRate
tomaxSubmissionRate
andmaxBurstFactor
tomaxSubmissionBurstFactor
.
Operation changed from
enum TopologyChangeOp {
// Adds a new or replaces an existing mapping
TOPOLOGY_CHANGE_OP_REPLACE_UNSPECIFIED = 0;
// Remove an existing mapping
TOPOLOGY_CHANGE_OP_REMOVE = 1;
}
to
enum TopologyChangeOp {
TOPOLOGY_CHANGE_OP_UNSPECIFIED = 0;
// Adds a new or replaces an existing mapping
TOPOLOGY_CHANGE_OP_ADD_REPLACE = 1;
// Remove an existing mapping
TOPOLOGY_CHANGE_OP_REMOVE = 2;
}
SequencerDriver.adminServices
now returnsSeq[ServerServiceDefinition]
The admin api for sequencer initialization has changed:
-
SequencerInitializationService.InitializeSequencer
is now calledSequencerInitializationService.InitializeSequencerFromGenesisState
. Thetopology_snapshot
field is a versioned serialization ofStoredTopologyTransactionsX
(scala) /TopologyTransactions
(protobuf). -
Onboarding a sequencer on an existing domain is now expected to work as follows:
- A node (usually one of the domain owners) uploads the new sequencer's identity transactions to the domain
- The domain owners add the sequencer to the SequencerDomainState
- A domain owner downloads the onboarding state via
SequencerAdministrationService.OnboardingState
and provides the returned opaquebytes onboarding_state
to the new sequencer. - The new sequencer then gets initialized with the opaque onboarding state via
SequencerInitializationService.InitializeSequencerFromOnboardingState
.
- The default mediator admin api port has been changed to
6002
. - Database sequencer writer and reader high throughput / high availability configuration defaults have been updated to optimize latency.
- Ledger API field
Commands.workflow_id
at command submission cannot be used anymore for specifying the prescribed domain. For this purpose the usage ofCommands.domain_id
is available.
SequencerConnections
now requires asubmissionRequestAmplification
field. By default, it should be set to 1.- A few classes and configs were renamed:
- Config
canton.mediators.mediator.caching.finalized-mediator-requests
->canton.mediators.mediator.caching.finalized-mediator-confirmation-requests
- DB column
response_aggregations.mediator_request
->response_aggregations.mediator_confirmation_request
- Proto:
com.digitalasset.canton.protocol.v30.MediatorResponse
->com.digitalasset.canton.protocol.v30.ConfirmationResponse
- Proto file renamed:
mediator_response.proto
->confirmation_response.proto
- Proto:
com.digitalasset.canton.protocol.v30.MalformedMediatorRequestResult
->com.digitalasset.canton.protocol.v30.MalformedMediatorConfirmationRequestResult
- Proto:
com.digitalasset.canton.protocol.v30.TypedSignedProtocolMessageContent
field:mediator_response
->confirmation_response
- Proto:
com.digitalasset.canton.protocol.v30.TypedSignedProtocolMessageContent
field:malformed_mediator_request_result
->malformed_mediator_confirmation_request_result
- Dynamic domain parameter and respective proto field:
com.digitalasset.canton.protocol.v30.DynamicDomainParameters.participant_response_timeout
->com.digitalasset.canton.protocol.v30.DynamicDomainParameters.confirmation_response_timeout
- Dynamic domain parameter:
maxRatePerParticipant
->confirmationRequestsMaxRate
and in its respective protocom.digitalasset.canton.protocol.v30.ParticipantDomainLimits
fieldmax_rate
->confirmation_requests_max_rate
- Config
- Removed support for optimistic validation of sequenced events (config option
optimistic-sequenced-event-validation
in the sequencer client config).
Console commands that allow to download an ACS snapshot now take a new mandatory argument to indicate whether the snapshot will be used in the context of a party offboarding (party replication or not). This allows Canton to performance additional checks and makes party offboarding safer.
Affected console command:
participant.repair.export_acs
New argument: partiesOffboarding: Boolean
.
- The scala type
ParticipantPermissionX
has been renamed toParticipantPermission
to reflect the changes in the proto files.
-
The GRPC proto files no longer contain the "X-nodes" or "topology-X" suffixes. Specifically the following changes require adaptation:
- Topology mappings X-suffix removals with pattern
TopologyMappingX
->TopologyMapping
:NamespaceDelegation
,IdentifierDelegation
,OwnerToKeyMapping
,TrafficControlState
,VettedPackages
,DecentralizedNamespaceDefinition
,DomainTrustCertificate
,ParticipantDomainPermission
,PartyHostingLimits
,PartyToParticipant
,AuthorityOf
,MediatorDomainState
,SequencerDomainState
,PurgeTopologyTransaction
,DomainParametersState
- Services X removals: *XService -> *Service, *XRequest -> *Request, *XResponse -> *Response, specifically:
TopologyManagerWriteService
,TopologyManagerReadService
- Miscellaneous messages whose X-suffix has been removed
StaticDomainParameters
,TopologyTransactionsBroadcast
EnumsX
->Enums
EnumsX.TopologyChangeOpX
->Enums.TopologyChangeOp
EnumsX.ParticipantPermissionX
->Enums.ParticipantPermission
: In addition the following previous had an embedded X:PARTICIPANT_PERMISSION_SUBMISSION
,PARTICIPANT_PERMISSION_CONFIRMATION
,PARTICIPANT_PERMISSION_OBSERVATION
,PARTICIPANT_PERMISSION_UNSPECIFIED
- Topology mappings X-suffix removals with pattern
-
Less importantly the old topology GRPC proto removals should not require adaptation. Note that some removals (marked
*
below) "make room" for the X-variants above to use the name, e.g.NamespaceDelegation
formerly referring to the old "NSD" mapping, is now used for the daml 3.x-variant:TopologyChangeOp
*,TrustLevel
,ParticipantState
,RequestSide
- Old topology mappings:
PartyToParticipant
,MediatorDomainState
,NamespaceDelegation
,IdentifierDelegation
,OwnerToKeyMapping
,SignedLegalIdentityClaim
,LegalIdentityClaim
,VettedPackages
*,TopologyStateUpdate
,DomainParametersChange
- Old topology transactions:
SignedTopologyTransaction
,TopologyTransaction
- Old topology services and messages:
TopologyManagerWriteService
,TopologyManagerReadService
,RegisterTopologyTransactionRequest
,RegisterTopologyTransactionResponse
,DomainTopologyTransactionMessage
- Renamed the following error codes: SEQUENCER_SIGNING_TIMESTAMP_TOO_EARLY to SEQUENCER_TOPOLOGY_TIMESTAMP_TOO_EARLY SEQUENCER_SIGNING_TIMESTAMP_AFTER_SEQUENCING_TIMESTAMP to SEQUENCER_TOPOLOGY_TIMESTAMP_AFTER_SEQUENCING_TIMESTAMP SEQUENCER_SIGNING_TIMESTAMP_MISSING to SEQUENCER_TOPOLOGY_TIMESTAMP_MISSING
- Check that packages are valid upgrades of the package they claim to upgrade at upload-time in
ApiPackageManagementService
.
-
Executor Service Metrics removed The metrics for the execution services have been removed:
- daml.executor.runtime.completed*
- daml.executor.runtime.duration*
- daml.executor.runtime.idle*
- daml.executor.runtime.running*
- daml.executor.runtime.submitted*
- daml_executor_pool_size
- daml_executor_pool_core
- daml_executor_pool_max
- daml_executor_pool_largest
- daml_executor_threads_active
- daml_executor_threads_running
- daml_executor_tasks_queued
- daml_executor_tasks_executing_queued
- daml_executor_tasks_stolen
- daml_executor_tasks_submitted
- daml_executor_tasks_completed
- daml_executor_tasks_queue_remaining
-
The recipe for sequencer onboarding has changed to fetch the sequencer snapshot before the topology snapshot. The topology snapshot transactions should be filtered by the last (sequenced) timestamp ("lastTs") of the sequencer snapshot.
- The
TrustLevel
was removed from theParticipantDomainPermissionX
proto and the fields were renumbered (see #16887)
- The
DomainAlias
in*connect_local
is now non-optional- (i.e
participant.connect_local(sequencer, alias=Some(domainName))
is nowparticipant.connect_local(sequencer, alias=domainName)
)
- (i.e
- Participants cannot submit on behalf of parties with confirmation threshold > 1, even if they have submission permission.
- When an access token expires and stream is terminated an UNAUTHENTICATED(ACCESS_TOKEN_EXPIRED) error is returned.
- Support for Unique Contract Key (UCK) semantics has been removed.
- The administration services have been restructured as follows:
EnterpriseMediatorAdministrationService
is nowMediatorAdministrationService
.Snapshot
andDisableMember
have been moved fromEnterpriseSequencerAdministrationService
toSequencerAdministrationService
.EnterpriseSequencerAdministrationService
is nowSequencerPruningAdministrationService
.EnterpriseSequencerConnectionService
is nowSequencerConnectionService
.- The
AuthorizeLedgerIdentity
endpoint has been removed.
token-expiry-grace-period-for-streams
config parameter added.- As part of daml 2.x, non-x-node removal:
- Canton configuration now refers to nodes as "canton.participants", "canton.sequencers", and "canton.mediators" (rather than as "canton.participants-x", "canton.sequencers-x", and "canton.mediators-x").
- Similarly remote nodes now reside under "canton.remote-participants", "canton.remote-sequencers", and "canton.remote-mediators" (i.e. the "-x" suffix has been removed).
- Packages for admin services and messages have been extracted to a dedicated project which results in
new package paths.
Migration:
- Renaming:
com.digitalasset.canton.xyz.admin
->com.digitalasset.canton.admin.xyz
com.digitalasset.canton.traffic.v0.MemberTrafficStatus
->com.digitalasset.canton.admin.traffic.v0.MemberTrafficStatus
- Some messages are moved from
api
toadmin
:SequencerConnection
:com.digitalasset.canton.domain.api.v0
->com.digitalasset.canton.admin.domain.v0
SequencerConnections
:com.digitalasset.canton.domain.api.v0
->com.digitalasset.canton.admin.domain.v0
- Renaming:
- Renamed
Unionspace
withDecentralized Namespace
. Affects all classes, fields, options, and RPC endpoints withunionspace
in their name. BaseResult.store
returned by theTopologyManagerReadServiceX
is now typed so that we can distinguish between authorized and domain stores.
- Replaced
KeyOwner
with theMember
trait in thekeys.private
andowner_to_key_mappings.rotate_key
commands. - Removed the deprecated
owner_to_key_mappings.rotate_key
command without thenodeInstance
parameter. - Removed the deprecated ACS download / upload functionality and
connect_ha
participant admin commands. - Removed the deprecated
update_dynamic_parameters
andset_max_inbound_message_size
domain admin commands. - Removed the deprecated
acs.load_from_file
repair macro. - v0.SignedContent is deprecated in favor of v1.SignedContent in SequencerService.
Migration: field
SignedContent.signatures
becomes repeated
- Split of the lines. From now on, snapshot will be 3.0.0-SNAPSHOT