-
Notifications
You must be signed in to change notification settings - Fork 23
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: R.I.Pienaar <[email protected]>
- Loading branch information
Showing
1 changed file
with
87 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,87 @@ | ||
# JetStream Per-Message TTL | ||
|
||
| Metadata | Value | | ||
|----------|---------------------------| | ||
| Date | 2024-07-11 | | ||
| Author | @ripienaar | | ||
| Status | Proposed | | ||
| Tags | jetstream, client, server | | ||
|
||
## Context and motivation | ||
|
||
Streams support a one-size-fits-all approach to message TTL based on the MaxAge setting. This causes any message in the | ||
Stream to expire at that age. | ||
|
||
There are numerous uses for a per-message version of this limit, some listed below: | ||
|
||
* KV tombstones are a problem in that they forever clog up the buckets with noise, these could have a TTL to make them expire once not useful anymore | ||
* Server-applied limits can result in tombstones with a short per message TTL so that consumers can be notified of limits being processed. Useful in KV watch scenarios being notified about TTL removals | ||
* KV tombstones can be expanded to also include Limits being applied by the server, with a TTL attached, to facilitate notification of limits in watchers | ||
* A stream may have a general MaxAge but some messages may have infinite retention, think a schema or type hints in a KV bucket that is forever while general keys have TTLs | ||
* TODO: user use cases | ||
|
||
Related issues [#3268](https://github.com/nats-io/nats-server/issues/3268) | ||
|
||
## Per-Message TTL | ||
|
||
We will allow a message to supply a TTL as a duration in a header called `Nats-TTL` followed by the duration as | ||
nanoseconds (TODO: or seconds? I prefer seconds but consistency suggests ns). | ||
|
||
The duration will be used by the server to calculate the deadline for removing the message based on its Stream | ||
timestamp and the stated duration. | ||
|
||
The TTL may not exceed the Stream MaxAge to give administrators a hard cap for the purposes of managing resources. | ||
|
||
A TTL of zero or other unparsable value will result in a error (TODO: or ignore? I prefer error) and reported in the | ||
Pub Ack. | ||
|
||
Any value less than 0 will result in a message being retained forever, even if the stream have a MaxAge set. | ||
|
||
## Limit Tombstones | ||
|
||
Several scenarios for server-created tombstones can be imagined, the most often requested one though is when MaxAge | ||
removes last value (ie. the current value) for a Key. | ||
|
||
In this case when the server removes a message and the message is the last in the subject it would place a message | ||
with a TTL matching the Stream configuration value. The following headers would be placed: | ||
|
||
``` | ||
Nats-Applied-Limit: MaxAge | ||
Nats-TTL: 1000000000 | ||
``` | ||
|
||
The `Nats-Limit-Applied` field is there to support future expansion of this feature. | ||
|
||
This behaviour is off by default unless opted in on the Stream Configuration. | ||
|
||
## Publish Acknowledgements | ||
|
||
We could optionally extend the `PubAck` as follows: | ||
|
||
```golang | ||
type PubAck struct { | ||
MsgTTL uint64 `json:"msg_ttl,omitempty"` | ||
} | ||
``` | ||
|
||
This gives clients a chance to confirm, without Stream Info or should the Stream be edited after Info, if the TTL | ||
got applied. | ||
|
||
## Stream Configuration | ||
|
||
Weather or not a stream support this behavior should be a configuration opt-in. We want clients to definitely know | ||
when this is supported which the opt-in approach with a boolean on the configuration would make clear. | ||
|
||
We have to assume someone will want to create a replication topology where at some point in the topology these tombstone | ||
type messages are retained for an audit trail. So a Stream with this feature enabled can replicate to one with it | ||
disabled and all the messages that would have been TTLed will be retained. | ||
|
||
```golang | ||
type StreamConfig struct { | ||
// AllowMsgTTL allows header initiated per-message TTLs | ||
AllowMsgTTL bool `json:"allow_msg_ttl"` | ||
|
||
// LimitsTTL activates writing of messages when limits are applied with a specific TTL | ||
LimitsTTL time.Duration `json:"limits_ttl"` | ||
} | ||
``` |