Analytics

ExoPlayer는 다양한 재생 분석 요구사항을 지원합니다. 궁극적으로 분석은 재생에서 데이터를 수집, 해석, 집계, 요약하는 것입니다. 이 데이터는 로깅, 디버깅 또는 향후 재생 결정을 알리는 등의 목적으로 기기에서 사용되거나 모든 기기에서 재생을 모니터링하기 위해 서버에 보고될 수 있습니다.

일반적으로 분석 시스템은 이벤트를 먼저 수집한 후 추가로 처리하여 유의미한 이벤트를 만들어야 합니다.

  • 이벤트 수집: ExoPlayer 인스턴스에 AnalyticsListener를 등록하면 됩니다. 등록된 분석 리스너는 플레이어 사용 중에 이벤트가 발생할 때 이벤트를 수신합니다. 각 이벤트는 재생목록의 해당 미디어 항목과 재생 위치 및 타임스탬프 메타데이터와 연결됩니다.
  • 이벤트 처리: 일부 애널리틱스 시스템은 모든 이벤트 처리가 서버 측에서 실행되는 방식으로 서버에 원시 이벤트를 업로드합니다. 또한 기기에서 이벤트를 처리할 수도 있으며, 이렇게 하면 더 간단하거나 업로드해야 하는 정보의 양을 줄일 수 있습니다. ExoPlayer는 다음 처리 단계를 실행할 수 있는 PlaybackStatsListener를 제공합니다.
    1. 이벤트 해석: 분석 목적으로 유용하려면 이벤트를 단일 재생 맥락에서 해석해야 합니다. 예를 들어 플레이어 상태가 STATE_BUFFERING로 변경되는 원시 이벤트는 초기 버퍼링, 재버퍼링 또는 탐색 후 발생하는 버퍼링에 해당할 수 있습니다.
    2. 상태 추적: 이 단계는 이벤트를 카운터로 변환합니다. 예를 들어 상태 변경 이벤트는 각 재생 상태에서 소비된 시간을 추적하는 카운터로 변환할 수 있습니다. 결과적으로 단일 재생에 대한 기본적인 분석 데이터 값의 집합이 생성됩니다.
    3. 집계: 이 단계에서는 일반적으로 카운터를 더하여 여러 재생의 분석 데이터를 결합합니다.
    4. 요약 측정항목 계산: 가장 유용한 측정항목은 평균을 계산하거나 다른 방법으로 기본 분석 데이터 값을 결합하는 측정항목입니다. 요약 측정항목은 단일 재생 또는 여러 재생에 대해 계산할 수 있습니다.

AnalyticsListener를 사용한 이벤트 수집

플레이어의 원시 재생 이벤트는 AnalyticsListener 구현에 보고됩니다. 자체 리스너를 쉽게 추가하고 관심 있는 메서드만 재정의할 수 있습니다.

Kotlin

exoPlayer.addAnalyticsListener(
  object : AnalyticsListener {
    override fun onPlaybackStateChanged(
      eventTime: EventTime, @Player.State state: Int
    ) {}

    override fun onDroppedVideoFrames(
      eventTime: EventTime,
      droppedFrames: Int,
      elapsedMs: Long,
    ) {}
  }
)

자바

exoPlayer.addAnalyticsListener(
    new AnalyticsListener() {
      @Override
      public void onPlaybackStateChanged(
          EventTime eventTime, @Player.State int state) {}

      @Override
      public void onDroppedVideoFrames(
          EventTime eventTime, int droppedFrames, long elapsedMs) {}
    });

