AuroraLive supports event notifications, to let your application know when things happen asynchronously, outside of an API request cycle. For example, you may want to update something on your end when a live stream starts or is interrupted.
When these asynchronous events happen, we'll make a POST request to the address you give us and you can do whatever you need with it on your end.
Event Notify is configured in the console under " Live Event Notifications".
AuroraLive will submit a POST request to the configured URL, which your application can treat the same as any other route. Your event handler can do things like update the state of the specified asset in your database, or trigger other work.
The event notify has a retry mechanism, up to 10 retries will be made at an interval of 1 minute.
A retry is triggered in the following cases:
header:AuroraLive-Signature: t=1659685897&sign=bb42114db71d8aa1bd361bd6ea1eaae631465c2991afde559cc6a724871b3d28body:{"push_time":1659685897,"live_stream_id":"Zjc0NjU3YmRiOWUxYzQyYmU1MGUzNmIwNGExNDkzNmE","event_type":"push"}
header:AuroraLive-Signature: t=1659684548&sign=e8f0afc48ee03faf91a0ff3859c7bd67fc4868417bdc1620bfb19291ada930c1body:{"push_time":1659684531,"interruption_time":1659684548,"live_stream_id":"MDdjYzg5MmFhNTA0MjdkN2Q0YjY4ZmE1NDY3ODUzOTg","event_type":"interruption"}
You have the option to verify event notification requests that AuroraLive sends to your endpoints. AuroraLive will include a signature in the request's header. You can use this signature in your code to make sure the request was sent by AuroraLive and not a third party.
AuroraLive-Signature: t=1659608231&sign=1005d2c384a0f9739fae2532f446342dfc4406bd5dd0310b633c26c66fda152b
Event notifications contain a header called auroraLive-signature with the timestamp and a signature. The timestamp is prefixed by t=
and the signature is prefixed by sign=
. AuroraLive generates signatures using HMAC
with SHA2-256
.
1.Extract the timestamp and signature
Split the header at the & character and get the values for t (timestamp) and sign (the signature)
2.Prepare the signed_payload string
You will need:
&
3.Determine the expected signature
Use the 3 components from Step 2 to compute an HMAC with the SHA256 hash function. Depending on the language that you are using this will look something like the following:
signKey = 'Your Signing Key' // You custom a Signing Key When you configure event notify.payload = timestamp + "." + request_bodyexpected_signature = createHmacSha256(payload, signKey)
4.Compare signature
Compare the signature in the header to the expected signature.
Examples Golang
func IsValidSignature(ctx *gin.Context, signature string, body []byte) error {if signature == "" {return errors.New("no AuroraLive-Signature in request header")}signArr := strings.Split(signature, "&")if len(signArr) != 2 {return domain.NewCustomError(domain.ErrBadRequest, fmt.Sprintf("AuroraLive-Signature in request header should be two values long: %s", signArr))}timestampArr := strings.Split(signArr[0], "=")signInfoArr := strings.Split(signArr[1], "=")if len(timestampArr) != 2 || len(signArr) != 2 {return domain.NewCustomError(domain.ErrBadRequest, fmt.Sprintf("missing timestamp: %s or missing signature: %s", timestampArr, signArr))}timestamp := timestampArr[1]signInfo := signInfoArr[1]signKey := "cmbt2utc08s4urpnnutgcmbt2utc08s4urpnnuu0"payload := fmt.Sprintf("%s.%s", timestamp, string(body))sign := createHmacSha256(payload, signKey)log.GetLogger().Info(ctx, fmt.Sprintf("webhook %s, time: %s, signInfo %s ", string(body), timestamp, signInfo))if sign != signInfo {return domain.NewCustomError(domain.BadRequest, fmt.Sprintf("not a valid AuroraLive-Signature, sign= %s, sign-header= %s", sign, signature))}return nil}func createHmacSha256(payload, signKey string) string {h := hmac.New(sha256.New, []byte(signKey))h.Write([]byte(payload)) //nolint:errcheckreturn hex.EncodeToString(h.Sum(nil))}
Examples Nodejs
function createHmacSha256(payload, signKey) {const hmac = crypto.createHmac('sha256', signKey);hmac.update(payload);return hmac.digest('hex');}function isValidSignature(signature, body) {if (!signature) {throw new Error('no AuroraLive-Signature in request header');}const signArr = signature.split('&');if (signArr.length !== 2) {throw new Error(`AuroraLive-Signature in request header should be two values long: ${signArr}`);}const timestampArr = signArr[0].split('=');const signInfoArr = signArr[1].split('=');if (timestampArr.length !== 2 || signArr.length !== 2) {throw new Error(`missing timestamp: ${timestampArr} or missing signature: ${signArr}`);}const timestamp = timestampArr[1];const signInfo = signInfoArr[1];const signKey = 'cmbt2utc08s4urpnnutgcmbt2utc08s4urpnnuu0';const payload = `${timestamp}.${body}`;const sign = createHmacSha256(payload, signKey);console.log(`\nWebhook verification details:`);console.log(`Body: ${body}`);console.log(`Timestamp: ${timestamp}`);console.log(`Received signature: ${signInfo}`);console.log(`Calculated signature: ${sign}`);if (sign !== signInfo) {throw new Error(`not a valid AuroraLive-Signature, sign= ${sign}, sign-header= ${signature}`);}return true;}
created
: A new live stream has been created. Broadcasters with a stream_key
can start sending encoder feed to this live stream.connected
: An encoder has successfully connected to this live stream.recording
: Recording on this live stream has started. AuroraCloud Live has successfully processed the first frames from the encoder.recording_completed
: Recording on this live stream has ended. AuroraCloud Live has successfully completed processing the recording.active
: This live stream is now "active". The live streams playback_id
OR the playback_id
associated with this live stream's asset can be used right now to created HLS URLs (https://stream.visionular.com/{PLAYBACK_ID}.m3u8
and start streaming in your player. Note that before the live stream is "active"
, trying to stream the HLS URL will result in HTTP 412
errors.disconnected
: An encoder has disconnected from this live stream. Note that while disconnected the live stream is still status: "active"
.idle
: The reconnect_window
for this live stream has elapsed. The live stream status
will now transition to "idle"
.updated
: This live stream has been updated. For example, after resetting the live stream's stream key API.enabled
: This live stream has been enabled. This event fires after enable live stream API.disabled
: This live stream has been disabled. This event fires after disable live stream API. Disabled live streams will no longer accept new RTMP connections.deleted
: This live stream has been deleted. This event fires after delete live stream API.These simulcast target events are useful when creating a UI that shows your users the status of their configured 3rd party endpoints. These events are handy when you want to build a UI that shows the state of each simulcast target and keep track of the state changes as they happen.
simulcast_target.created
: A new simulcast target has been created for this live stream.simulcast_target.idle
: When the parent live stream is "disconnected"
, all simulcast targets will have be "idle"
.simulcast_target.active
: When the parent live stream fires "connected"
then the simulcast targets transition to "active"
.simulcast_target.deleted
: This simulcast target has been deleted.{"type": "disabled","id": "adc1d50f-9a25-f032-0be8-f2280f243967","data": {"stream_key": "b28edd57-2091-ca2f-7ec4-dd0a43b19c34","status": "disabled","reconnect_window": 60,..."max_continuous_duration": 43200,"latency_mode": "standard","id": "3dDAFFK9vJQdVZMqDBIOh8PChlcNt02fEWM02XicGSXjs","created_at": 1699950576},"created_at": "2023-11-20T06:39:01.047000Z"}