Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Colibri msg initiated pcap capture #2072

Draft
wants to merge 7 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions jitsi-media-transform/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,11 @@
<groupId>${project.groupId}</groupId>
<artifactId>jitsi-metaconfig</artifactId>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.module</groupId>
<artifactId>jackson-module-kotlin</artifactId>
<version>${jackson.version}</version>
</dependency>
<dependency>
<groupId>org.jetbrains.kotlin</groupId>
<artifactId>kotlin-reflect</artifactId>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,8 @@ abstract class RtpReceiver :
abstract fun forceMuteAudio(shouldMute: Boolean)

abstract fun forceMuteVideo(shouldMute: Boolean)

abstract fun setPcapRecording(mode: String?, contextId: String?)
}

interface RtpReceiverEventHandler {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ import org.jitsi.nlj.transform.NodeEventVisitor
import org.jitsi.nlj.transform.NodeStatsVisitor
import org.jitsi.nlj.transform.NodeTeardownVisitor
import org.jitsi.nlj.transform.node.ConsumerNode
import org.jitsi.nlj.transform.node.MetadataPcapWriter
import org.jitsi.nlj.transform.node.Node
import org.jitsi.nlj.transform.node.PacketLossConfig
import org.jitsi.nlj.transform.node.PacketLossNode
Expand Down Expand Up @@ -140,6 +141,7 @@ class RtpReceiverImpl @JvmOverloads constructor(
})
}
private val toggleablePcapWriter = ToggleablePcapWriter(logger, "$id-rx")
private val metadataPcapWriter = MetadataPcapWriter(logger, streamInformationStore, id, "ne")
private val videoBitrateCalculator = VideoBitrateCalculator(parentLogger)
private val audioBitrateCalculator = BitrateCalculator("Audio bitrate calculator")