각 콜백에 전달되는 EventTime는 재생 위치 및 타임스탬프 메타데이터와 함께 이벤트를 재생목록의 미디어 항목에 연결합니다.

  • realtimeMs: 이벤트의 실제 시간입니다.
  • timeline, windowIndex, mediaPeriodId: 이벤트가 속한 재생목록과 재생목록 내 항목을 정의합니다. mediaPeriodId에는 이벤트가 상품 내 광고에 속하는지 여부를 나타내는 등 선택적 추가 정보가 포함됩니다.
  • eventPlaybackPositionMs: 이벤트가 발생한 항목의 재생 위치입니다.
  • currentTimeline, currentWindowIndex, currentMediaPeriodId, currentPlaybackPositionMs: 위와 같지만 현재 재생 중인 항목에 적용됩니다. 현재 재생 중인 항목은 이벤트가 속한 항목과 다를 수 있습니다. 예를 들어 이벤트가 재생할 다음 항목의 미리 버퍼링에 해당하는 경우입니다.

PlaybackStatsListener를 사용한 이벤트 처리

PlaybackStatsListener는 기기 내 이벤트 처리를 구현하는 AnalyticsListener입니다. 다음을 포함한 카운터 및 파생 측정항목으로 PlaybackStats를 계산합니다.

  • 요약 측정항목(예: 총 재생 시간)
  • 적응형 재생 품질 측정항목(예: 평균 동영상 해상도)
  • 렌더링 품질 측정항목(예: 프레임 누락 비율)
  • 리소스 사용량 측정항목(예: 네트워크를 통해 읽은 바이트 수)

사용 가능한 개수 및 파생 측정항목의 전체 목록은 PlaybackStats Javadoc에서 확인할 수 있습니다.

PlaybackStatsListener는 재생목록의 각 미디어 항목과 이러한 항목 내에 삽입된 각 클라이언트 측 광고에 대해 별도의 PlaybackStats를 계산합니다. PlaybackStatsListener에 콜백을 제공하여 완료된 재생에 관해 알리고 콜백에 전달된 EventTime를 사용하여 완료된 재생을 식별할 수 있습니다. 여러 재생의 분석 데이터를 집계할 수 있습니다. 언제든지 PlaybackStatsListener.getPlaybackStats()를 사용하여 현재 재생 세션의 PlaybackStats를 쿼리할 수도 있습니다.

Kotlin

exoPlayer.addAnalyticsListener(
  PlaybackStatsListener(/* keepHistory= */ true) {
    eventTime: EventTime?,
    playbackStats: PlaybackStats?,
    -> // Analytics data for the session started at `eventTime` is ready.
  }
)

자바

exoPlayer.addAnalyticsListener(
    new PlaybackStatsListener(
        /* keepHistory= */ true,
        (eventTime, playbackStats) -> {
          // Analytics data for the session started at `eventTime` is ready.
        }));

PlaybackStatsListener의 생성자는 처리된 이벤트의 전체 기록을 유지하는 옵션을 제공합니다. 재생 시간과 이벤트 수에 따라 알 수 없는 메모리 오버헤드가 발생할 수 있습니다. 따라서 최종 분석 데이터가 아닌 처리된 이벤트의 전체 기록에 액세스해야 하는 경우에만 이 옵션을 사용 설정해야 합니다.

PlaybackStats는 확장된 상태 집합을 사용하여 미디어 상태뿐만 아니라 재생 의도와 재생이 중단되거나 종료된 이유와 같은 더 자세한 정보를 나타냅니다.

재생 상태 사용자의 플레이 의도 재생 의도 없음
재생 전 JOINING_FOREGROUND JOINING_BACKGROUND NOT_STARTED
재생 중 PLAYING
재생 중단됨 SEEKING BUFFERING PAUSED, PAUSED_BUFFERING, SUPPRESSED, SUPPRESSED_BUFFERING, INTERRUPTED_BY_AD
종료 상태 ENDED, STOPPED, FAILED, ABANDONED

사용자가 재생을 계속하기 위해 적극적으로 대기한 시간과 수동적 대기 시간을 구분하는 데 사용자의 재생 의도가 중요합니다. 예를 들어 PlaybackStats.getTotalWaitTimeMsJOINING_FOREGROUND, BUFFERING, SEEKING 상태에서 소비된 총 시간을 반환하지만 재생이 일시중지된 시간은 반환하지 않습니다. 마찬가지로 PlaybackStats.getTotalPlayAndWaitTimeMs는 사용자가 재생할 의도가 있는 총 시간을 반환합니다. 즉, 총 활성 대기 시간과 PLAYING 상태에서 소비된 총 시간입니다.

