This file tracks changes to this project. It follows the Keep a Changelog format, and this project adheres to Semantic Versioning.
If you're working with LLMs in Go this update will make you happy! sse.Read
is now a thing – it just parses all events from an io.Reader
. Use it with your response bodies and forget about any sse.Client
configuration. It also makes use of the new Go 1.23 iterators to keep your code neat and tidy.
Read
andReadConfig
This is the replayer update. Oh, what is a "replayer"? It's how we call replay providers starting with this version! Anyway, besides renaming, this update removes many replaying bugs, improves performance, robustness and error handling and better defines expected behavior for ReplayProviders
... err, Replayers
.
More such overhauls are planned. I'm leaving it up to you to guess which comes next – the server or the client? ;)
FiniteReplayer.{Count, AutoIDs}
– use the constructor instead.ValidReplayer.{TTL, AutoIDs}
– use the constructor instead.
- The
ReplayProvider
and related entities are renamed to justReplayer
.go-sse
strives to have a minimal and expressive API, and minimal and expressive names are an important step in that direction. The changelog will use the new names onwards. - Due to a change in the internal implementation, the
FiniteReplayer
is now able to replay events only if the event with the LastEventID provided by the client is still buffered. Previously if the LastEventID was that of the latest removed event, events would still be replayed. This detail added complexity to the implementation without an apparent significant win, so it was dropped. FiniteReplayer.GCInterval
should be set to0
now in order to disable GC.- Automatic ID generation for both replayers does not overwrite already existing message IDs and errors instead. Ensure that your events do not have IDs when using replayers configured to generate IDs.
Replayer.Put
now returns an error instead of being required to panic. Read the method documentation for more info.Joe
also propagates this error throughJoe.Publish
.- Replayers are now required to not overwrite message IDs and return errors instead. Sending unsupported messages to replayers is a bug which should not go unnoticed. Both replayers in this library now implement this behavior.
Joe
does not log replayer panics to the console anymore. Handle these panics inside the replay provider itself.
NewFiniteReplayer
constructorNewValidReplayer
constructorConnection.Buffer
FiniteReplayer
doesn't leak memory anymore and respects the stored messages count it was given. Previously when a new message was put after the messages count was reached and some other messages were removed, the total messages count would grow unexpectedly andFiniteReplayer
would store and replay more events than it was configured to.ValidReplayer
was also susceptible to a similar memory leak, which is also fixed now.- #41 –
sse.Session
now writes the header explicitly when upgrading.
This version removes all external dependencies of go-sse
. All our bugs are belong to us! It also does some API and documentation cleanups.
Client.DefautReconnectionTime
,Client.MaxRetries
have been replaced with the newClient.Backoff
configuration field. See the Added section for more info.ErrReplayFailed
is removed from the public API.ReplayProviderWithGC
andJoe.ReplayGCInterval
are no more. The responsibility for garbage collection is assigned to the replay providers.
Server.Logger
is now of a new type: theLogger
interface. The dependency on x/exp/slog is removed. This opens up the possibility to adapt any existing logger to be usable withServer
.- The default backoff behavior has changed. The previous defaults map to the new
Backoff
configuration as follows:
sse.Backoff{
InitialInterval: 5 * time.Second, // currently 500ms
Multiplier: 1.5, // currently the same
Jitter: 0.5, // currently the same
MaxInterval: 60 * time.Second, // currently unbounded
MaxElapsedDuration: 15 * time.Minute, // currently unbounded
MaxRetries: -1, // previously no retries by default, currently unbounded
}
Joe
now accepts new subscriptions even if replay providers panic (previouslyErrReplayFailed
would be returned).Server.ServeHTTP
panics if a customOnSession
handler returns aSubscription
with 0 topics
- The
Logger
interface,LogLevel
type, andLogLevel(Info|Warn|Error)
values. Backoff
andClient.Backoff
– the backoff strategy is now fully configurable. See the code documentation for info.ValidReplayProvider.GCInterval
, to configure at which interval expired events should be cleaned up.
This version overhauls connection retry and fixes the connection event dispatch order issue. Some internal changes to Joe were also made, which makes it faster and more resilient.
ConnectionError.Temporary
ConnectionError.Timeout
- Go's
Timeout
andTemporary
interfaces are not used anymore – the client makes no assumptions and retries on every network or response read error. The only cases whenConnection.Connect
returns now are either when there are no more retries left (when the number is not infinite), or when the request context was cancelled. *url.Error
s that occur on the HTTP request are now unwrapped and their cause is put inside aConnectionError
.Connection.Connect
doesn't suppress any errors anymore: the request context errors are returned as is, all other errors are wrapped insideConnectionError
.- On reconnection attempt, the response reset error is now wrapped inside
ConnectionError
. With this change, all errors other than the context errors are wrapped insideConnectionError
. - Subscription callbacks are no longer called in individual goroutines. This caused messages to be received in an indereminate order. Make sure that your callbacks do not block for too long!
- If a
ReplayProvider
method panics when called byJoe
, instead of closing itself completely it just stops replaying, putting or GC-ing messages to upcoming clients.Joe
continues to function as if no replay provider was given. A stack trace is printed to stderr when such a panic occurs.
0.6.0 - 2023-07-22
This version brings a number of refactors to the server-side tooling the library offers. Constructors and construction related types are removed, for ease of use and reduced API size, concerns regarding topics and expiry were separated from Message
, logging of the Server
is upgraded to structured logging and messages can be now published to multiple topics at once. Request upgrading has also been refactored to provide a more functional API, and the Server
logic can now be customized without having to create a distinct handler.
Message.ExpiresAt
is no more.Message.Topic
is no more. See the changes toServer
,Provider
andReplayProvider
for handling topics – you can now publish a message to multiple topics at once.Message.Writer
is no more. The API was redundant – one can achieve the same usingstrings.Builder
andMessage.AppendData
. See theMessageWriter
example for more.NewValidReplayProvider
is no more.NewFiniteReplayProvider
is no more.NewJoe
is no more.JoeConfig
is no more.Server.Subscribe
is no more – it never made sense.Server.Provider
is no more.NewServer
,ServerOption
and friends are no more.- The
Logger
interface and the capability of theServer
to use types that implementLogger
as logging systems is removed. SubscriptionCallback
is no more (see the change to theSubscription
type in the "Changed" section).
- Because the
ValidReplayProvider
constructor was removed, the fieldsValidReplayProvider.{TTL,AutoIDs}
were added for configuration. - Because the
FiniteReplayProvider
constructor was removed, the fieldsFiniteReplayProvider.{Count,AutoIDs}
were added for configuration. - Because the
Joe
constructor was removed, the fieldsJoe.{ReplayProvider,ReplayGCInterval}
were added for configuration. - Because the
Server
constructor was removed, the fieldServer.Provider
was added for configuration. - New
MessageWriter
interface; used by providers to send messages and implemented bySession
(previously namedRequest
). - New
ResponseWriter
interface, which is ahttp.ResponseWriter
augmented with aFlush
method. ValidReplayProvider
has a new fieldNow
which allows providing a custom current time getter, liketime.Now
, to the provider. Enables deterministic testing of dependents onValidReplayProvider
.- New
Server.OnSession
field, which enables customization ofServer
's response and subscriptions. - New
Server.Logger
field, which enables structured logging with logger retrieved from the request and customizable config of logged information.
ReplayProvider.Put
takes a simple*Message
and returns a*Message
, instead of changing the*Message
to which the**Message
parameter points. It also takes a slice of topics, given that theMessage
doesn't hold the topic itself anymore. If the Message cannot be put, the method must now panic – see documentation for info.- Because
Message.ExpiresAt
is removed, theValidReplayProvider
sets the expiry itself. Server.Publish
now takes a list of topics.Provider.Publish
now takes a non-empty slice of topics.ReplayProvider.Put
now takes a non-empty slice of topics.Provider.Stop
is nowProvider.Shutdown
and takes now acontext.Context
as a parameter.Server.Shutdown
takes now acontext.Context
as a parameter.Request
is now namedSession
and exposes the HTTP request, response writer, and the last event ID of the request.- A new method
Flush
is added toSession
; messages are no longer flushed by default, which allows providers, replay providers to batch send messages. Upgrade
now takes an*http.Request
as its second parameter.Subscription
now has aClient
field of typeMessageWriter
instead of aCallback
.- Given the
Subscription
change,Provider.Subscribe
andReplayProvider.Replay
now report message sending errors.
0.5.2 - 2023-07-12
- The new
Message.Writer
– write to theMessage
as if it is anio.Writer
.
Message.UnmarshalText
now strips the leading Unicode BOM, if it exists, as per the specification.- When parsing events client-side, BOM removal was attempted on each event input. Now the BOM is correctly removed only when parsing is started.
0.5.1 - 2023-07-12
Message.WriteTo
now writes nothing ifMessage
is empty.Message.WriteTo
does not attempt to write theretry
field ifMessage.Retry
is not at least 1ms.NewType
error message is updated to say "event type", not "event name".
0.5.0 - 2023-07-11
This version comes with a series of internal refactorings that improve code readability and performance. It also replaces usage of []byte
for event data with string
– SSE is a UTF-8 encoded text-based protocol, so raw bytes never made sense. This migration improves code safety (less unsafe
usage and less worry about ownership) and reduces the memory footprint of some objects.
Creating events on the server is also revised – fields that required getters and setters, apart from data
and comments, are now simple public fields on the sse.Message
struct.
Across the codebase, to refer to the value of the event
field the name "event type" is used, which is the nomenclature used in the SSE specification.
Documentation and examples were also fixed and improved.
- New
sse.EventName
type, which holds valid values for theevent
field, together with constructors (sse.Name
andsse.NewName
).
sse.Message
:AppendText
was removed, as part of the migration from byte slices to strings. SSE is a UTF-8 encoded text-based protocol – raw bytes never made sense.
- Minimum supported Go version was bumped from 1.16 to 1.19. From now on, the latest two major Go versions will be supported.
sse.Message
:AppendData
takesstring
s instead of[]byte
.sse.Message
:Comment
is now namedAppendComment
, for consistency withAppendData
.sse.Message
: The message's expiration is not reset anymore byUnmarshalText
.sse.Message
:UnmarshalText
now unmarshals comments aswell.sse.Message
:WriteTo
(andMarshalText
andString
as a result) replaces all newline sequences in data with LF.sse.Message
: TheExpiry
getter andSetExpiresAt
,SetTTL
setters are replaced by the public fieldExpiresAt
.sse.Message
: Event ID getter and setter are replaced by the publicID
field.sse.Message
: Event type (previously namedName
) getter and setter are replaced by the publicType
field.sse.Message
: Theretry
field value is now a public field on the struct. As a byproduct,WriteTo
will now make 1 allocation when writing events with theretry
field set.sse.NewEventID
is nowsse.NewID
, andsse.MustEventID
issse.ID
.sse.Event
: TheData
field is now of typestring
, not[]byte
.sse.Event
: TheName
field is now namedType
.
sse.Message
:Clone
now copies the topic of the message to the new value.sse.Message
: ID fields that contain NUL characters are now ignored, as required by the spec, inUnmarshalText
.
0.4.3 - 2023-07-08
- Messages longer than 4096 bytes are no longer being dropped (#2, thanks @aldld)
- Event parsing no longer panics on empty field with colon after name, see test case for example (#5)
0.4.2 - 2021-10-17
- Get the event name of a Message
0.4.1 - 2021-10-15
- Set a custom logger for Server
0.4.0 - 2021-10-15
- Server does not set any other headers besides
Content-Type
. - UpgradedRequest does not return a SendError anymore when Write errors.
- Providers don't handle callback errors anymore. Callbacks return a flag that indicates whether the provider should keep calling it for new messages instead.
- Client's default response validator now ignores
Content-Type
parameters when checking if the response's content type istext/event-stream
. - Various optimizations
0.3.0 - 2021-09-18
- ReplayProviderWithGC interface, which must be satisfied by replay providers that must be cleaned up periodically.
- Subscriptions now take a callback function instead of a channel.
- Server response headers are now sent on the first Send call, not when Upgrade is called.
- Providers are not required to add the default topic anymore. Callers of Subscribe should ensure at least a topic is specified.
- Providers' Subscribe method now blocks until the subscriber is removed.
- Server's Subscribe method automatically adds the default topic if no topic is specified.
- ReplayProvider does not require for GC to be implemented.
- Client connections take callback functions instead of channels as event listeners.
- Client connections' Unsubscribe methods are replaced by functions returned by their Subscribe counterparts.
- Fix replay providers not replaying the oldest message if the ID provided is of the one before that one.
- Fix replay providers hanging the caller's goroutine when a write error occurs using the default ServeHTTP implementation.
- Fix providers hanging when a write error occurs using the default ServeHTTP implementation.
0.2.0 - 2021-09-13
- Text/JSON marshalers and unmarshalers, and SQL scanners and valuers for the EventID type (previously event.ID).
- Check for http.NoBody before resetting the request body on client reconnect.
- Package structure. The module is now refactored into a single package with an idiomatic name. This has resulted in various name changes:
client.Error
-sse.ConnectionError
event.Event
-sse.Message
(previousserver.Message
is removed, see next change)event.ID
-sse.EventID
event.NewID
-sse.NewEventID
event.MustID
-sse.MustEventID
server.Connection
-sse.UpgradedRequest
server.NewConnection
-sse.Upgrade
server.ErrUnsupported
-sse.ErrUpgradeUnsupported
server.New
-sse.NewServer
.
event.Event
is merged withserver.Message
, becomingsse.Message
. This affects thesse.Server.Publish
function, which doesn't take atopic
parameter anymore.- The server's constructor doesn't take an
Provider
as a parameter. It instead takes multiple optionalServerOptions
. TheWithProvider
option is now used to pass custom providers to the server. - The
ReplayProvider
interface'sPut
method now takes a**Message
instead of a*Message
. This change also affects the replay providers in this package:ValidReplayProvider
andFiniteReplayProvider
. - The
Provider
interface'sPublish
method now takes a*Message
instead of aMessage
. This change also affectsJoe
, the provider in this package. - The
UpgradedRequest
'sSend
now method takes a*Message
as parameter.