大多数浏览器和
Developer App 均支持流媒体播放。
-
探索 AVFoundation 中的媒体性能指标
探索如何利用新的媒体性能 API 来监测、分析并改进用户体验。了解如何使用不同的 AVMetricEvents 来监测音视频播放器在 HLS 素材方面的性能,并学习如何利用这些指标来理解播放器性能问题,以及进行相关分类。
章节
- 0:00 - Introduction
- 1:33 - What are events?
- 5:36 - Subscribing to events
资源
-
下载
Hello. My name is Nishant, and I'm an AVFoundation HLS engineer. I'm excited to share what we have been working on. Today, we will look at a new way to gather media performance metrics. In iOS 18, these metrics are represented as events. We will explore events, the different types, and later we will learn how your app can subscribe to these events. Before we do that, lets look at some common issues with media playback.
The first one being media playback taking too long to start and the other one is the playback stalling in between.
Triaging such issues in the field is hard but if we have access to client-side metrics that makes it a lot easier.
Apart from that, client-side metrics provides insights into the workings of AVFoundation APIs.
This is useful to understand and improve performance.
Previously, for media playback, you could obtain some of the client-side metrics via access logs, error logs and various AVPlayer notifications.
In iOS 18, for HLS Streaming assets, we are enhancing the metrics which are available and providing more data points.
Now, let's explore some of these new metrics by looking at the playback session with media playback issues. Here we have a timeline for video playback of duration 15 min. By zooming into the first two seconds, we can see what happened during the startup.
To begin, AVPlayer would start fetching the multi-variant playlist. This would appear as an event on the timeline.
Later AVPlayer would fetch the video, the audio and subtitle playlist. This would similarly appear as events on the timeline. With the media playlist, AVPlayer would fetch the media segments along with any required content keys. When the AVPlayer has buffered enough, it has reached a stage where it can resume playback. This is represented by the "likely to keep up" event on the timeline. This event provides the startup time as well as details about things which affected startup such as related playlist, media segment and content key events.
Here we see that startup took almost 2 seconds.
Since the startup time is high, let's look at the preceding events.
We see that the majority of time at startup was spent waiting for the content key request.
We either need to improve the key server performance or look into preloading keys ahead of playback on the client-side. This would help improve startup time. With client-side metrics, you now have insights into what is going on within AVFoundation playback session.
Moving on, let's take a look at the other issue where media playback stalls in between.
Zooming into the timeline between minutes 7 and 8 we see that a stall occurred.
This is represented as event on the timeline.
Let's take a look at events just before the stall.
We see that AVPlayer is fetching media segments. Looking more closely, here we see that the video media segment seems to return HTTP 404 error.
It explains the variant switch from 20Mbps to 15Mbps.
After the variant switch, the video media segment in the new variant again returns HTTP 404 error. AVPlayer at this point seems to run out of buffer resulting in the stall.
With access to the media segment event, we can now get a reference to NSURLSessionTask metrics. This can provide additional details on why the request failed such as the complete set of response headers and other network transaction details. We could use such information and session identifier to help narrow down the issue when working with our CDN partners. In this fashion, you can use client-side metrics to triage issues in the field.
Lastly, if you would like to monitor sessions at scale, AVPlayer also creates a summary event at the end.
This event provides different key performance indicators such as stall count, switch count for the session at glance. You can use the KPIs to keep track of media playback health at scale.
To recap, we looked at playlist, media segment and content key events. They provide information about the individual resource requests. Then we looked at stall and variant switch events which provide information about media playback stalls and variant switches. Finally, we have the summary event with KPIs at the end. Apart from these, AVPlayerItem also generates rate change events whenever playback rate changes, seek events when user seeks and error events when an error occurs. Now, lets explore how your app can subscribe to receive these events. In iOS 18, we are introducing a new AVFoundation API called AVMetrics. It provides a unified way to gather metrics from various AVFoundation interfaces. Your app would receive events as they happen on the media timeline and they are generated at the end of a particular activity such as media playlist fetch.
It follows the publisher/subscriber model which allows you to subscribe to only those events you are interested in. Now, let's look at some of the core concepts.
Publishers are the AVFoundation interfaces which support publishing events and they conform to AVMetricEventStreamPublisher protocol.
This protocol provides support for publishing events of a particular metric event type or all the metrics.
In iOS 18, AVPlayerItem conforms to AVMetricEventStreamPublisher protocol. which means it can publish events. Now, lets take a look at an example. Here in this example, we are interested in "likely to keep up" and "summary events" and then later report them to our backend analytics server. First we grab the AVMetrics async sequence stream for "likely to keep up" and "summary events." Then we would create chronologically merged stream. This merged stream would deliver both "likely to keep up" and "summary events" in a chronological fashion efficiently. On receiving events, we can serialize them and send them to our backend analytics server.
If you're a Objective-C developer, the AVMetrics API is a bit different. Here you would create AVMetricEventStream instance and set your subscriber.
Later you subscribe to events you're interested in.
Then, you would add the AVPlayerItem.
At this point, your subscriber will start receiving those events.
You serialize those events to be sent to your backend server.
With this, we now know how your app can subscribe and report client-side metrics to your back end analytics servers. To wrap up, we explored client-side metrics for AVPlayer, the various different event types and then we looked at how your app can subscribe to receive these metrics.
We are excited for you to adopt the new APIs in your app. We hope with the enriched client-side analytics, it would help triage issues in the field and improve media playback performance for your users. Thank you.
-
-
6:27 - AVMetric Publishers
public protocol AVMetricEventStreamPublisher { func metrics<MetricType: AVMetricEvent>(forType metricType: MetricType.Type) -> AVMetrics<MetricType> func allMetrics() -> AVMetrics<AVMetricEvent> } extension AVPlayerItem : AVMetricEventStreamPublisher
-
6:50 - Example showing how to obtain likely to keep up and summary metrics from AVPlayerItem - Swift
let playerItem : AVPlayerItem = ... let ltkuMetrics = item.metrics(forType: AVMetricPlayerItemLikelyToKeepUpEvent.self) let summaryMetrics = item.metrics(forType: AVMetricPlayerItemPlaybackSummaryEvent.self) for await (metricEvent, publisher) in ltkuMetrics.chronologicalMerge(with: summaryMetrics) { // send metricEvent to server }
-
7:26 - Example showing how to obtain likely to keep up and summary metrics from AVPlayerItem - Objective-C
AVPlayerItem *item = ... AVMetricEventStream *eventStream = [AVMetricEventStream eventStream]; id<AVMetricEventStreamSubscriber> subscriber = [[MyMetricSubscriber alloc] init]; [eventStream setSubscriber:subscriber queue:mySerialQueue] [eventStream subscribeToMetricEvent:[AVMetricPlayerItemLikelyToKeepUpEvent class]]; [eventStream subscribeToMetricEvent:[AVMetricPlayerItemPlaybackSummaryEvent class]]; [eventStream addPublisher:item];
-
-
正在查找特定内容?在上方输入一个主题,就能直接跳转到相应的精彩内容。
提交你查询的内容时出现错误。请检查互联网连接,然后再试一次。