
Live Streaming from Unreal Engine to Amazon IVS
Explore the integration between Unreal Engine and Amazon Interactive Video Service (IVS), demonstrating how you can stream live content from Unreal Engine directly to Amazon IVS.
Use Case | Description |
---|---|
Live Streaming of Gameplay | Engage the gamer’s community around their skills and experiences by enabling them to live stream their gameplay. |
Real-time Sharing of Game Development Processes | Game development companies can share behind-the-scenes game development processes in real-time, allowing their audience to witness it firsthand. |
Exclusive Sneak Peeks of Upcoming Titles | Game studios can build excitement and anticipation among their fan base by offering exclusive sneak peaks of upcoming gaming titles |
Live Streaming Unreal Engine Tutorials and Workshops | Allows Unreal Engine experts to teach a mass audience how to use the software to create gaming or other content. This provides an interactive and engaging learning experience for the audience. |
- Unreal Engine. You will need to have Unreal Engine version 5.4.3 or later installed.
- AWS Account. We will be leveraging the power of Amazon Web Services (AWS) to create the backend resources needed to live stream from Unreal Engine to Amazon IVS.
- Part 1 - Create an Unreal Editor project and Install Pixel Streaming Plugin
- Part 2 - Create an IVS Stage on AWS
- Part 3 - Write code to forward the media stream from Unreal Editor to IVS
- Part 4 - Test it out
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
//go:build !js
// +build !js
package main
import (
"bytes"
"encoding/json"
"flag"
"fmt"
"io/ioutil"
"net/http"
"time"
"github.com/pion/webrtc/v3"
"golang.org/x/net/websocket"
)
type websocketMessage struct {
Type string `json:"type"`
PeerConnectionOptions webrtc.Configuration `json:"peerConnectionOptions"`
Count int `json:"count"`
SDP string `json:"sdp"`
Candidate webrtc.ICECandidateInit `json:"candidate"`
IDs []string `json:"ids"`
StreamerId string `json:"streamerId"`
}
func main() {
url := flag.String("url", "ws://localhost/", "URL to UE5 Pixel Streaming WebSocket endpoint")
origin := flag.String("origin", "http://localhost", "Origin that is passed in HTTP header")
bearerToken := flag.String("token", "", "IVS Bearer Token")
flag.Parse()
if *bearerToken == "" {
panic("Bearer Token must not be empty")
}
conn, err := websocket.Dial(*url, "", *origin)
if err != nil {
panic(err)
}
defer func() {
if err = conn.Close(); err != nil {
panic(err)
}
}()
if err = websocket.JSON.Send(conn, websocketMessage{Type: "listStreamers"}); err != nil {
panic(err)
}
peerConnection := &webrtc.PeerConnection{}
peerConnectionConfig := webrtc.Configuration{}
data := []byte{}
jsonMessage := websocketMessage{}
for {
if err = websocket.Message.Receive(conn, &data); err != nil {
panic(err)
} else if err = json.Unmarshal(data, &jsonMessage); err != nil {
panic(err)
}
switch jsonMessage.Type {
case "config":
peerConnectionConfig = jsonMessage.PeerConnectionOptions
case "offer":
audioTrack, videoTrack := createStagesSession(*bearerToken)
peerConnection = createUnrealPeerConnection(conn, peerConnectionConfig, audioTrack, videoTrack)
if err = peerConnection.SetRemoteDescription(webrtc.SessionDescription{Type: webrtc.SDPTypeOffer, SDP: jsonMessage.SDP}); err != nil {
panic(err)
}
answer, err := peerConnection.CreateAnswer(nil)
if err != nil {
panic(err)
}
if err = peerConnection.SetLocalDescription(answer); err != nil {
panic(err)
}
if err = websocket.JSON.Send(conn, answer); err != nil {
panic(err)
}
case "iceCandidate":
if err = peerConnection.AddICECandidate(jsonMessage.Candidate); err != nil {
panic(err)
}
case "playerCount":
fmt.Println("Player Count", jsonMessage.Count)
case "streamerList":
if len(jsonMessage.IDs) >= 1 {
if err = websocket.JSON.Send(conn, websocketMessage{Type: "subscribe", StreamerId: jsonMessage.IDs[0]}); err != nil {
panic(err)
}
}
default:
fmt.Println("Unhandled type", jsonMessage.Type)
}
}
}
func createUnrealPeerConnection(conn *websocket.Conn, configuration webrtc.Configuration, stagesAudioTrack, stagesVideoTrack *webrtc.TrackLocalStaticRTP) *webrtc.PeerConnection {
m := &webrtc.MediaEngine{}
if err := m.RegisterCodec(webrtc.RTPCodecParameters{
RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeH264, ClockRate: 90000, Channels: 0, SDPFmtpLine: "level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42e034", RTCPFeedback: nil},
PayloadType: 96,
}, webrtc.RTPCodecTypeVideo); err != nil {
panic(err)
} else if err := m.RegisterCodec(webrtc.RTPCodecParameters{
RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeOpus, ClockRate: 48000, Channels: 2, SDPFmtpLine: "minptime=10;useinbandfec=1", RTCPFeedback: nil},
PayloadType: 111,
}, webrtc.RTPCodecTypeAudio); err != nil {
panic(err)
}
peerConnection, err := webrtc.NewAPI(webrtc.WithMediaEngine(m)).NewPeerConnection(configuration)
if err != nil {
panic(err)
}
if _, err := peerConnection.AddTransceiverFromKind(webrtc.RTPCodecTypeAudio); err != nil {
panic(err)
} else if _, err := peerConnection.AddTransceiverFromKind(webrtc.RTPCodecTypeVideo); err != nil {
panic(err)
} else if _, err = peerConnection.CreateDataChannel("cirrus", nil); err != nil {
panic(err)
}
peerConnection.OnICEConnectionStateChange(func(connectionState webrtc.ICEConnectionState) {
fmt.Printf("Unreal Connection State has changed %s \n", connectionState.String())
})
peerConnection.OnICECandidate(func(c *webrtc.ICECandidate) {
if c == nil {
return
}
if err = websocket.JSON.Send(conn, &websocketMessage{Type: "iceCandidate", Candidate: c.ToJSON()}); err != nil {
panic(err)
}
})
peerConnection.OnTrack(func(t *webrtc.TrackRemote, _ *webrtc.RTPReceiver) {
fmt.Printf("Track has started, of type %d: %s \n", t.PayloadType(), t.Codec().RTPCodecCapability.MimeType)
buf := make([]byte, 1500)
for {
n, _, err := t.Read(buf)
if err != nil {
panic(err)
}
if t.Kind() == webrtc.RTPCodecTypeAudio {
if _, err := stagesAudioTrack.Write(buf[:n]); err != nil {
panic(err)
}
} else {
if _, err := stagesVideoTrack.Write(buf[:n]); err != nil {
panic(err)
}
}
}
})
go func() {
for range time.NewTicker(20 * time.Second).C {
if err = websocket.JSON.Send(conn, &websocketMessage{Type: "keepalive"}); err != nil {
panic(err)
}
}
}()
return peerConnection
}
func createStagesSession(bearerToken string) (*webrtc.TrackLocalStaticRTP, *webrtc.TrackLocalStaticRTP) {
addToken := func(req *http.Request) {
req.Header.Add("Authorization", "Bearer "+bearerToken)
}
peerConnection, err := webrtc.NewPeerConnection(webrtc.Configuration{})
if err != nil {
panic(err)
}
peerConnection.OnICEConnectionStateChange(func(connectionState webrtc.ICEConnectionState) {
fmt.Printf("Stages Connection State has changed %s \n", connectionState.String())
})
videoTrack, err := webrtc.NewTrackLocalStaticRTP(webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeH264}, "video", "pion")
if err != nil {
panic(err)
}
audioTrack, err := webrtc.NewTrackLocalStaticRTP(webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeOpus}, "audio", "pion")
if err != nil {
panic(err)
}
if _, err = peerConnection.AddTransceiverFromTrack(audioTrack, webrtc.RTPTransceiverInit{Direction: webrtc.RTPTransceiverDirectionSendonly}); err != nil {
panic(err)
} else if _, err = peerConnection.AddTransceiverFromTrack(videoTrack, webrtc.RTPTransceiverInit{Direction: webrtc.RTPTransceiverDirectionSendonly}); err != nil {
panic(err)
}
offer, err := peerConnection.CreateOffer(nil)
if err != nil {
panic(err)
}
if err := peerConnection.SetLocalDescription(offer); err != nil {
panic(err)
}
req, err := http.NewRequest("POST", "https://global.whip.live-video.net", bytes.NewBuffer([]byte(offer.SDP)))
if err != nil {
panic(err)
}
addToken(req)
req.Header.Add("Content-Type", "application/sdp")
client := &http.Client{
CheckRedirect: func(req *http.Request, via []*http.Request) error {
addToken(req)
return nil
},
}
resp, err := client.Do(req)
if err != nil {
panic(err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusCreated {
panic(fmt.Sprintf("POST failed with error: %s", resp.Status))
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
panic(err)
}
if err = peerConnection.SetRemoteDescription(webrtc.SessionDescription{Type: webrtc.SDPTypeAnswer, SDP: string(body)}); err != nil {
panic(err)
}
return audioTrack, videoTrack
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
//go:build !js
// +build !js
package main
import (
"bytes"
"encoding/json"
"flag"
"fmt"
"io/ioutil"
"net/http"
"time"
"github.com/pion/webrtc/v3"
"golang.org/x/net/websocket"
)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//go:build !js
// +build !js
package main
import (
"bytes"
"encoding/json"
"flag"
"fmt"
"io/ioutil"
"net/http"
"time"
"github.com/pion/webrtc/v3"
"golang.org/x/net/websocket"
)
1
2
3
4
5
6
7
8
9
type websocketMessage struct {
Type string `json:"type"`
PeerConnectionOptions webrtc.Configuration `json:"peerConnectionOptions"`
Count int `json:"count"`
SDP string `json:"sdp"`
Candidate webrtc.ICECandidateInit `json:"candidate"`
IDs []string `json:"ids"`
StreamerId string `json:"streamerId"`
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
func main() {
url := flag.String("url", "ws://localhost/", "URL to UE5 Pixel Streaming WebSocket endpoint")
origin := flag.String("origin", "http://localhost", "Origin that is passed in HTTP header")
bearerToken := flag.String("token", "", "IVS Bearer Token")
flag.Parse()
if *bearerToken == "" {
panic("Bearer Token must not be empty")
}
conn, err := websocket.Dial(*url, "", *origin)
if err != nil {
panic(err)
}
defer func() {
if err = conn.Close(); err != nil {
panic(err)
}
}()
if err = websocket.JSON.Send(conn, websocketMessage{Type: "listStreamers"}); err != nil {
panic(err)
}
peerConnection := &webrtc.PeerConnection{}
peerConnectionConfig := webrtc.Configuration{}
data := []byte{}
jsonMessage := websocketMessage{}
for {
if err = websocket.Message.Receive(conn, &data); err != nil {
panic(err)
} else if err = json.Unmarshal(data, &jsonMessage); err != nil {
panic(err)
}
switch jsonMessage.Type {
case "config":
peerConnectionConfig = jsonMessage.PeerConnectionOptions
case "offer":
audioTrack, videoTrack := createStagesSession(*bearerToken)
peerConnection = createUnrealPeerConnection(conn, peerConnectionConfig, audioTrack, videoTrack)
if err = peerConnection.SetRemoteDescription(webrtc.SessionDescription{Type: webrtc.SDPTypeOffer, SDP: jsonMessage.SDP}); err != nil {
panic(err)
}
answer, err := peerConnection.CreateAnswer(nil)
if err != nil {
panic(err)
}
if err = peerConnection.SetLocalDescription(answer); err != nil {
panic(err)
}
if err = websocket.JSON.Send(conn, answer); err != nil {
panic(err)
}
case "iceCandidate":
if err = peerConnection.AddICECandidate(jsonMessage.Candidate); err != nil {
panic(err)
}
case "playerCount":
fmt.Println("Player Count", jsonMessage.Count)
case "streamerList":
if len(jsonMessage.IDs) >= 1 {
if err = websocket.JSON.Send(conn, websocketMessage{Type: "subscribe", StreamerId: jsonMessage.IDs[0]}); err != nil {
panic(err)
}
}
default:
fmt.Println("Unhandled type", jsonMessage.Type)
}
}
}
- Command-line flag parsing: First we define command line flags for the URL we want to connect to, an origin URL for the request that is passed in the HTTP header, and a participant token. The URL we want to connect to will default to http://localhost since our signaling server is running locally and is listening on port 80 for websocket connections. The origin will be http://localhost as well since we will be running this Go program locally. Finally, we also specify a participant token to authenticate with the IVS stage we just created. This can be thought of as the password that enables us to send audio and video from Unreal Engine to IVS.
- Websocket connection: Next, we establish a websocket connection to the URL specified from above. We also set up a deferred function to close the connection when we exit the program.
- Initial communication: After connecting to the websocket URL, we set up a for loop that continually parses incoming JSON messages from the websocket connection. The different type of messages we handle include
config
: Updates peer connection with configuration informationoffer
: Also known as a WebRTC offer, we create a session with our IVS stage (more on this later), set up a peer connection with the signaling server from the Pixel Streaming Plugin (more on this later), and send an answer. This offer and answer process allows the our Go program and the signaling server to negotiate and agree on the parameters of the connection.iceCandidate
: We use this to add ICE candidates to the peer connection. ICE stands for Interactive Connectivity Establishment. ICE candidates are network addresses that represent potential connection points for a device. Without ICE candidates, we wouldn’t be able to find the best connection point between our Go program and the signaling server. ICE candidates are also used to overcome network address translation (NAT) and firewall traversal issues that can prevent direct connections between peers. Here, both our Go program and our signaling server are running locally, so that won’t be an issue here.playerCount
: Prints the number of connected players or the number of participants that have joined the stagestreamerList
: We use this JSON message to subscribe to the first available streamer. In this case our “streamer” is the media stream provided by the Pixel Streaming Plugin running in Unreal Engine.
createStagesSession
. A stage session just represents a period of activity when a participant joins a stage. In this case, our Go program is the participant joining the stage.1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
func createStagesSession(bearerToken string) (*webrtc.TrackLocalStaticRTP, *webrtc.TrackLocalStaticRTP) {
addToken := func(req *http.Request) {
req.Header.Add("Authorization", "Bearer "+bearerToken)
}
peerConnection, err := webrtc.NewPeerConnection(webrtc.Configuration{})
if err != nil {
panic(err)
}
peerConnection.OnICEConnectionStateChange(func(connectionState webrtc.ICEConnectionState) {
fmt.Printf("Stages Connection State has changed %s \n", connectionState.String())
})
videoTrack, err := webrtc.NewTrackLocalStaticRTP(webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeH264}, "video", "pion")
if err != nil {
panic(err)
}
audioTrack, err := webrtc.NewTrackLocalStaticRTP(webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeOpus}, "audio", "pion")
if err != nil {
panic(err)
}
if _, err = peerConnection.AddTransceiverFromTrack(audioTrack, webrtc.RTPTransceiverInit{Direction: webrtc.RTPTransceiverDirectionSendonly}); err != nil {
panic(err)
} else if _, err = peerConnection.AddTransceiverFromTrack(videoTrack, webrtc.RTPTransceiverInit{Direction: webrtc.RTPTransceiverDirectionSendonly}); err != nil {
panic(err)
}
offer, err := peerConnection.CreateOffer(nil)
if err != nil {
panic(err)
}
if err := peerConnection.SetLocalDescription(offer); err != nil {
panic(err)
}
req, err := http.NewRequest("POST", "https://global.whip.live-video.net", bytes.NewBuffer([]byte(offer.SDP)))
if err != nil {
panic(err)
}
addToken(req)
req.Header.Add("Content-Type", "application/sdp")
client := &http.Client{
CheckRedirect: func(req *http.Request, via []*http.Request) error {
addToken(req)
return nil
},
}
resp, err := client.Do(req)
if err != nil {
panic(err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusCreated {
panic(fmt.Sprintf("POST failed with error: %s", resp.Status))
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
panic(err)
}
if err = peerConnection.SetRemoteDescription(webrtc.SessionDescription{Type: webrtc.SDPTypeAnswer, SDP: string(body)}); err != nil {
panic(err)
}
return audioTrack, videoTrack
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
func createUnrealPeerConnection(conn *websocket.Conn, configuration webrtc.Configuration, stagesAudioTrack, stagesVideoTrack *webrtc.TrackLocalStaticRTP) *webrtc.PeerConnection {
m := &webrtc.MediaEngine{}
if err := m.RegisterCodec(webrtc.RTPCodecParameters{
RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeH264, ClockRate: 90000, Channels: 0, SDPFmtpLine: "level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42e034", RTCPFeedback: nil},
PayloadType: 96,
}, webrtc.RTPCodecTypeVideo); err != nil {
panic(err)
} else if err := m.RegisterCodec(webrtc.RTPCodecParameters{
RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypeOpus, ClockRate: 48000, Channels: 2, SDPFmtpLine: "minptime=10;useinbandfec=1", RTCPFeedback: nil},
PayloadType: 111,
}, webrtc.RTPCodecTypeAudio); err != nil {
panic(err)
}
peerConnection, err := webrtc.NewAPI(webrtc.WithMediaEngine(m)).NewPeerConnection(configuration)
if err != nil {
panic(err)
}
if _, err := peerConnection.AddTransceiverFromKind(webrtc.RTPCodecTypeAudio); err != nil {
panic(err)
} else if _, err := peerConnection.AddTransceiverFromKind(webrtc.RTPCodecTypeVideo); err != nil {
panic(err)
} else if _, err = peerConnection.CreateDataChannel("cirrus", nil); err != nil {
panic(err)
}
peerConnection.OnICEConnectionStateChange(func(connectionState webrtc.ICEConnectionState) {
fmt.Printf("Unreal Connection State has changed %s \n", connectionState.String())
})
peerConnection.OnICECandidate(func(c *webrtc.ICECandidate) {
if c == nil {
return
}
if err = websocket.JSON.Send(conn, &websocketMessage{Type: "iceCandidate", Candidate: c.ToJSON()}); err != nil {
panic(err)
}
})
peerConnection.OnTrack(func(t *webrtc.TrackRemote, _ *webrtc.RTPReceiver) {
fmt.Printf("Track has started, of type %d: %s \n", t.PayloadType(), t.Codec().RTPCodecCapability.MimeType)
buf := make([]byte, 1500)
for {
n, _, err := t.Read(buf)
if err != nil {
panic(err)
}
if t.Kind() == webrtc.RTPCodecTypeAudio {
if _, err := stagesAudioTrack.Write(buf[:n]); err != nil {
panic(err)
}
} else {
if _, err := stagesVideoTrack.Write(buf[:n]); err != nil {
panic(err)
}
}
}
})
go func() {
for range time.NewTicker(20 * time.Second).C {
if err = websocket.JSON.Send(conn, &websocketMessage{Type: "keepalive"}); err != nil {
panic(err)
}
}
}()
return peerConnection
}
YOUR_PARTICIPANT_TOKEN_HERE
with the participant token you created earlier in the IVS console> go run ./main.go -token YOUR_PARTICIPANT_TOKEN_HERE
1
2
3
4
5
6
Stages Connection State has changed checking
Unreal Connection State has changed checking
Stages Connection State has changed connected
Unreal Connection State has changed connected
Track has started, of type 111: audio/opus
Track has started, of type 96: video/H264
Any opinions in this post are those of the individual author and may not reflect the opinions of AWS.