처리되고 해석된 이벤트

keepHistory=true와 함께 PlaybackStatsListener를 사용하여 처리되고 해석된 이벤트를 기록할 수 있습니다. 결과 PlaybackStats에는 다음과 같은 이벤트 목록이 포함됩니다.

  • playbackStateHistory: 확장 재생 상태의 순서가 지정된 목록으로, 적용이 시작된 EventTime가 포함됩니다. PlaybackStats.getPlaybackStateAtTime를 사용하여 지정된 실제 시간의 상태를 조회할 수도 있습니다.
  • mediaTimeHistory: 실제 시간 및 미디어 시간 쌍의 기록으로, 이를 통해 어느 시간에 미디어의 어떤 부분이 재생되었는지 재구성할 수 있습니다. PlaybackStats.getMediaTimeMsAtRealtimeMs을 사용하여 지정된 실제 시간의 재생 위치를 조회할 수도 있습니다.
  • videoFormatHistoryaudioFormatHistory: 재생 중에 사용된 동영상 및 오디오 형식의 정렬된 목록으로, EventTime와 함께 사용이 시작된 시점이 포함됩니다.
  • fatalErrorHistorynonFatalErrorHistory: 치명적 오류 및 치명적이지 않은 오류의 순서가 지정된 목록으로, 발생 위치가 EventTime입니다. 치명적인 오류는 재생을 종료한 오류이고 심각하지 않은 오류는 복구할 수 있었던 오류입니다.

단일 재생 분석 데이터

이 데이터는 PlaybackStatsListener를 사용하면 keepHistory=false와 함께 자동으로 수집됩니다. 최종 값은 PlaybackStats Javadoc에서 찾을 수 있는 공개 필드와 getPlaybackStateDurationMs에서 반환한 재생 상태 시간입니다. 편의를 위해 특정 재생 상태 조합의 길이를 반환하는 getTotalPlayTimeMsgetTotalWaitTimeMs와 같은 메서드도 있습니다.

Kotlin

Log.d(
  "DEBUG",
  "Playback summary: " +
    "play time = " +
    playbackStats.totalPlayTimeMs +
    ", rebuffers = " +
    playbackStats.totalRebufferCount
)

자바

Log.d(
    "DEBUG",
    "Playback summary: "
        + "play time = "
        + playbackStats.getTotalPlayTimeMs()
        + ", rebuffers = "
        + playbackStats.totalRebufferCount);

여러 재생의 분석 데이터 집계

PlaybackStats.merge를 호출하여 여러 PlaybackStats를 함께 결합할 수 있습니다. 결과 PlaybackStats에는 병합된 모든 재생의 집계된 데이터가 포함됩니다. 개별 재생 이벤트의 기록은 집계할 수 없으므로 포함되지 않습니다.

PlaybackStatsListener.getCombinedPlaybackStats를 사용하면 PlaybackStatsListener의 전체 기간 동안 수집된 모든 분석 데이터를 집계된 보기로 확인할 수 있습니다.

계산된 요약 측정항목

PlaybackStats는 기본 분석 데이터 외에도 요약 측정항목을 계산하는 여러 메서드를 제공합니다.

Kotlin

Log.d(
  "DEBUG",
  "Additional calculated summary metrics: " +
    "average video bitrate = " +
    playbackStats.meanVideoFormatBitrate +
    ", mean time between rebuffers = " +
    playbackStats.meanTimeBetweenRebuffers
)

자바

Log.d(
    "DEBUG",
    "Additional calculated summary metrics: "
        + "average video bitrate = "
        + playbackStats.getMeanVideoFormatBitrate()
        + ", mean time between rebuffers = "
        + playbackStats.getMeanTimeBetweenRebuffers());