Expand Down Expand Up @@ -225,6 +227,7 @@ class RtpReceiverImpl @JvmOverloads constructor(
// This reads audio levels from packets that use cryptex. TODO: should it go in the Audio path?
node(audioLevelReader.postDecryptNode)
node(toggleablePcapWriter.newObserverNode())
node(metadataPcapWriter.newObserverNode())
node(statsTracker)
node(PaddingTermination(logger))
demux("Media Type") {
Expand Down Expand Up @@ -350,9 +353,14 @@ class RtpReceiverImpl @JvmOverloads constructor(
NodeTeardownVisitor().visit(inputTreeRoot)
incomingPacketQueue.close()
toggleablePcapWriter.disable()
metadataPcapWriter.disable()
}

override fun onRttUpdate(newRttMs: Double) {
remoteBandwidthEstimator.onRttUpdate(newRttMs)
}

override fun setPcapRecording(mode: String?, contextId: String?) {
metadataPcapWriter.configure(mode, contextId)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ abstract class RtpSender :
abstract fun setFeature(feature: Features, enabled: Boolean)
abstract fun isFeatureEnabled(feature: Features): Boolean
abstract fun tearDown()
abstract fun setPcapRecording(mode: String?, contextId: String?)

abstract val bandwidthEstimator: BandwidthEstimator
}
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ import org.jitsi.nlj.transform.NodeStatsVisitor
import org.jitsi.nlj.transform.NodeTeardownVisitor
import org.jitsi.nlj.transform.node.AudioRedHandler
import org.jitsi.nlj.transform.node.ConsumerNode
import org.jitsi.nlj.transform.node.MetadataPcapWriter
import org.jitsi.nlj.transform.node.Node
import org.jitsi.nlj.transform.node.PacketCacher
import org.jitsi.nlj.transform.node.PacketLossConfig
Expand Down Expand Up @@ -108,6 +109,7 @@ class RtpSenderImpl(
private val srtpEncryptWrapper = SrtpEncryptNode()
private val srtcpEncryptWrapper = SrtcpEncryptNode()
private val toggleablePcapWriter = ToggleablePcapWriter(logger, "$id-tx")
private val metadataPcapWriter = MetadataPcapWriter(logger, streamInformationStore, id, "fe")
private val outgoingPacketCache = PacketCacher()
private val absSendTime = AbsSendTime(streamInformationStore)
private val statsTracker = OutgoingStatisticsTracker()
Expand Down Expand Up @@ -148,6 +150,7 @@ class RtpSenderImpl(
node(TccSeqNumTagger(transportCcEngine, streamInformationStore))
node(HeaderExtEncoder(streamInformationStore, logger))
node(toggleablePcapWriter.newObserverNode())
node(metadataPcapWriter.newObserverNode())
node(srtpEncryptWrapper)
node(packetStreamStats.createNewNode())
node(PacketLossNode(packetLossConfig), condition = { packetLossConfig.enabled })
Expand Down Expand Up @@ -321,6 +324,11 @@ class RtpSenderImpl(
NodeTeardownVisitor().reverseVisit(outputPipelineTerminationNode)
incomingPacketQueue.close()
toggleablePcapWriter.disable()
metadataPcapWriter.disable()
}

override fun setPcapRecording(mode: String?, contextId: String?) {
metadataPcapWriter.configure(mode, contextId)
}

companion object {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -379,6 +379,11 @@ class Transceiver(
return rtpReceiver.isFeatureEnabled(feature)
}

fun setPcapRecording(mode: String?, contextId: String?) {
rtpReceiver.setPcapRecording(mode, contextId)
rtpSender.setPcapRecording(mode, contextId)
}

companion object {
init {
// Node.plugins.add(BufferTracePlugin)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,265 @@
/*
* Copyright @ 2018 - Present, 8x8 Inc
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jitsi.nlj.transform.node

import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
import org.jitsi.config.JitsiConfig
import org.jitsi.metaconfig.config
import org.jitsi.metaconfig.from
import org.jitsi.nlj.PacketInfo
import org.jitsi.nlj.rtp.AudioRtpPacket
import org.jitsi.nlj.rtp.VideoRtpPacket
import org.jitsi.nlj.util.ReadOnlyStreamInformationStore
import org.jitsi.utils.MediaType
import org.jitsi.utils.logging2.Logger
import org.json.simple.JSONArray
import org.json.simple.JSONObject
import java.io.File
import java.nio.file.Files
import java.nio.file.Paths

class MetadataPcapWriter(
private val logger: Logger,
private val streamInformationStore: ReadOnlyStreamInformationStore,
private val id: String,
private val captureEnd: String
) {
private var writer: PcapWriter? = null
private val lock = Any()
private var contextId: String? = null
private var mode: String = CAP_MODE_NONE

private var captureAudio = false
private var captureVideo = false

fun capId(): String {
return "${id}_$captureEnd"
}

fun filename(): String {
return "$basePath/${contextId}_${capId()}"
}

fun pcapFilename(): String {
return "${filename()}.pcap"
}

fun jsonFilename(): String {
return "${filename()}.json"
}

fun validJmtConfig(): Boolean {
if (!allowed) {
logger.info("${capId()} is not allowed in jmt.metadata-pcap-recording.enabled")
return false
}

if (basePath.isEmpty()) {
logger.error("${capId()} jmt.metadata-pcap-recording.base-path is not configured")
return false
}

if (!Files.isDirectory(Paths.get(basePath))) {
logger.error("${capId()} base-path:$basePath is not a valid directory path")
return false
}

return true
}

fun writeMetadata(): Boolean {
val meta = JSONObject()

meta.put("endpoint_id", id)
meta.put("context_id", contextId)
meta.put("capture_end", captureEnd)
meta.put("capture_mode", mode)

val payloadsMap = JSONArray()
streamInformationStore.rtpPayloadTypes.forEach {
if ((it.value.mediaType == MediaType.AUDIO && (mode == CAP_MODE_AUDIO || mode == CAP_MODE_AUDIO_VIDEO)) ||
(it.value.mediaType == MediaType.VIDEO && (mode == CAP_MODE_VIDEO || mode == CAP_MODE_AUDIO_VIDEO))
) {
val entry = JSONObject()
val encoding = it.value.encoding.toString().lowercase()
entry["payload_type"] = it.value.pt
entry["encoding"] = encoding
entry["media_type"] = it.value.mediaType.name.lowercase()
entry["clock_rate"] = it.value.clockRate
if (encoding == "rtx") {
val apt = it.value.parameters["apt"]
entry["associated_pt"] = apt?.toInt() ?: -1
if (apt == null) {
logger.error("${capId()} no associated payload type for rtx:${it.value.pt}")
}
}

payloadsMap.add(entry)
}
}

meta.put("payload_map", payloadsMap)

try {
// jackson pretty printing
val writer = jacksonObjectMapper().writerWithDefaultPrettyPrinter()
File(jsonFilename()).writeText(writer.writeValueAsString(meta))
} catch (ex: Exception) {
logger.error("${capId()} exception: $ex")
return false
}

return true
}

fun configure(newMode: String?, newContextId: String?) {
// TODO - how to trace long line exceeding 120 chars
logger.info("${capId()} configure mode:$mode -> $newMode")
logger.info("${capId()} contextId:$contextId -> $newContextId")

// Disable conditions
if (newMode.isNullOrEmpty() || newMode == CAP_MODE_NONE) {
disable()
synchronized(lock) {
mode = "none"
contextId = null
}
return
}

// Checks for valid enable conditions
if (!validJmtConfig()) {
return
}

if (newMode !in listOf(CAP_MODE_AUDIO, CAP_MODE_VIDEO, CAP_MODE_AUDIO_VIDEO)) {
logger.error("${capId()} invalid mode:$newMode")
return
}

if (newContextId.isNullOrEmpty()) {
logger.error("${capId()} invalid context-id:$newContextId")
return
}

synchronized(lock) {
if (mode == newMode && contextId == newContextId) {
// nothing to do
logger.warn("${capId()} duplicate configuration request ignored.")
return
}
}

if (isEnabled()) {
// reconfiguring to different mode on the fly?
logger.warn("${capId()} is already enabled - resetting by disabling first.")
disable()
}

synchronized(lock) {
mode = newMode
contextId = newContextId
if (!writeMetadata()) {
contextId = null
mode = "none"
return
}
}

// Enable / open the writer with the new mode and context-id
enable()
}

fun enable() {
logger.info("${capId()} enable ${pcapFilename()}")

if (!validJmtConfig()) {
return
}

if (contextId.isNullOrEmpty()) {
logger.error("${capId()} is not configured with context-id")
return
}

synchronized(lock) {
captureAudio = false
captureVideo = false
when (mode) {
CAP_MODE_AUDIO -> captureAudio = true
CAP_MODE_VIDEO -> captureVideo = true
CAP_MODE_AUDIO_VIDEO -> {
captureAudio = true
captureVideo = true
}
else -> {
logger.error("${capId()} invalid mode:$mode")
return
}
}

if (writer == null) {
logger.info("${capId()} enable ${pcapFilename()}")
writer = PcapWriter(logger, pcapFilename())
}
}
}

fun disable() {
synchronized(lock) {
captureAudio = false
captureVideo = false

if (writer != null) {
logger.info("${capId()} disable ${pcapFilename()}")
writer?.close()
writer = null
}
}
}

fun isEnabled(): Boolean = writer != null

fun newObserverNode(): Node = PcapWriterNode("${capId()} metadata pcap writer")

private inner class PcapWriterNode(name: String) : ObserverNode(name) {
override fun observe(packetInfo: PacketInfo) {
synchronized(lock) {
if (packetInfo.packet is AudioRtpPacket) {
if (captureAudio) {
writer?.processPacket(packetInfo)
}
} else if (packetInfo.packet is VideoRtpPacket) {
if (captureVideo) {
writer?.processPacket(packetInfo)
}
}
}
}

override fun trace(f: () -> Unit) = f.invoke()
}

companion object {
private val allowed: Boolean by config("jmt.metadata-pcap-recording.enabled".from(JitsiConfig.newConfig))
private val basePath: String by config("jmt.metadata-pcap-recording.base-path".from(JitsiConfig.newConfig))

private const val CAP_MODE_NONE = "none"
private const val CAP_MODE_AUDIO = "audio"
private const val CAP_MODE_VIDEO = "video"
private const val CAP_MODE_AUDIO_VIDEO = "audio-video"
}
}
7 changes: 7 additions & 0 deletions jitsi-media-transform/src/main/resources/reference.conf
Original file line number Diff line number Diff line change
Expand Up @@ -160,4 +160,11 @@ jmt {
}
}
}

metadata-pcap-recording {
enabled=true
// Base path for storing the pcap and metadata files
base-path="/tmp/jvb-rec"
}

}
Loading
Loading