고급 주제

분석 데이터를 재생 메타데이터와 연결

개별 재생에 대한 분석 데이터를 수집할 때 재생 분석 데이터를 재생 중인 미디어에 관한 메타데이터와 연결하는 것이 좋습니다.

MediaItem.Builder.setTag를 사용하여 미디어별 메타데이터를 설정하는 것이 좋습니다. 미디어 태그는 원시 이벤트에 대해 보고되고 PlaybackStats가 완료될 때 보고되는 EventTime의 일부이므로 상응하는 애널리틱스 데이터를 처리할 때 쉽게 검색할 수 있습니다.

Kotlin

PlaybackStatsListener(/* keepHistory= */ false) {
  eventTime: EventTime,
  playbackStats: PlaybackStats ->
  val mediaTag =
    eventTime.timeline
      .getWindow(eventTime.windowIndex, Timeline.Window())
      .mediaItem
      .localConfiguration
      ?.tag
    // Report playbackStats with mediaTag metadata.
}

자바

new PlaybackStatsListener(
    /* keepHistory= */ false,
    (eventTime, playbackStats) -> {
      Object mediaTag =
          eventTime.timeline.getWindow(eventTime.windowIndex, new Timeline.Window())
              .mediaItem
              .localConfiguration
              .tag;
      // Report playbackStats with mediaTag metadata.
    });

맞춤 분석 이벤트 보고

분석 데이터에 맞춤 이벤트를 추가해야 하는 경우 이러한 이벤트를 자체 데이터 구조에 저장하고 나중에 보고된 PlaybackStats와 결합해야 합니다. 도움이 된다면 다음 예와 같이 맞춤 이벤트의 EventTime 인스턴스를 생성하고 이미 등록된 리스너에게 전송할 수 있도록 DefaultAnalyticsCollector를 확장할 수 있습니다.

Kotlin

private interface ExtendedListener : AnalyticsListener {
  fun onCustomEvent(eventTime: EventTime)
}

private class ExtendedCollector : DefaultAnalyticsCollector(Clock.DEFAULT) {
  fun customEvent() {
    val eventTime = generateCurrentPlayerMediaPeriodEventTime()
    sendEvent(eventTime, CUSTOM_EVENT_ID) { listener: AnalyticsListener ->
      if (listener is ExtendedListener) {
        listener.onCustomEvent(eventTime)
      }
    }
  }
}

// Usage - Setup and listener registration.
val player = ExoPlayer.Builder(context).setAnalyticsCollector(ExtendedCollector()).build()
player.addAnalyticsListener(
  object : ExtendedListener {
    override fun onCustomEvent(eventTime: EventTime?) {
      // Save custom event for analytics data.
    }
  }
)
// Usage - Triggering the custom event.
(player.analyticsCollector as ExtendedCollector).customEvent()

Java

private interface ExtendedListener extends AnalyticsListener {
  void onCustomEvent(EventTime eventTime);
}

private static class ExtendedCollector extends DefaultAnalyticsCollector {
  public ExtendedCollector() {
    super(Clock.DEFAULT);
  }

  public void customEvent() {
    AnalyticsListener.EventTime eventTime = generateCurrentPlayerMediaPeriodEventTime();
    sendEvent(
        eventTime,
        CUSTOM_EVENT_ID,
        listener -> {
          if (listener instanceof ExtendedListener) {
            ((ExtendedListener) listener).onCustomEvent(eventTime);
          }
        });
  }
}

// Usage - Setup and listener registration.
ExoPlayer player =
    new ExoPlayer.Builder(context).setAnalyticsCollector(new ExtendedCollector()).build();
player.addAnalyticsListener(
    (ExtendedListener) eventTime -> {
      // Save custom event for analytics data.
    });
// Usage - Triggering the custom event.
((ExtendedCollector) player.getAnalyticsCollector()).customEvent();